rubyfox-server 2.17.3.2 → 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 -263
  32. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.properties +209 -207
  33. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/context.xml +31 -31
  34. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xml +23 -23
  35. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xsd +52 -52
  36. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/keystore.jks +0 -0
  37. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/server.xml +177 -161
  38. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xml +18 -7
  39. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xsd +59 -59
  40. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/web.xml +4740 -4737
  41. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/annotations-api.jar +0 -0
  42. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ant.jar +0 -0
  43. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ha.jar +0 -0
  44. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ssi.jar +0 -0
  45. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-storeconfig.jar +0 -0
  46. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-tribes.jar +0 -0
  47. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina.jar +0 -0
  48. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/el-api.jar +0 -0
  49. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper-el.jar +0 -0
  50. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper.jar +0 -0
  51. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jaspic-api.jar +0 -0
  52. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jsp-api.jar +0 -0
  53. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/servlet-api.jar +0 -0
  54. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/sfs2x-ws-helper.jar +0 -0
  55. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-api.jar +0 -0
  56. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-coyote.jar +0 -0
  57. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-dbcp.jar +0 -0
  58. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-cs.jar +0 -0
  59. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-de.jar +0 -0
  60. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-es.jar +0 -0
  61. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-fr.jar +0 -0
  62. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ja.jar +0 -0
  63. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ko.jar +0 -0
  64. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-pt-BR.jar +0 -0
  65. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ru.jar +0 -0
  66. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-zh-CN.jar +0 -0
  67. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jdbc.jar +0 -0
  68. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jni.jar +0 -0
  69. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util-scan.jar +0 -0
  70. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util.jar +0 -0
  71. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-websocket.jar +0 -0
  72. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/websocket-api.jar +0 -0
  73. data/lib/rubyfox/server/data/lib/javax.activation-1.2.0.jar +0 -0
  74. data/lib/rubyfox/server/data/lib/javax.mail.jar +0 -0
  75. data/lib/rubyfox/server/data/lib/js/JSApi.js +2 -1
  76. data/lib/rubyfox/server/data/lib/js/LibApi.js +181 -48
  77. data/lib/rubyfox/server/data/lib/sfs2x-admin.jar +0 -0
  78. data/lib/rubyfox/server/data/lib/sfs2x-cluster.jar +0 -0
  79. data/lib/rubyfox/server/data/lib/sfs2x-core.jar +0 -0
  80. data/lib/rubyfox/server/data/lib/sfs2x.jar +0 -0
  81. data/lib/rubyfox/server/data/sfs2x-service +26 -30
  82. data/lib/rubyfox/server/data/www/BlueBox.war +0 -0
  83. data/lib/rubyfox/server/data/www/HelloServlet/WEB-INF/web.xml +1 -3
  84. data/lib/rubyfox/server/data/www/ROOT/_css_/default.css +14 -6
  85. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/style.css +44 -2
  86. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/application.bundle.js +98 -61
  87. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/endors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  88. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-0.bundle.js +4 -4
  89. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-1.bundle.js +3 -3
  90. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-10.bundle.js +101 -66
  91. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-11.bundle.js +544 -8
  92. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12.bundle.js +915 -1480
  93. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-15~module-16~module-4.bundle.js +2665 -0
  94. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13.bundle.js +606 -3093
  95. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13~module-16~module-17~module-4.bundle.js +2665 -0
  96. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-14.bundle.js +764 -0
  97. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-15.bundle.js +71 -0
  98. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-16.bundle.js +1787 -0
  99. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-17.bundle.js +3383 -0
  100. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-4.bundle.js +121 -1009
  101. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-5.bundle.js +1214 -1744
  102. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-6.bundle.js +398 -666
  103. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-7.bundle.js +717 -192
  104. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-8.bundle.js +2117 -665
  105. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-9.bundle.js +613 -690
  106. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-10~mod-11~mod-16~mod-5~mod-6~mod-7~mod-8.bundle.js +17357 -0
  107. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  108. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/{vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8.bundle.js → vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-7~mod-8~mod-9.bundle.js} +2 -2
  109. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-12.bundle.js +807 -0
  110. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-13.bundle.js +807 -0
  111. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-configurator.html +32 -0
  112. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-monitor.html +185 -0
  113. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-updater.html +47 -0
  114. data/lib/rubyfox/server/data/www/ROOT/admin/modules/extension-deployer.html +84 -0
  115. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-monitor.html +15 -8
  116. data/lib/rubyfox/server/data/www/ROOT/index.html +13 -23
  117. data/lib/rubyfox/server/data/www/host-manager/META-INF/context.xml +2 -2
  118. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/jsp/404.jsp +2 -2
  119. data/lib/rubyfox/server/data/www/host-manager/{manager.xml → WEB-INF/manager.xml} +5 -1
  120. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/web.xml +17 -0
  121. data/lib/rubyfox/server/data/www/host-manager/css/manager.css +141 -0
  122. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.svg +967 -0
  123. data/lib/rubyfox/server/data/www/manager/META-INF/context.xml +2 -0
  124. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCerts.jsp +1 -1
  125. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCiphers.jsp +1 -1
  126. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorTrustedCerts.jsp +1 -1
  127. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionDetail.jsp +3 -3
  128. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionsList.jsp +1 -1
  129. data/lib/rubyfox/server/data/www/manager/WEB-INF/web.xml +17 -0
  130. data/lib/rubyfox/server/data/www/manager/css/manager.css +141 -0
  131. data/lib/rubyfox/server/data/www/manager/images/tomcat.svg +967 -0
  132. data/lib/rubyfox/server/data/www/manager/xform.xsl +74 -59
  133. data/lib/rubyfox/server/version.rb +1 -1
  134. metadata +30 -31
  135. data/lib/rubyfox/server/data/config/admin/icons/Analytics.png +0 -0
  136. data/lib/rubyfox/server/data/config/admin/icons/BanManager.png +0 -0
  137. data/lib/rubyfox/server/data/config/admin/icons/BlueBoxMonitor.png +0 -0
  138. data/lib/rubyfox/server/data/config/admin/icons/Console.png +0 -0
  139. data/lib/rubyfox/server/data/config/admin/icons/Dashboard.png +0 -0
  140. data/lib/rubyfox/server/data/config/admin/icons/ExtensionManager.png +0 -0
  141. data/lib/rubyfox/server/data/config/admin/icons/LicenseManager.png +0 -0
  142. data/lib/rubyfox/server/data/config/admin/icons/LogViewer.png +0 -0
  143. data/lib/rubyfox/server/data/config/admin/icons/ServerConfigurator.png +0 -0
  144. data/lib/rubyfox/server/data/config/admin/icons/ServletManager.png +0 -0
  145. data/lib/rubyfox/server/data/config/admin/icons/ZoneConfigurator.png +0 -0
  146. data/lib/rubyfox/server/data/config/admin/icons/ZoneMonitor.png +0 -0
  147. data/lib/rubyfox/server/data/lib/BlueBox.war +0 -0
  148. data/lib/rubyfox/server/data/lib/apache-tomcat/LICENSE +0 -1061
  149. data/lib/rubyfox/server/data/lib/apache-tomcat/NOTICE +0 -68
  150. data/lib/rubyfox/server/data/lib/apache-tomcat/README.md +0 -81
  151. data/lib/rubyfox/server/data/lib/apache-tomcat/RELEASE-NOTES +0 -174
  152. data/lib/rubyfox/server/data/lib/imap.jar +0 -0
  153. data/lib/rubyfox/server/data/lib/mailapi.jar +0 -0
  154. data/lib/rubyfox/server/data/lib/pop3.jar +0 -0
  155. data/lib/rubyfox/server/data/lib/smtp.jar +0 -0
  156. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-13~module-9.bundle.js +0 -2634
  157. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-9.bundle.js +0 -807
  158. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.gif +0 -0
  159. data/lib/rubyfox/server/data/www/manager/images/tomcat.gif +0 -0
  160. /data/lib/rubyfox/server/data/data/buddylists/{BasicExamples/.keep → .keep} +0 -0
@@ -0,0 +1,1787 @@
1
+ /*! (c) gotoAndPlay | All rights reserved */
2
+ (window["webpackJsonpapplication"] = window["webpackJsonpapplication"] || []).push([["module-16"],{
3
+
4
+ /***/ "./src/components/module-specific/words-files-manager.js":
5
+ /*!***************************************************************!*\
6
+ !*** ./src/components/module-specific/words-files-manager.js ***!
7
+ \***************************************************************/
8
+ /*! exports provided: WordsFilesManager */
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__, "WordsFilesManager", function() { return WordsFilesManager; });
14
+ /* harmony import */ var _utils_utilities__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../utils/utilities */ "./src/utils/utilities.js");
15
+
16
+
17
+ class WordsFilesManager extends HTMLElement
18
+ {
19
+ constructor()
20
+ {
21
+ super();
22
+
23
+ this.REFRESH_WORDS_FILES_CLICK_EVENT = 'refreshWordsFilesClick';
24
+ this.EDIT_WORDS_FILE_CLICK_EVENT = 'editWordsFileClick';
25
+ this.SAVE_WORDS_FILE_CLICK_EVENT = 'saveWordsFileClick';
26
+ this.REMOVE_WORDS_FILE_CLICK_EVENT = 'removeWordsFileClick';
27
+ this.ASSIGN_WORDS_FILE_CLICK_EVENT = 'assingWordsFileClick';
28
+
29
+ this.CONFIG_FOLDER = 'config/';
30
+ this.WORDS_FILE_EXT = '.words.txt';
31
+
32
+ this._modalHtml = `
33
+ <div class="modal" id="editModal" tabindex="-1" role="dialog" aria-labelledby="editModalTitle" aria-hidden="true">
34
+ <div class="modal-dialog modal-dialog-centered" role="document">
35
+ <div class="modal-content">
36
+ <div class="modal-header">
37
+ <h5 class="modal-title text-primary" id="editModalTitle">Word File Editor</h5>
38
+ <button type="button" class="close" data-dismiss="modal" aria-label="Close">
39
+ <span aria-hidden="true">&times;</span>
40
+ </button>
41
+ </div>
42
+ <div class="modal-body in-flow-invalid-msg">
43
+ <fieldset id="editFieldset">
44
+ <div class="form-group">
45
+ <div class="col-form-label form-label-container">
46
+ <label for="filename" class="form-label">Filename <i class="fas fa-question-circle text-muted help" title="Name of the words file. The file will be saved in server's <em>config</em> folder."></i></label>
47
+ </div>
48
+ <div class="inner-widget">
49
+ <div class="input-group">
50
+ <input type="text" id="filename" name="filename" class="form-control k-textbox" autocomplete="off" required data-required-msg="Required" aria-describedby="extension" />
51
+ <div class="input-group-append">
52
+ <span class="input-group-text" id="extension">.words.txt</span>
53
+ </div>
54
+ </div>
55
+ <span class="k-invalid-msg position-static" data-for="filename"></span>
56
+ </div>
57
+ </div>
58
+ <div class="form-group">
59
+ <div class="col-form-label form-label-container">
60
+ <label for="content" class="form-label">Content <i class="fas fa-question-circle text-muted help" title="Enter a word or a valid regular expression per line. See configuration's <em>Words file</em> field description for additional information."></i></label>
61
+ </div>
62
+ <div class="inner-widget">
63
+ <textarea id="content" name="content" class="form-control k-textarea w-100" rows="10"></textarea>
64
+ <span class="k-invalid-msg position-static" data-for="content"></span>
65
+ </div>
66
+ </div>
67
+ </fieldset>
68
+ </div>
69
+ <div class="modal-footer flex-column">
70
+ <div class="d-flex w-100">
71
+ <div class="flex-grow-1 text-left">
72
+ <button id="saveWordFileButton" type="button" class="k-button k-primary"><i class="fas fa-save mr-1"></i>Save word file</button>
73
+ <i id="saveSpinner" class="fas fa-circle-notch fa-spin text-primary align-middle ml-1"></i>
74
+ </div>
75
+ <div class="flex-grow-1 text-right">
76
+ <button type="button" class="k-button k-secondary" data-dismiss="modal">Cancel</button>
77
+ </div>
78
+ </div>
79
+ </div>
80
+ </div>
81
+ </div>
82
+ </div>
83
+ `;
84
+
85
+ //-------------------------------------------
86
+
87
+ $(this).append(`
88
+ <div class="col-sm-5 col-lg-4 col-form-label form-label-container">
89
+ <label class="form-label">Available words files <i class="fas fa-question-circle text-muted help" title="The list of words files found in server's <em>config</em> folder. Click on the Assign button to set the <strong>Words file</strong> field to the selected file. Configuration submission is then required to save the new value."></i></label>
90
+ </div>
91
+ <div class="inner-widget align-self-center col-sm">
92
+ <div>
93
+ <div id="wordsFiles" class="limited-height"></div>
94
+ <div id="actionButtons" class="mt-2 text-right" disabled>
95
+ <i id="actionSpinner" class="fas fa-circle-notch fa-spin text-primary align-middle"></i>
96
+ <button id="refreshButton" type="button" class="k-button k-secondary ml-2" title="Refresh"><i class="fas fa-redo-alt"></i></button>
97
+ <button id="addButton" type="button" class="k-button k-secondary ml-2" title="Add"><i class="fas fa-plus"></i></button>
98
+ <button id="editButton" type="button" class="k-button k-secondary ml-2" title="Edit" disabled><i class="fas fa-pen"></i></button>
99
+ <button id="removeButton" type="button" class="k-button k-secondary ml-2" title="Remove" disabled><i class="fas fa-times"></i></button>
100
+ <button id="assignButton" type="button" class="k-button k-secondary ml-2" title="Assign" disabled><i class="fas fa-compress-arrows-alt"></i></button>
101
+ </div>
102
+ </div>
103
+ </div>
104
+ `);
105
+
106
+ //-------------------------------------------
107
+
108
+ // Initialize grid
109
+ this._wordsFilesGrid = $('#wordsFiles', $(this)).kendoGrid({
110
+ resizable: true,
111
+ selectable: 'row',
112
+ change: $.proxy(this._onWordsFilesGridSelectionChange, this),
113
+ columns: [
114
+ {
115
+ field: 'name',
116
+ title: 'Filename',
117
+ width: 120
118
+ },
119
+ {
120
+ field: 'date',
121
+ title: 'Date',
122
+ width: 80
123
+ },
124
+ {
125
+ field: 'size',
126
+ title: 'Size',
127
+ width: 80
128
+ }
129
+ ],
130
+ noRecords: {
131
+ template: 'No files.'
132
+ }
133
+ }).data('kendoGrid');
134
+
135
+ // Add listeners to button clicks
136
+ $('#refreshButton', $(this)).on('click', $.proxy(this._onReloadClick, this));
137
+ $('#addButton', $(this)).on('click', $.proxy(this._onAddClick, this));
138
+ $('#editButton', $(this)).on('click', $.proxy(this._onEditClick, this));
139
+ $('#removeButton', $(this)).on('click', $.proxy(this._onRemoveClick, this));
140
+ $('#assignButton', $(this)).on('click', $.proxy(this._onAssignClick, this));
141
+ }
142
+
143
+ destroy()
144
+ {
145
+ // Destroy grid
146
+ this._wordsFilesGrid.destroy();
147
+
148
+ // Remove event listeners
149
+ $('#refreshButton', $(this)).off('click');
150
+ $('#addButton', $(this)).off('click');
151
+ $('#editButton', $(this)).off('click');
152
+ $('#removeButton', $(this)).off('click');
153
+ $('#assignButton', $(this)).off('click');
154
+
155
+ // Hide modal (which in turn destroys it)
156
+ let modalElement = $('#editModal', $(this));
157
+
158
+ if (modalElement)
159
+ modalElement.modal('hide');
160
+ }
161
+
162
+ get enabled()
163
+ {
164
+ return this._isEnabled;
165
+ }
166
+
167
+ set enabled(value)
168
+ {
169
+ this._isEnabled = value;
170
+
171
+ // Enable/disable buttons
172
+ $('#actionButtons', this).attr('disabled', !value);
173
+
174
+ // Hide spinner
175
+ if (value)
176
+ this.actionSpinnerVisible = false;
177
+
178
+ // Enable/disable modal
179
+ let modalElement = $('#editModal', $(this));
180
+
181
+ if (modalElement)
182
+ {
183
+ // Disable modal close buttons
184
+ $('button[data-dismiss="modal"]', modalElement).attr('disabled', !value);
185
+
186
+ // Disable save button
187
+ $('#saveWordFileButton', modalElement).attr('disabled', !value);
188
+
189
+ // Disable fieldset
190
+ $('#editFieldset', modalElement).attr('disabled', !value);
191
+
192
+ // Hide spinner
193
+ if (value)
194
+ this.saveSpinnerVisible = false;
195
+ }
196
+ }
197
+
198
+ set actionSpinnerVisible(value)
199
+ {
200
+ if (value)
201
+ $('#actionSpinner', $(this)).show();
202
+ else
203
+ $('#actionSpinner', $(this)).hide();
204
+ }
205
+
206
+ set saveSpinnerVisible(value)
207
+ {
208
+ let modalElement = $('#editModal', $(this));
209
+
210
+ if (modalElement)
211
+ {
212
+ if (value)
213
+ $('#saveSpinner', modalElement).show();
214
+ else
215
+ $('#saveSpinner', modalElement).hide();
216
+ }
217
+ }
218
+
219
+ refreshWordsFilesList(wordsFilesList, hideEditModal)
220
+ {
221
+ if (hideEditModal)
222
+ {
223
+ let modalElement = $('#editModal', $(this));
224
+
225
+ if (modalElement)
226
+ modalElement.modal('hide');
227
+ }
228
+
229
+ let files = [];
230
+ this._existingFilenames = [];
231
+
232
+ for (let f = 0; f < wordsFilesList.size(); f++)
233
+ {
234
+ const file = wordsFilesList.getSFSObject(f);
235
+
236
+ const fileObj = {};
237
+ fileObj.name = file.getUtfString('name');
238
+ fileObj.date = file.getUtfString('date') + ' ' + file.getUtfString('time');
239
+ fileObj.size = Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_0__["bytesToSize"])(file.getLong('size'), 2);
240
+
241
+ // Populate files list
242
+ files.push(fileObj);
243
+
244
+ // Save ref to existing filenames, to check them when a new file is created
245
+ this._existingFilenames.push(fileObj.name);
246
+ }
247
+
248
+ // Assign data source to grid
249
+ this._setWordsFilesGridDataSource(files);
250
+ this._onWordsFilesGridSelectionChange();
251
+
252
+ // Enable
253
+ this.enabled = true;
254
+ }
255
+
256
+ getSelectedWordsFileName()
257
+ {
258
+ if (this._wordsFilesGrid.select() != null)
259
+ {
260
+ let selectedIndex = this._wordsFilesGrid.select().index();
261
+ return this._wordsFilesGrid.dataSource.at(selectedIndex).name;
262
+ }
263
+ else
264
+ return null;
265
+ }
266
+
267
+ editWordsFile(filename, content)
268
+ {
269
+ this._isNewFile = false;
270
+
271
+ this.enabled = true;
272
+
273
+ // Show modal
274
+ this._showModal();
275
+
276
+ // Remove default extension from filename
277
+ filename = filename.substring(0, filename.lastIndexOf(this.WORDS_FILE_EXT));
278
+
279
+ // Enter content filename and content in modal form
280
+ $('#editModal #filename', $(this)).val(filename);
281
+ $('#editModal #content', $(this)).val(content);
282
+
283
+ // Set filename field as not editable and hide note
284
+ $('#editModal #filename', $(this)).attr('disabled', true);
285
+ $('#editModal #filenameNote', $(this)).hide();
286
+ }
287
+
288
+ getExistingFilenames()
289
+ {
290
+ return this._existingFilenames;
291
+ }
292
+
293
+ _setWordsFilesGridDataSource(ds)
294
+ {
295
+ // Read current horizontal scroll value
296
+ const scrollLeft = $('#wordsFiles .k-grid-content', this._wordsFilesGrid.wrapper).scrollLeft();
297
+
298
+ // Assign data source to grid
299
+ this._wordsFilesGrid.setDataSource(ds);
300
+
301
+ // Set horizontal scroll
302
+ $('#wordsFiles .k-grid-content', this._wordsFilesGrid.wrapper).scrollLeft(scrollLeft);
303
+ }
304
+
305
+ _onWordsFilesGridSelectionChange()
306
+ {
307
+ // Enable/disable buttons
308
+ const selectedRows = this._wordsFilesGrid.select();
309
+ $('#editButton').attr('disabled', selectedRows.length == 0);
310
+ $('#removeButton').attr('disabled', selectedRows.length == 0);
311
+ $('#assignButton').attr('disabled', selectedRows.length == 0);
312
+ }
313
+
314
+ _onReloadClick()
315
+ {
316
+ // Fire event to request file content to server
317
+ let evt = new CustomEvent(this.REFRESH_WORDS_FILES_CLICK_EVENT, {
318
+ detail: null,
319
+ bubbles: false,
320
+ cancelable: false
321
+ });
322
+
323
+ this.dispatchEvent(evt);
324
+ }
325
+
326
+ _onAddClick()
327
+ {
328
+ this._isNewFile = true;
329
+
330
+ // Show modal
331
+ this._showModal();
332
+ }
333
+
334
+ _onEditClick()
335
+ {
336
+ // Disable buttons
337
+ this.enabled = false;
338
+ this.actionSpinnerVisible = true;
339
+
340
+ // Fire event to request file content to server
341
+ let evt = new CustomEvent(this.EDIT_WORDS_FILE_CLICK_EVENT, {
342
+ detail: this.getSelectedWordsFileName(),
343
+ bubbles: false,
344
+ cancelable: false
345
+ });
346
+
347
+ this.dispatchEvent(evt);
348
+ }
349
+
350
+ _onRemoveClick()
351
+ {
352
+ // Fire event to request file removal to server
353
+ let evt = new CustomEvent(this.REMOVE_WORDS_FILE_CLICK_EVENT, {
354
+ detail: this.getSelectedWordsFileName(),
355
+ bubbles: false,
356
+ cancelable: false
357
+ });
358
+
359
+ this.dispatchEvent(evt);
360
+ }
361
+
362
+ _onAssignClick()
363
+ {
364
+ // Fire event to substitute path in configuration
365
+ let evt = new CustomEvent(this.ASSIGN_WORDS_FILE_CLICK_EVENT, {
366
+ detail: this.CONFIG_FOLDER + this.getSelectedWordsFileName(),
367
+ bubbles: false,
368
+ cancelable: false
369
+ });
370
+
371
+ this.dispatchEvent(evt);
372
+ }
373
+
374
+ _onSaveWordFileClick()
375
+ {
376
+ if (this._validator.validate())
377
+ {
378
+ // Show spinner
379
+ $('#editModal #saveSpinner', $(this)).show();
380
+
381
+ // Fire event to request file to be saved by server
382
+ let evt = new CustomEvent(this.SAVE_WORDS_FILE_CLICK_EVENT, {
383
+ detail: {
384
+ filename: $('#editModal #filename', $(this)).val() + this.WORDS_FILE_EXT,
385
+ isNew: this._isNewFile,
386
+ content: $('#editModal #content', $(this)).val()
387
+ },
388
+ bubbles: false,
389
+ cancelable: false
390
+ });
391
+
392
+ this.dispatchEvent(evt);
393
+ }
394
+ }
395
+
396
+ _showModal()
397
+ {
398
+ // Append modal html
399
+ $(this).append(this._modalHtml);
400
+
401
+ let modalElement = $('#editModal', $(this));
402
+
403
+ // Initialize kendo validation
404
+ this._validator = modalElement.find('#editFieldset').kendoValidator({
405
+ validateOnBlur: true,
406
+ rules: {
407
+ requiredFilename: $.proxy(function(input) {
408
+ let valid = true;
409
+ if (input.is('[name=filename]'))
410
+ valid = input.val() !== '';
411
+ return valid;
412
+ }, this),
413
+ validFilename: $.proxy(function(input) {
414
+ let valid = true;
415
+ if (input.is('[name=filename]'))
416
+ valid = (!input.val().includes('\\') && !input.val().includes('/'));
417
+ return valid;
418
+ }, this)
419
+ },
420
+ messages: {
421
+ requiredFilename: 'Required',
422
+ validFilename: 'Contains invalid characters (slash, backslash)'
423
+ }
424
+ }).data('kendoValidator');
425
+
426
+ // Hide save spinner
427
+ $('#saveSpinner', modalElement).hide();
428
+
429
+ // Add listener to Save button click
430
+ $('#saveWordFileButton', modalElement).on('click', $.proxy(this._onSaveWordFileClick, this));
431
+
432
+ // Add listener to modal hide event
433
+ modalElement.on('hidden.bs.modal', $.proxy(this._destroyModal, this));
434
+
435
+ // Initialize bootstrap modal
436
+ modalElement.modal({
437
+ backdrop: 'static',
438
+ keyboard: false,
439
+ });
440
+ }
441
+
442
+ _destroyModal()
443
+ {
444
+ let modalElement = $('#editModal', $(this));
445
+
446
+ if (modalElement)
447
+ {
448
+ // Remove listeners
449
+ $('#saveWordFileButton', modalElement).off('click');
450
+ modalElement.off('hidden.bs.modal');
451
+
452
+ // Destroy everything Kendo
453
+ kendo.destroy(modalElement);
454
+
455
+ // Dispose modal
456
+ modalElement.modal('dispose');
457
+
458
+ // Remove html
459
+ modalElement.remove();
460
+ modalElement = null;
461
+ }
462
+ }
463
+ }
464
+
465
+ // DEFINE COMPONENT
466
+ if (!window.customElements.get('words-files-manager'))
467
+ window.customElements.define('words-files-manager', WordsFilesManager);
468
+
469
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
470
+
471
+ /***/ }),
472
+
473
+ /***/ "./src/components/sidebar-layout.js":
474
+ /*!******************************************!*\
475
+ !*** ./src/components/sidebar-layout.js ***!
476
+ \******************************************/
477
+ /*! exports provided: SidebarLayout */
478
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
479
+
480
+ "use strict";
481
+ __webpack_require__.r(__webpack_exports__);
482
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "SidebarLayout", function() { return SidebarLayout; });
483
+ class SidebarLayout extends HTMLElement
484
+ {
485
+ constructor()
486
+ {
487
+ super();
488
+
489
+ // Attach a shadow root
490
+ const shadowRoot = this.attachShadow({mode: 'open'});
491
+ shadowRoot.innerHTML = `
492
+ <style>
493
+ :host {
494
+ display: flex;
495
+ flex-direction: row;
496
+ }
497
+
498
+ @media (max-width: 575.98px) {
499
+ :host(.split-xs) ::slotted(:not([aria-selected="true"])) {
500
+ display: none !important;
501
+ }
502
+
503
+ :host(.split-xs) ::slotted([aria-selected="true"]) {
504
+ flex-grow: 1;
505
+ }
506
+ }
507
+
508
+ @media (max-width: 767.98px) {
509
+ :host(.split-sm) ::slotted(:not([aria-selected="true"])) {
510
+ display: none !important;
511
+ }
512
+
513
+ :host(.split-sm) ::slotted([aria-selected="true"]) {
514
+ flex-grow: 1;
515
+ }
516
+ }
517
+
518
+ @media (max-width: 991.98px) {
519
+ :host(.split-md) ::slotted(:not([aria-selected="true"])) {
520
+ display: none !important;
521
+ }
522
+
523
+ :host(.split-md) ::slotted([aria-selected="true"]) {
524
+ flex-grow: 1;
525
+ }
526
+ }
527
+
528
+ @media (max-width: 1199.98px) {
529
+ :host(.split-lg) ::slotted(:not([aria-selected="true"])) {
530
+ display: none !important;
531
+ }
532
+
533
+ :host(.split-lg) ::slotted([aria-selected="true"]) {
534
+ flex-grow: 1;
535
+ }
536
+ }
537
+
538
+ .side-col::slotted(*) {
539
+ }
540
+
541
+ .main-col::slotted(*) {
542
+ flex-grow: 1;
543
+ }
544
+ </style>
545
+
546
+ <slot class="side-col" name="side-column"></slot>
547
+ <slot class="main-col" name="main-column"></slot>
548
+ `;
549
+
550
+ // Set initial selection
551
+ this.selectedIndex = 0;
552
+ }
553
+
554
+ get selectedPanel()
555
+ {
556
+ return this._selectedPanel;
557
+ }
558
+
559
+ set selectedPanel(element) // 'side' or 'main'
560
+ {
561
+ if (element != null && element.parentNode == this)
562
+ {
563
+ this._selectedPanel = element;
564
+
565
+ for (let element of this.children)
566
+ {
567
+ if (element == this._selectedPanel)
568
+ element.setAttribute('aria-selected', 'true');
569
+ else
570
+ element.removeAttribute('aria-selected');
571
+ }
572
+ }
573
+ else
574
+ {
575
+ console.error('Element is not a child of SidebarLayout');
576
+ }
577
+ }
578
+
579
+ get selectedIndex()
580
+ {
581
+ return Array.from(this.children).indexOf(this._selectedPanel);
582
+ }
583
+
584
+ set selectedIndex(index)
585
+ {
586
+ if (this.children.length > 0)
587
+ {
588
+ if (this.children[index] == null)
589
+ {
590
+ console.error('Invalid SidebarLayout index');
591
+ return;
592
+ }
593
+
594
+ let element = this.children[index];
595
+ this.selectedPanel = element;
596
+ }
597
+ }
598
+ }
599
+
600
+ // DEFINE COMPONENT
601
+ if (!window.customElements.get('sidebar-layout'))
602
+ window.customElements.define('sidebar-layout', SidebarLayout);
603
+
604
+
605
+ /***/ }),
606
+
607
+ /***/ "./src/modules/zone-configurator.js":
608
+ /*!******************************************!*\
609
+ !*** ./src/modules/zone-configurator.js ***!
610
+ \******************************************/
611
+ /*! exports provided: default */
612
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
613
+
614
+ "use strict";
615
+ __webpack_require__.r(__webpack_exports__);
616
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return ZoneConfigurator; });
617
+ /* harmony import */ var _base_module__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-module */ "./src/modules/base-module.js");
618
+ /* harmony import */ var _components_view_stack__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../components/view-stack */ "./src/components/view-stack.js");
619
+ /* harmony import */ var _components_sidebar_layout__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../components/sidebar-layout */ "./src/components/sidebar-layout.js");
620
+ /* harmony import */ var _utils_uibuilder_config_interface_builder__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../utils/uibuilder/config-interface-builder */ "./src/utils/uibuilder/config-interface-builder.js");
621
+ /* harmony import */ var _utils_utilities__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../utils/utilities */ "./src/utils/utilities.js");
622
+ /* harmony import */ var _components_module_specific_words_files_manager__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../components/module-specific/words-files-manager */ "./src/components/module-specific/words-files-manager.js");
623
+
624
+
625
+
626
+
627
+
628
+
629
+
630
+ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"]
631
+ {
632
+ constructor()
633
+ {
634
+ super('zoneConfig');
635
+
636
+ this.ITEM_TYPE_ZONE = 'zone';
637
+ this.ITEM_TYPE_ROOM = 'room';
638
+
639
+ // Outgoing requests
640
+ this.REQ_GET_ZONES = 'getZones';
641
+
642
+ this.REQ_GET_ZONE_CONFIG = 'getZoneConfig';
643
+ this.REQ_SAVE_ZONE_CONFIG = 'saveZoneConfig';
644
+ this.REQ_NEW_ZONE_CONFIG = 'newZoneConfig';
645
+ this.REQ_DELETE_ZONE_CONFIG = 'delZoneConfig';
646
+ this.REQ_ACTIVATE_ZONE = 'actZone';
647
+
648
+ this.REQ_GET_ROOM_CONFIG = 'getRoomConfig';
649
+ this.REQ_SAVE_ROOM_CONFIG = 'saveRoomConfig';
650
+ this.REQ_NEW_ROOM_CONFIG = 'newRoomConfig';
651
+ this.REQ_DELETE_ROOM_CONFIG = 'delRoomConfig';
652
+
653
+ this.REQ_REFRESH_WORDS_FILE = 'refreshWordsFiles';
654
+ this.REQ_EDIT_WORDS_FILE = 'editWordsFile';
655
+ this.REQ_SAVE_WORDS_FILE = 'saveWordsFile';
656
+ this.REQ_DELETE_WORDS_FILE = 'delWordsFile';
657
+
658
+ // Incoming responses
659
+ this.RESP_ZONES = 'zones';
660
+
661
+ this.RESP_ZONE_CONFIG = 'zoneConfig';
662
+ this.RESP_ZONE_CONFIG_UPDATE_CONFIRM = 'zoneCfgUpd';
663
+ this.RESP_ZONE_ADDED = 'zoneAdded';
664
+ this.RESP_ZONE_REFUSED = 'zoneRefused';
665
+ this.RESP_ZONE_DELETED = 'zoneDel';
666
+ this.RESP_ZONE_ACTIVATED = 'zoneAct';
667
+ this.RESP_ZONE_ACTIVATION_ERROR = 'zoneActErr';
668
+
669
+ this.RESP_ROOM_CONFIG = 'roomConfig';
670
+ this.RESP_ROOM_CONFIG_UPDATE_CONFIRM = 'roomCfgUpd';
671
+ this.RESP_ROOM_ADDED = 'roomAdded';
672
+ this.RESP_ROOM_REFUSED = 'roomRefused';
673
+ this.RESP_ROOM_DELETED = 'roomDel';
674
+
675
+ this.RESP_REFRESH_WORDS_FILES = 'refreshWordsFiles';
676
+ this.RESP_WORDS_FILE_CONTENT = 'wordsFile';
677
+ this.RESP_WORDS_FILE_ERROR = 'wordsFileErr';
678
+ }
679
+
680
+ //------------------------------------
681
+ // COMMON MODULE INTERFACE METHODS
682
+ // This members are used by the main controller
683
+ // to communicate with the module's controller.
684
+ // This methods override those in BaseModule class.
685
+ //------------------------------------
686
+
687
+ initialize(idData, shellController)
688
+ {
689
+ // Call super method
690
+ super.initialize(idData, shellController);
691
+
692
+ // Create interface builder instance
693
+ this._interfaceBuilder = new _utils_uibuilder_config_interface_builder__WEBPACK_IMPORTED_MODULE_3__["ConfigInterfaceBuilder"]();
694
+
695
+ // Set listener for custom actions triggered by configuration interface
696
+ $('#znc-tabNavigator').on('value-set', $.proxy(this._onConfigValueSet, this));
697
+
698
+ // Initialize Zones/Rooms treeview
699
+ this._treeview = $('#znc-treeView').kendoTreeView({
700
+ loadOnDemand: false,
701
+ dataTextField: 'name',
702
+ template: kendo.template('<span class="# if (!item.active) { # inactive-list-item # } #">#: item.name #</span>'),
703
+ change: $.proxy(this._onZoneRoomChange, this),
704
+ }).data('kendoTreeView');
705
+
706
+ // Listen to treeview double-click event
707
+ $('#znc-treeView').on('dblclick', $.proxy(this._onTreeItemDoubleClick, this));
708
+
709
+ // Request zones & rooms list to server instance
710
+ this.sendExtensionRequest(this.REQ_GET_ZONES);
711
+
712
+ // Initialize progress bar
713
+ $('#znc-progressBar').kendoProgressBar({
714
+ min: 0,
715
+ max: 100,
716
+ value: false,
717
+ type: 'value',
718
+ animation: {
719
+ duration: 400
720
+ }
721
+ });
722
+
723
+ // Add listeners to utility buttons
724
+ $('#znc-addZoneButton').on('click', $.proxy(this._onAddZoneClick, this));
725
+ $('#znc-addRoomButton').on('click', $.proxy(this._onAddRoomClick, this));
726
+ $('#znc-editButton').on('click', $.proxy(this._onEditClick, this));
727
+ $('#znc-removeButton').on('click', $.proxy(this._onRemoveClick, this));
728
+ $('#znc-activateButton').on('click', $.proxy(this._onActivateClick, this));
729
+
730
+ // Add listener to interface buttons
731
+ $('#znc-cancelButton').on('click', $.proxy(this._onCancelClick, this));
732
+ $('#znc-reloadButton').on('click', $.proxy(this._onReloadClick, this));
733
+ $('#znc-submitButton').on('click', $.proxy(this._onSubmitClick, this));
734
+
735
+ // Save ref to words files manager and add custom event listeners
736
+ this._wordsFilesManager = document.getElementById('znc-wordsFilesManager');
737
+ $(this._wordsFilesManager).on(this._wordsFilesManager.REFRESH_WORDS_FILES_CLICK_EVENT, $.proxy(this._onWordsFileReloadClick, this));
738
+ $(this._wordsFilesManager).on(this._wordsFilesManager.EDIT_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileEditClick, this));
739
+ $(this._wordsFilesManager).on(this._wordsFilesManager.SAVE_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileSaveClick, this));
740
+ $(this._wordsFilesManager).on(this._wordsFilesManager.REMOVE_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileRemoveClick, this));
741
+ $(this._wordsFilesManager).on(this._wordsFilesManager.ASSIGN_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileAssignClick, this));
742
+ }
743
+
744
+ destroy()
745
+ {
746
+ // Call super method
747
+ super.destroy();
748
+
749
+ // Remove tree view doubleclick listener
750
+ $('#znc-treeView').off('dblclick');
751
+
752
+ // Remove listener for custom actions triggered by configuration interface
753
+ $('#znc-tabNavigator').off('value-set');
754
+
755
+ // Remove listener for zone/room activation event
756
+ $('#znc-addZoneButton').off('click');
757
+ $('#znc-addRoomButton').off('click');
758
+ $('#znc-editButton').off('click');
759
+ $('#znc-removeButton').off('click');
760
+ $('#znc-activateButton').off('click');
761
+
762
+ // Remove interface buttons click listeners
763
+ $('#znc-cancelButton').off('click');
764
+ $('#znc-reloadButton').off('click');
765
+ $('#znc-submitButton').off('click');
766
+
767
+ // Remove listeners to words files manager
768
+ $(this._wordsFilesManager).off(this._wordsFilesManager.REFRESH_WORDS_FILES_CLICK_EVENT);
769
+ $(this._wordsFilesManager).off(this._wordsFilesManager.EDIT_WORDS_FILE_CLICK_EVENT);
770
+ $(this._wordsFilesManager).off(this._wordsFilesManager.SAVE_WORDS_FILE_CLICK_EVENT);
771
+ $(this._wordsFilesManager).off(this._wordsFilesManager.REMOVE_WORDS_FILE_CLICK_EVENT);
772
+ $(this._wordsFilesManager).off(this._wordsFilesManager.ASSIGN_WORDS_FILE_CLICK_EVENT);
773
+
774
+ // Clear tabs container
775
+ this._clearTabs();
776
+ }
777
+
778
+ onExtensionCommand(command, data)
779
+ {
780
+ const username = data.getUtfString('user');
781
+
782
+ /****** ZONES & ROOMS ******/
783
+
784
+ // Zones & rooms list received
785
+ if (command == this.RESP_ZONES)
786
+ this._populateTree(data);
787
+
788
+ // Zone or room configuration data received
789
+ else if (command == this.RESP_ZONE_CONFIG || command == this.RESP_ROOM_CONFIG)
790
+ {
791
+ // Build user interface based on received data
792
+ this._interfaceBuilder.buildInterface(data.getSFSArray('settings'), 'znc-tabNavigator', false);
793
+
794
+ // Enable scrolling tabs (if needed)
795
+ if (this._reinitTabs)
796
+ {
797
+ $('#znc-tabNavigator #tabs').scrollingTabs({
798
+ bootstrapVersion: 4,
799
+ scrollToTabEdge: true,
800
+ enableSwiping: true,
801
+ disableScrollArrowsOnFullyScrolled: true,
802
+ cssClassLeftArrow: 'fa fa-chevron-left',
803
+ cssClassRightArrow: 'fa fa-chevron-right'
804
+ });
805
+ }
806
+
807
+ // Enable interface
808
+ this._enableConfigInterface(true);
809
+ }
810
+
811
+ /****** ZONES ******/
812
+
813
+ // Zone configuration update confirmation
814
+ else if (command == this.RESP_ZONE_CONFIG_UPDATE_CONFIRM)
815
+ {
816
+ // If a 'name' parameter is received, it means the zone name changed, and we have to update the zones list
817
+ if (data.getUtfString('zName') != null)
818
+ this._updateZoneNameInList(data.getInt('zId'), data.getUtfString('zName'));
819
+
820
+ // If the current user is the updater, show a notification; otherwise, show a dialog box suggesting to reload
821
+ if (username == this.smartFox.mySelf.name)
822
+ {
823
+ // Enable interface
824
+ this._enableConfigInterface(true);
825
+
826
+ // Display notification
827
+ this.shellCtrl.showNotification('Zone modified', `Zone settings updated successfully; changes will be applied on next <strong>server restart</strong>`);
828
+
829
+ // Reset the 'modified' flag
830
+ this._interfaceBuilder.resetIsModified();
831
+ }
832
+ else
833
+ {
834
+ // An alert box is displayed if the user is currently editing the same zone
835
+ if (data.getInt('zId') == this._editedZoneId)
836
+ {
837
+ // Show alert
838
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has modified the Zone you are currently editing; please reload to update your view.`);
839
+
840
+ // Disable submit button
841
+ $('#znc-submitButton').attr('disabled', true);
842
+ }
843
+ else
844
+ {
845
+ // Display notification
846
+ if (data.getUtfString('zName') != null)
847
+ this.shellCtrl.showNotification('Zone renamed', `Administrator ${username} has changed the name on one of the Zones`);
848
+ }
849
+ }
850
+ }
851
+
852
+ // New zone added
853
+ else if (command == this.RESP_ZONE_ADDED)
854
+ {
855
+ const zoneName = data.getSFSObject('zone').getUtfString('name');
856
+
857
+ // If the current user is the updater, reset the interface; otherwise, just show a notification
858
+ if (username == this.smartFox.mySelf.name)
859
+ {
860
+ // Reset interface
861
+ this._onCancelClick();
862
+
863
+ // Display notification
864
+ this.shellCtrl.showNotification('Zone added', `Zone '${zoneName}' created successfully`);
865
+ }
866
+ else
867
+ {
868
+ // Display notification
869
+ this.shellCtrl.showNotification('Zone added', `Administrator ${username} created Zone '${zoneName}'`);
870
+ }
871
+
872
+ // Add new zone to tree
873
+ let zonesDS = this._treeview.dataSource;
874
+ zonesDS.add(this._createZoneObject(data.getSFSObject('zone')));
875
+ zonesDS.sync();
876
+ }
877
+
878
+ // New zone creation refused due to invalid zone name
879
+ else if (command == this.RESP_ZONE_REFUSED)
880
+ {
881
+ // Re-enable interface
882
+ this._enableConfigInterface(true);
883
+
884
+ // Show warning
885
+ this.shellCtrl.showSimpleAlert('Zone configuration can\'t be saved because another Zone with the same name already exists.', true);
886
+ }
887
+
888
+ // Existing zone deleted
889
+ else if (command == this.RESP_ZONE_DELETED)
890
+ {
891
+ // If the current user is the deleter, reset the interface; otherwise, just show a notification
892
+ if (username == this.smartFox.mySelf.name)
893
+ {
894
+ // Re-enable interface
895
+ this._enableListInterface(true);
896
+
897
+ // Display notification
898
+ this.shellCtrl.showNotification('Zone removed', `Zone '${data.getUtfString('zName')}' deleted successfully`);
899
+ }
900
+ else
901
+ {
902
+ // An alert box is displayed if the user is currently editing the same zone
903
+ if (data.getInt('zId') == this._editedZoneId)
904
+ {
905
+ // Show alert
906
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has deleted the Zone you are currently modifying; you have to cancel your editing.`);
907
+
908
+ // Disable submit and reload buttons
909
+ $('#znc-reloadButton').attr('disabled', true);
910
+ $('#znc-submitButton').attr('disabled', true);
911
+ }
912
+ else
913
+ {
914
+ // Display notification
915
+ this.shellCtrl.showNotification('Zone removed', `Administrator ${username} deleted Zone '${data.getUtfString('zName')}'`);
916
+ }
917
+ }
918
+
919
+ // Reset selection if the currently selected item or its parent is being removed
920
+ let selectedNode = this._treeview.select();
921
+ let selectedDataItem = this._treeview.dataItem(selectedNode);
922
+ if (selectedDataItem)
923
+ {
924
+ if (selectedDataItem.type == this.ITEM_TYPE_ZONE && selectedDataItem.id == data.getInt('zId'))
925
+ this._deselectTreeItem();
926
+
927
+ if (selectedDataItem.type == this.ITEM_TYPE_ROOM)
928
+ {
929
+ let parentDataItem = this._treeview.dataItem(this._treeview.parent(selectedNode));
930
+
931
+ if (parentDataItem.id == data.getInt('zId'))
932
+ this._deselectTreeItem();
933
+ }
934
+ }
935
+
936
+ // Remove zone from tree
937
+ let dataItem = this._getZoneDataItemById(data.getInt('zId'));
938
+ let zonesDS = this._treeview.dataSource;
939
+ zonesDS.remove(dataItem);
940
+ zonesDS.sync();
941
+ }
942
+
943
+ // Zone activated
944
+ else if (command == this.RESP_ZONE_ACTIVATED)
945
+ {
946
+ // Set zone activation status
947
+ const zoneName = this._setZoneActivationStatus(data.getInt('zId'), data.getUtfString('actRooms'), true);
948
+
949
+ // Display notification
950
+ if (username == this.smartFox.mySelf.name)
951
+ this.shellCtrl.showNotification('Zone activated', `Zone '${zoneName}' activated successfully`);
952
+ else
953
+ this.shellCtrl.showNotification('Zone activated', `Administrator ${username} activated Zone '${zoneName}'`);
954
+ }
955
+
956
+ // Zone activation error
957
+ else if (command == this.RESP_ZONE_ACTIVATION_ERROR)
958
+ {
959
+ // Set zone activation status
960
+ this._setZoneActivationStatus(data.getInt('zId'), '', false);
961
+
962
+ // Show alert
963
+ this.shellCtrl.showSimpleAlert(data.getUtfString('error'), true);
964
+ }
965
+
966
+ /****** ROOMS ******/
967
+
968
+ // Room configuration update confirmation
969
+ else if (command == this.RESP_ROOM_CONFIG_UPDATE_CONFIRM)
970
+ {
971
+ if (data.getUtfString('rName') != null)
972
+ this._updateRoomNameInList(data.getInt('zId'), data.getInt('rId'), data.getUtfString('rName'));
973
+
974
+ // If the current user is the updater, show a notification; otherwise, show a dialog box suggesting to reload
975
+ if (username == this.smartFox.mySelf.name)
976
+ {
977
+ // Enable interface
978
+ this._enableConfigInterface(true);
979
+
980
+ // Display notification
981
+ this.shellCtrl.showNotification('Room modified', `Room settings updated successfully; changes will be applied on next <strong>server restart</strong>`);
982
+
983
+ // Reset the 'modified' flag
984
+ this._interfaceBuilder.resetIsModified();
985
+ }
986
+ else
987
+ {
988
+ // An alert box is displayed if the user is currently editing the same room
989
+ if (data.getInt('rId') == this._editedRoomId)
990
+ {
991
+ // Show alert
992
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has modified the Room you are currently editing; please reload to update your view.`);
993
+
994
+ // Disable submit button
995
+ $('#znc-submitButton').attr('disabled', true);
996
+ }
997
+ else
998
+ {
999
+ // Display notification
1000
+ if (data.getUtfString('rName') != null)
1001
+ this.shellCtrl.showNotification('Room renamed', `Administrator ${username} has changed the name on one of the Rooms`);
1002
+ }
1003
+ }
1004
+ }
1005
+
1006
+ // New room added
1007
+ else if (command == this.RESP_ROOM_ADDED)
1008
+ {
1009
+ const roomData = data.getSFSObject('room');
1010
+ const zoneId = data.getInt('zId');
1011
+
1012
+ let zonesDS = this._treeview.dataSource;
1013
+ let zoneItem = zonesDS.get(zoneId);
1014
+
1015
+ // If the current user is the updater, reset the interface; otherwise, just show a notification
1016
+ if (username == this.smartFox.mySelf.name)
1017
+ {
1018
+ // Reset interface
1019
+ this._onCancelClick();
1020
+
1021
+ // Display notification
1022
+ this.shellCtrl.showNotification('Room added', `Room '${roomData.getUtfString('name')}' created successfully`);
1023
+ }
1024
+ else
1025
+ {
1026
+ // Display notification
1027
+ this.shellCtrl.showNotification('Room added', `Administrator ${username} created Room '${roomData.getUtfString('name')}' in Zone '${zoneItem.name}'`);
1028
+ }
1029
+
1030
+ // Add new room to tree
1031
+ zoneItem.append(this._createRoomObject(roomData, zoneId));
1032
+ zonesDS.sync();
1033
+
1034
+ // Expand zone node where room was added
1035
+ this._treeview.expand(this._treeview.select());
1036
+ }
1037
+
1038
+ // New room creation refused due to invalid room name
1039
+ else if (command == this.RESP_ROOM_REFUSED)
1040
+ {
1041
+ // Re-enable interface
1042
+ this._enableConfigInterface(true);
1043
+
1044
+ // Show warning
1045
+ this.shellCtrl.showSimpleAlert('Room configuration can\'t be saved because another Room with the same name already exists.', true);
1046
+ }
1047
+
1048
+ // Existing room deleted
1049
+ else if (command == this.RESP_ROOM_DELETED)
1050
+ {
1051
+ let zoneItem = this._getZoneDataItemById(data.getInt('zId'));
1052
+ let roomItem = this._getRoomDataItemById(data.getInt('zId'), data.getInt('rId'));
1053
+
1054
+ // If the current user is the deleter, reset the interface; otherwise, just show a notification
1055
+ if (username == this.smartFox.mySelf.name)
1056
+ {
1057
+ // Re-enable interface
1058
+ this._enableListInterface(true);
1059
+
1060
+ // Display notification
1061
+ this.shellCtrl.showNotification('Room removed', `Room '${roomItem.name}' deleted successfully`);
1062
+ }
1063
+ else
1064
+ {
1065
+ // An alert box is displayed if the user is currently editing the same room
1066
+ if (data.getInt('rId') == this._editedRoomId)
1067
+ {
1068
+ // Show alert
1069
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has deleted the Room you are currently modifying; you have to cancel your editing.`);
1070
+
1071
+ // Disable submit and reload buttons
1072
+ $('#znc-reloadButton').attr('disabled', true);
1073
+ $('#znc-submitButton').attr('disabled', true);
1074
+ }
1075
+ else
1076
+ {
1077
+ // Display notification
1078
+ this.shellCtrl.showNotification('Room removed', `Administrator ${username} deleted Room '${roomItem.name}' from Zone '${zoneItem.name}'`);
1079
+ }
1080
+ }
1081
+
1082
+ // Reset selection if the currently selected item or its parent is being removed
1083
+ let selectedNode = this._treeview.select();
1084
+ let selectedDataItem = this._treeview.dataItem(selectedNode);
1085
+ if (selectedDataItem)
1086
+ {
1087
+ if (selectedDataItem.type == this.ITEM_TYPE_ROOM && selectedDataItem.id == data.getInt('rId'))
1088
+ this._deselectTreeItem();
1089
+ }
1090
+
1091
+ // Remove room from tree
1092
+ zoneItem.children.remove(roomItem);
1093
+ this._treeview.dataSource.sync();
1094
+ }
1095
+
1096
+ /****** WORDS FILES ******/
1097
+
1098
+ // Words files list received
1099
+ else if (command == this.RESP_REFRESH_WORDS_FILES)
1100
+ {
1101
+ this._wordsFilesManager.refreshWordsFilesList(data.getSFSArray('wf'), username == this.smartFox.mySelf.name);
1102
+
1103
+ // If another user caused a refresh (for example deleting a file, or adding a new one) show a notification
1104
+ if (username != null && username != this.smartFox.mySelf.name && this._editedZoneId > -1)
1105
+ this.shellCtrl.showNotification('Words files modified', `Administrator ${username} has added, modified or deleted a words file.`);
1106
+ }
1107
+
1108
+ // Words file content received
1109
+ else if (command == this.RESP_WORDS_FILE_CONTENT)
1110
+ {
1111
+ this._wordsFilesManager.editWordsFile(data.getUtfString('filename'), data.getText('content'));
1112
+ }
1113
+
1114
+ // Words file error (edit/save)
1115
+ else if (command == this.RESP_WORDS_FILE_ERROR)
1116
+ {
1117
+ // Enable buttons
1118
+ this._wordsFilesManager.enabled = true;
1119
+
1120
+ // Show alert
1121
+ this.shellCtrl.showSimpleAlert(data.getUtfString('error'), true);
1122
+ }
1123
+
1124
+ // else if ()
1125
+ }
1126
+
1127
+ //---------------------------------
1128
+ // UI EVENT LISTENERS
1129
+ //---------------------------------
1130
+
1131
+ _onTreeItemDoubleClick(e)
1132
+ {
1133
+ // Get event target's closest tree node
1134
+ let treeNode = $(e.target).closest('.k-item[role=treeitem]');
1135
+
1136
+ // Get associated data item
1137
+ let dataItem = this._treeview.dataItem(treeNode);
1138
+
1139
+ // Load configuration
1140
+ this._loadConfiguration(dataItem.type);
1141
+ }
1142
+
1143
+ _onZoneRoomChange()
1144
+ {
1145
+ // Reset utility buttons
1146
+ this._setUtilityButtonsState(this._selectedItem);
1147
+ }
1148
+
1149
+ // Utility buttons listeners
1150
+
1151
+ _onAddZoneClick()
1152
+ {
1153
+ // Deselect list item
1154
+ this._deselectTreeItem();
1155
+
1156
+ // Load configuration
1157
+ this._loadConfiguration(this.ITEM_TYPE_ZONE);
1158
+ }
1159
+
1160
+ _onAddRoomClick()
1161
+ {
1162
+ // Select parent list item
1163
+ this._selectParentTreeItem();
1164
+
1165
+ // Load configuration
1166
+ this._loadConfiguration(this.ITEM_TYPE_ROOM);
1167
+ }
1168
+
1169
+ _onEditClick()
1170
+ {
1171
+ // Load configuration
1172
+ this._loadConfiguration(this._selectedItem.type);
1173
+ }
1174
+
1175
+ _onRemoveClick()
1176
+ {
1177
+ this.shellCtrl.showConfirmWarning(`Are you sure you want to delete the selected ${this._selectedItem.type == this.ITEM_TYPE_ZONE ? 'Zone' : 'Room'} configuration?`, $.proxy(this._onRemoveConfirm, this));
1178
+ }
1179
+
1180
+ _onRemoveConfirm()
1181
+ {
1182
+ // Disable zone/room selection list
1183
+ this._enableListInterface(false);
1184
+
1185
+ let params = new SFS2X.SFSObject();
1186
+
1187
+ // Request zone removal
1188
+ if (this._selectedItem.type == this.ITEM_TYPE_ZONE)
1189
+ {
1190
+ params.putInt('zId', this._selectedItem.id);
1191
+ this.sendExtensionRequest(this.REQ_DELETE_ZONE_CONFIG, params);
1192
+ }
1193
+ else
1194
+ {
1195
+ params.putInt('zId', this._selectedItemParent.id);
1196
+ params.putInt('rId', this._selectedItem.id);
1197
+ this.sendExtensionRequest(this.REQ_DELETE_ROOM_CONFIG, params);
1198
+ }
1199
+ }
1200
+
1201
+ _onActivateClick()
1202
+ {
1203
+ // Get selected data item
1204
+ if (this._selectedItem.type == this.ITEM_TYPE_ZONE)
1205
+ {
1206
+ let params = new SFS2X.SFSObject();
1207
+ params.putInt('zId', this._selectedItem.id);
1208
+
1209
+ this.sendExtensionRequest(this.REQ_ACTIVATE_ZONE, params);
1210
+ }
1211
+ }
1212
+
1213
+ // Configuration buttons listeners
1214
+
1215
+ _onCancelClick()
1216
+ {
1217
+ // Enable zone/room selection lists
1218
+ this._enableListInterface(true);
1219
+
1220
+ // Disable configuration interface
1221
+ this._enableConfigInterface(false);
1222
+
1223
+ // Clear main container
1224
+ this._resetTabsContainer(false, true);
1225
+
1226
+ // Set isEditing flag
1227
+ this._isEditing = false;
1228
+ this._editedItemType = '';
1229
+
1230
+ // Switch panel
1231
+ this._switchPanel('znc-sidebarPanel');
1232
+ }
1233
+
1234
+ _onReloadClick()
1235
+ {
1236
+ // Hide validation messages
1237
+ this._interfaceBuilder.resetValidation();
1238
+
1239
+ // Reload configuration
1240
+ this._loadConfiguration(this._editedItemType, false);
1241
+ }
1242
+
1243
+ _onSubmitClick()
1244
+ {
1245
+ // Check validity
1246
+ if (this._interfaceBuilder.checkIsValid())
1247
+ {
1248
+ let changes = this._interfaceBuilder.getChangedData();
1249
+
1250
+ if (changes.size() > 0)
1251
+ {
1252
+ //console.log(changes.getDump())
1253
+ //return;
1254
+
1255
+ // In case the zone/room name changed, check it against the list (duplicate names not allowed!)
1256
+ if (this._validateName(changes))
1257
+ {
1258
+ // Disable configuration interface
1259
+ this._enableConfigInterface(false);
1260
+
1261
+ // Send settings to server instance
1262
+ let params = new SFS2X.SFSObject();
1263
+ params.putSFSArray('settings', changes);
1264
+ params.putBool('backup', $('#znc-backupCheck').prop('checked'));
1265
+ params.putInt('zId', this._editedZoneId);
1266
+ params.putInt('rId', this._editedRoomId);
1267
+
1268
+ if (this._editedItemType == this.ITEM_TYPE_ZONE)
1269
+ {
1270
+ // Submit zone settings
1271
+ if (this._editedZoneId > -1)
1272
+ this.sendExtensionRequest(this.REQ_SAVE_ZONE_CONFIG, params);
1273
+ else
1274
+ this.sendExtensionRequest(this.REQ_NEW_ZONE_CONFIG, params);
1275
+ }
1276
+ else
1277
+ {
1278
+ // Submit room settings
1279
+ if (this._editedRoomId > -1)
1280
+ this.sendExtensionRequest(this.REQ_SAVE_ROOM_CONFIG, params);
1281
+ else
1282
+ this.sendExtensionRequest(this.REQ_NEW_ROOM_CONFIG, params);
1283
+ }
1284
+ }
1285
+ else
1286
+ {
1287
+ // Show alert
1288
+ this.shellCtrl.showSimpleAlert(`Unable to submit configuration because the ${Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_4__["capitalizeFirst"])(this._editedItemType)} name already exists; duplicate names are not allowed.`, true);
1289
+ }
1290
+ }
1291
+ }
1292
+ else
1293
+ {
1294
+ // Show alert
1295
+ this.shellCtrl.showSimpleAlert('Unable to submit configuration changes due to an invalid value; please verify the highlighted form fields in all tabs.', true);
1296
+ }
1297
+ }
1298
+
1299
+ _onConfigValueSet(e) // SAME METHOD DUPLICATED IN zone-monitor.js
1300
+ {
1301
+ const configParam = e.target.data;
1302
+
1303
+ // Handle extension name/type dropdowns update and update the main class dropdown datasource accordingly
1304
+ if (configParam.name == 'extension.name' || configParam.name == 'extension.type' || configParam.name == 'extension.filterClass')
1305
+ {
1306
+ // All involved ConfigFormItems must be available and initialized to proceed
1307
+ const nameFormItem = this._interfaceBuilder.getConfigFormItem('extension.name');
1308
+ const typeFormItem = this._interfaceBuilder.getConfigFormItem('extension.type');
1309
+ const classFormItem = this._interfaceBuilder.getConfigFormItem('extension.file');
1310
+ const filterFormItem = this._interfaceBuilder.getConfigFormItem('extension.filterClass');
1311
+
1312
+ if (nameFormItem != null && typeFormItem != null && classFormItem != null && filterFormItem != null)
1313
+ {
1314
+ const source = nameFormItem.data;
1315
+ let classesList = [];
1316
+
1317
+ let data = source.triggerData;
1318
+ for (let i = 0; i < data.size(); i++)
1319
+ {
1320
+ let ext = data.getSFSObject(i);
1321
+
1322
+ if (ext.getUtfString('name') == nameFormItem.data.value && ext.getUtfString('type') == typeFormItem.data.value)
1323
+ {
1324
+ let classes = ext.get('classesString').split(','); // We don't use "getUTfString" because the type is Text in case a very large list of classes is returned
1325
+
1326
+ if (filterFormItem.data.value == true)
1327
+ {
1328
+ let filteredClasses = classes.filter(_utils_utilities__WEBPACK_IMPORTED_MODULE_4__["filterClassName"]);
1329
+ classes = filteredClasses;
1330
+ }
1331
+
1332
+ classesList = classesList.concat(classes);
1333
+ }
1334
+ }
1335
+
1336
+ let currentClass = classFormItem.data.value;
1337
+
1338
+ // If the classes list doesn't contain the current value, create an empty entry and reset the value
1339
+ if (classesList.indexOf(currentClass) < 0)
1340
+ {
1341
+ if (classesList.length == 0)
1342
+ {
1343
+ classesList.push('');
1344
+ currentClass = '';
1345
+ }
1346
+ else
1347
+ currentClass = classesList[0];
1348
+ }
1349
+
1350
+ let mainClassDropDown = classFormItem._innerWidget;
1351
+ mainClassDropDown.setDataSource(classesList);
1352
+
1353
+ classFormItem.data.value = currentClass;
1354
+ classFormItem._setWidgetValue();
1355
+ }
1356
+ }
1357
+ }
1358
+
1359
+ _onWordsFileReloadClick(evt)
1360
+ {
1361
+ // Send request to server
1362
+ this.sendExtensionRequest(this.REQ_REFRESH_WORDS_FILE);
1363
+ }
1364
+
1365
+ _onWordsFileEditClick(evt)
1366
+ {
1367
+ // Send request to server
1368
+ let params = new SFS2X.SFSObject();
1369
+ params.putUtfString('filename', evt.detail);
1370
+ this.sendExtensionRequest(this.REQ_EDIT_WORDS_FILE, params);
1371
+ }
1372
+
1373
+ _onWordsFileSaveClick(evt)
1374
+ {
1375
+ this._tempWordsFileData = evt.detail;
1376
+
1377
+ // Check if a new file is being created
1378
+ if (this._tempWordsFileData.isNew)
1379
+ {
1380
+ // If yes, check if name already exists
1381
+ if (this._wordsFilesManager.getExistingFilenames().includes(this._tempWordsFileData.filename))
1382
+ {
1383
+ // Show confirm dialog
1384
+ this.shellCtrl.showConfirmWarning('A words file with the entered name already exists; do you want to proceed anyway? The existing file will be overwritten.', $.proxy(this._onWordsFileSaveConfirm, this));
1385
+ return;
1386
+ }
1387
+ }
1388
+
1389
+ // Proceed
1390
+ this._onWordsFileSaveConfirm();
1391
+ }
1392
+
1393
+ _onWordsFileSaveConfirm()
1394
+ {
1395
+ // Disable words files manager buttons
1396
+ this._wordsFilesManager.enabled = false;
1397
+ this._wordsFilesManager.saveSpinnerVisible = true;
1398
+
1399
+ // Send request to server
1400
+ let params = new SFS2X.SFSObject();
1401
+ params.putUtfString('filename', this._tempWordsFileData.filename);
1402
+ params.putText('content', this._tempWordsFileData.content);
1403
+ this.sendExtensionRequest(this.REQ_SAVE_WORDS_FILE, params);
1404
+ }
1405
+
1406
+ _onWordsFileRemoveClick(evt)
1407
+ {
1408
+ this.shellCtrl.showConfirmWarning('Are you sure you want to delete the selected words file?', $.proxy(this._onWordsFileRemoveConfirm, this));
1409
+ }
1410
+
1411
+ _onWordsFileRemoveConfirm()
1412
+ {
1413
+ let wordsFile = this._wordsFilesManager.getSelectedWordsFileName();
1414
+
1415
+ if (wordsFile != null)
1416
+ {
1417
+ // Disable words files manager buttons
1418
+ this._wordsFilesManager.enabled = false;
1419
+ this._wordsFilesManager.actionSpinnerVisible = true;
1420
+
1421
+ // Send request to server
1422
+ let params = new SFS2X.SFSObject();
1423
+ params.putUtfString('filename', wordsFile);
1424
+ this.sendExtensionRequest(this.REQ_DELETE_WORDS_FILE, params);
1425
+ }
1426
+ }
1427
+
1428
+ _onWordsFileAssignClick(evt)
1429
+ {
1430
+ let path = evt.detail;
1431
+
1432
+ // Write path of the selected words file in "wordsFilter.wordsFile" dynamically created field
1433
+ const wordsFileFormItem = this._interfaceBuilder.getConfigFormItem('wordsFilter.wordsFile');
1434
+ wordsFileFormItem.data.value = path;
1435
+ wordsFileFormItem._setWidgetValue();
1436
+ }
1437
+
1438
+ //---------------------------------
1439
+ // PRIVATE METHODS
1440
+ //---------------------------------
1441
+
1442
+ _enableListInterface(enabled)
1443
+ {
1444
+ $('#znc-utilButtons').attr('disabled', !enabled);
1445
+ $('#znc-treeView').attr('disabled', !enabled);
1446
+ }
1447
+
1448
+ _setUtilityButtonsState(dataItem = null)
1449
+ {
1450
+ let disable = true;
1451
+
1452
+ if (dataItem)
1453
+ {
1454
+ // Enable 'activate zone' button if zone is inactive
1455
+ $('#znc-activateButton').attr('disabled', (dataItem.type != this.ITEM_TYPE_ZONE || dataItem.active));
1456
+
1457
+ disable = false;
1458
+ }
1459
+ else
1460
+ {
1461
+ // Disable 'activate zone' button
1462
+ $('#znc-activateButton').attr('disabled', true);
1463
+ }
1464
+
1465
+ // Enable/disable other utility buttons
1466
+ $('#znc-addZoneButton').attr('disabled', false); // Always enabled
1467
+ $('#znc-addRoomButton').attr('disabled', disable);
1468
+ $('#znc-editButton').attr('disabled', disable);
1469
+ $('#znc-removeButton').attr('disabled', disable);
1470
+ }
1471
+
1472
+ _enableConfigInterface(enabled)
1473
+ {
1474
+ $('#znc-cancelButton').attr('disabled', !enabled);
1475
+ $('#znc-reloadButton').attr('disabled', !enabled);
1476
+ $('#znc-submitButton').attr('disabled', !enabled);
1477
+ $('#znc-backupCheck').attr('disabled', !enabled);
1478
+
1479
+ this._interfaceBuilder.disableInterface(!enabled);
1480
+
1481
+ // Also switch view when enabled
1482
+ if (enabled)
1483
+ this._switchView('znc-main');
1484
+ }
1485
+
1486
+ _switchView(viewId)
1487
+ {
1488
+ document.getElementById('znc-viewstack').selectedElement = document.getElementById(viewId);
1489
+ }
1490
+
1491
+ _clearTabs()
1492
+ {
1493
+ // Destroy scrolling tabs
1494
+ $('#znc-tabNavigator #tabs').scrollingTabs('destroy');
1495
+
1496
+ // Remove all tab navigator content
1497
+ this._interfaceBuilder.destroyInterface();
1498
+
1499
+ // Set flag to re-initialize tabs if needed
1500
+ this._reinitTabs = true;
1501
+ }
1502
+
1503
+ _populateTree(data)
1504
+ {
1505
+ let zData = data.getSFSArray('zones');
1506
+
1507
+ let zonesArr = [];
1508
+ for (let z = 0; z < zData.size(); z++)
1509
+ zonesArr.push( this._createZoneObject(zData.getSFSObject(z)) );
1510
+
1511
+ // Create datasource
1512
+ let zones = new kendo.data.HierarchicalDataSource({
1513
+ data: zonesArr,
1514
+ sort: {
1515
+ field: 'name',
1516
+ dir: 'asc'
1517
+ },
1518
+ schema: {
1519
+ model: {
1520
+ id: 'id',
1521
+ children: {
1522
+ schema: {
1523
+ data: 'rooms',
1524
+ sort: {
1525
+ field: 'name',
1526
+ dir: 'asc'
1527
+ }
1528
+ }
1529
+ }
1530
+ }
1531
+ }
1532
+ });
1533
+
1534
+ // Set tree view dataprovider
1535
+ this._treeview.setDataSource(zones);
1536
+
1537
+ // Set utility buttons state (add, remove, edit, etc)
1538
+ this._setUtilityButtonsState();
1539
+ }
1540
+
1541
+ _createZoneObject(zoneData)
1542
+ {
1543
+ let zone = {
1544
+ type: this.ITEM_TYPE_ZONE,
1545
+ name: zoneData.getUtfString('name'),
1546
+ id: zoneData.getInt('id'),
1547
+ active: zoneData.getBool('act')
1548
+ }
1549
+
1550
+ // Create rooms list dataprovider
1551
+ let rData = zoneData.getSFSArray('rooms');
1552
+
1553
+ let roomsArr = [];
1554
+ for (let r = 0; r < rData.size(); r++)
1555
+ roomsArr.push( this._createRoomObject(rData.getSFSObject(r), zoneData.getInt('id')) );
1556
+
1557
+ zone.rooms = roomsArr;
1558
+
1559
+ return zone;
1560
+ }
1561
+
1562
+ _createRoomObject(roomData, zoneId)
1563
+ {
1564
+ let room = {
1565
+ type: this.ITEM_TYPE_ROOM,
1566
+ name: roomData.getUtfString('name'),
1567
+ id: roomData.getInt('id'),
1568
+ active: roomData.getBool('act'),
1569
+ parentId: zoneId,
1570
+ spriteCssClass: this._getRoomListIconCssClass(roomData.getBool('act'))
1571
+ };
1572
+
1573
+ return room;
1574
+ }
1575
+
1576
+ _getRoomListIconCssClass(isActive)
1577
+ {
1578
+ return isActive ? 'fas fa-door-open' : 'fas fa-door-closed inactive-list-item';
1579
+ }
1580
+
1581
+ _setZoneActivationStatus(zoneId, activeRooms, isActive)
1582
+ {
1583
+ let zoneDI = this._getZoneDataItemById(zoneId);
1584
+
1585
+ zoneDI.active = isActive;
1586
+
1587
+ let activeRoomsArr = activeRooms.split(',');
1588
+
1589
+ if (zoneDI.hasChildren)
1590
+ {
1591
+ for (let i = 0; i < zoneDI.children.data().length; i++)
1592
+ {
1593
+ let room = zoneDI.children.data()[i];
1594
+ room.active = (isActive && activeRoomsArr.indexOf(room.name) > -1);
1595
+ room.spriteCssClass = this._getRoomListIconCssClass(room.active)
1596
+ }
1597
+ }
1598
+
1599
+ // Refresh list
1600
+ this._treeview.dataSource.sync();
1601
+
1602
+ // Return zone name
1603
+ return zoneDI.name;
1604
+ }
1605
+
1606
+ _deselectTreeItem()
1607
+ {
1608
+ this._treeview.select($());
1609
+ }
1610
+
1611
+ _selectParentTreeItem()
1612
+ {
1613
+ let selectedNode = this._treeview.select();
1614
+ let selectedDataItem = this._treeview.dataItem(selectedNode);
1615
+
1616
+ if (selectedDataItem.type == this.ITEM_TYPE_ROOM)
1617
+ {
1618
+ let parentNode = this._treeview.parent(selectedNode);
1619
+ this._treeview.select(parentNode);
1620
+ }
1621
+ }
1622
+
1623
+ _loadConfiguration(type, resetTabs = true)
1624
+ {
1625
+ // Disable zone/room selection list
1626
+ this._enableListInterface(false);
1627
+
1628
+ // Disable configuration interface
1629
+ this._enableConfigInterface(false);
1630
+
1631
+ // Clear main container
1632
+ this._resetTabsContainer(true, resetTabs);
1633
+
1634
+ // Set isEditing flag
1635
+ this._isEditing = true;
1636
+ this._editedItemType = type;
1637
+
1638
+ // Request zone or room configuration data to server instance
1639
+ let params = new SFS2X.SFSObject();
1640
+ params.putInt('zId', this._editedZoneId);
1641
+ params.putInt('rId', this._editedRoomId);
1642
+
1643
+ // If no room is selected, then we are editing a zone
1644
+ if (this._editedItemType == this.ITEM_TYPE_ZONE)
1645
+ this.sendExtensionRequest(this.REQ_GET_ZONE_CONFIG, params);
1646
+ else
1647
+ this.sendExtensionRequest(this.REQ_GET_ROOM_CONFIG, params);
1648
+
1649
+ // Switch panel
1650
+ this._switchPanel('znc-mainPanel');
1651
+ }
1652
+
1653
+ _resetTabsContainer(isLoading, resetTabs)
1654
+ {
1655
+ if (resetTabs)
1656
+ this._clearTabs();
1657
+ else
1658
+ this._reinitTabs = false;
1659
+
1660
+ if (!isLoading)
1661
+ this._switchView('znc-select');
1662
+ else
1663
+ this._switchView('znc-loading');
1664
+ }
1665
+
1666
+ _switchPanel(panelId)
1667
+ {
1668
+ document.getElementById('znc-view').selectedPanel = document.getElementById(panelId);
1669
+ }
1670
+
1671
+ _getZoneDataItemById(zoneId)
1672
+ {
1673
+ let zonesDS = this._treeview.dataSource;
1674
+ return zonesDS.get(zoneId);
1675
+ }
1676
+
1677
+ _getRoomDataItemById(zoneId, roomId)
1678
+ {
1679
+ let zoneDI = this._getZoneDataItemById(zoneId);
1680
+
1681
+ if (zoneDI.hasChildren)
1682
+ return zoneDI.children.get(roomId);
1683
+
1684
+ return null;
1685
+ }
1686
+
1687
+ _updateZoneNameInList(zoneId, zoneName)
1688
+ {
1689
+ this._getZoneDataItemById(zoneId).name = zoneName;
1690
+ this._treeview.dataSource.sync();
1691
+ }
1692
+
1693
+ _updateRoomNameInList(zoneId, roomId, roomName)
1694
+ {
1695
+ this._getRoomDataItemById(zoneId, roomId).name = roomName;
1696
+ this._treeview.dataSource.sync();
1697
+ }
1698
+
1699
+ _validateName(changes)
1700
+ {
1701
+ const zoneId = this._editedZoneId;
1702
+
1703
+ for (let i = 0; i < changes.size(); i++)
1704
+ {
1705
+ const setting = changes.getSFSObject(i);
1706
+
1707
+ if (setting.containsKey('name') && setting.getUtfString('name') == 'name')
1708
+ {
1709
+ // Get name value
1710
+ const name = setting.getText('value');
1711
+
1712
+ // Get data source
1713
+ let ds = [];
1714
+
1715
+ if (this._editedItemType == this.ITEM_TYPE_ZONE)
1716
+ ds = this._treeview.dataSource.data();
1717
+ else
1718
+ {
1719
+ if (this._getZoneDataItemById(zoneId).hasChildren)
1720
+ ds = this._getZoneDataItemById(zoneId).children.data();
1721
+ }
1722
+
1723
+
1724
+ // Check if name exists in data source
1725
+ for (let j = 0; j < ds.length; j++)
1726
+ {
1727
+ if (ds[j].name == name)
1728
+ {
1729
+ return false;
1730
+ }
1731
+ }
1732
+
1733
+ break;
1734
+ }
1735
+ }
1736
+
1737
+ return true;
1738
+ }
1739
+
1740
+ //---------------------------------
1741
+ // PRIVATE GETTERS
1742
+ //---------------------------------
1743
+
1744
+ get _selectedItem()
1745
+ {
1746
+ return this._treeview.dataItem(this._treeview.select());
1747
+ }
1748
+
1749
+ get _selectedItemParent()
1750
+ {
1751
+ let selectedNode = this._treeview.select();
1752
+ let parentNode = this._treeview.parent(selectedNode);
1753
+
1754
+ return this._treeview.dataItem(parentNode);
1755
+ }
1756
+
1757
+ get _editedZoneId()
1758
+ {
1759
+ if (this._isEditing && this._selectedItem)
1760
+ {
1761
+ if (this._selectedItem.type == this.ITEM_TYPE_ZONE)
1762
+ return this._selectedItem.id;
1763
+ else
1764
+ return this._selectedItemParent.id;
1765
+ }
1766
+
1767
+ return -1;
1768
+ }
1769
+
1770
+ get _editedRoomId()
1771
+ {
1772
+ if (this._isEditing && this._selectedItem)
1773
+ {
1774
+ if (this._selectedItem.type == this.ITEM_TYPE_ROOM)
1775
+ return this._selectedItem.id;
1776
+ }
1777
+
1778
+ return -1;
1779
+ }
1780
+ }
1781
+
1782
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1783
+
1784
+ /***/ })
1785
+
1786
+ }]);
1787
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-16.bundle.js","sources":["webpack://application/./src/components/module-specific/words-files-manager.js","webpack://application/./src/components/sidebar-layout.js","webpack://application/./src/modules/zone-configurator.js"],"sourcesContent":["import {bytesToSize} from '../../utils/utilities';\n\nexport class WordsFilesManager extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis.REFRESH_WORDS_FILES_CLICK_EVENT = 'refreshWordsFilesClick';\n\t\tthis.EDIT_WORDS_FILE_CLICK_EVENT = 'editWordsFileClick';\n\t\tthis.SAVE_WORDS_FILE_CLICK_EVENT = 'saveWordsFileClick';\n\t\tthis.REMOVE_WORDS_FILE_CLICK_EVENT = 'removeWordsFileClick';\n\t\tthis.ASSIGN_WORDS_FILE_CLICK_EVENT = 'assingWordsFileClick';\n\n\t\tthis.CONFIG_FOLDER = 'config/';\n\t\tthis.WORDS_FILE_EXT = '.words.txt';\n\n\t\tthis._modalHtml = `\n\t\t\t<div class=\"modal\" id=\"editModal\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"editModalTitle\" aria-hidden=\"true\">\n\t\t\t\t<div class=\"modal-dialog modal-dialog-centered\" role=\"document\">\n\t\t\t\t\t<div class=\"modal-content\">\n\t\t\t\t\t\t<div class=\"modal-header\">\n\t\t\t\t\t\t\t<h5 class=\"modal-title text-primary\" id=\"editModalTitle\">Word File Editor</h5>\n\t\t\t\t\t\t\t<button type=\"button\" class=\"close\" data-dismiss=\"modal\" aria-label=\"Close\">\n\t\t\t\t\t\t\t\t<span aria-hidden=\"true\">&times;</span>\n\t\t\t\t\t\t\t</button>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-body in-flow-invalid-msg\">\n\t\t\t\t\t\t\t<fieldset id=\"editFieldset\">\n\t\t\t\t\t\t\t\t<div class=\"form-group\">\n\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t<label for=\"filename\" class=\"form-label\">Filename <i class=\"fas fa-question-circle text-muted help\" title=\"Name of the words file. The file will be saved in server's <em>config</em> folder.\"></i></label>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t<div class=\"input-group\">\n\t\t\t\t\t\t\t\t\t\t<input type=\"text\" id=\"filename\" name=\"filename\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" aria-describedby=\"extension\" />\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"input-group-append\">\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"input-group-text\" id=\"extension\">.words.txt</span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"filename\"></span>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"form-group\">\n\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t<label for=\"content\" class=\"form-label\">Content <i class=\"fas fa-question-circle text-muted help\" title=\"Enter a word or a valid regular expression per line. See configuration's <em>Words file</em> field description for additional information.\"></i></label>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t<textarea id=\"content\" name=\"content\" class=\"form-control k-textarea w-100\" rows=\"10\"></textarea>\n\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"content\"></span>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</fieldset>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-footer flex-column\">\n\t\t\t\t\t\t\t<div class=\"d-flex w-100\">\n\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-left\">\n\t\t\t\t\t\t\t\t\t<button id=\"saveWordFileButton\" type=\"button\" class=\"k-button k-primary\"><i class=\"fas fa-save mr-1\"></i>Save word file</button>\n\t\t\t\t\t\t\t\t\t<i id=\"saveSpinner\" class=\"fas fa-circle-notch fa-spin text-primary align-middle ml-1\"></i>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-right\">\n\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-secondary\" data-dismiss=\"modal\">Cancel</button>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t`;\n\n\t\t//-------------------------------------------\n\n\t\t$(this).append(`\n\t\t\t<div class=\"col-sm-5 col-lg-4 col-form-label form-label-container\">\n\t\t\t\t<label class=\"form-label\">Available words files <i class=\"fas fa-question-circle text-muted help\" title=\"The list of words files found in server's <em>config</em> folder. Click on the Assign button to set the <strong>Words file</strong> field to the selected file. Configuration submission is then required to save the new value.\"></i></label>\n\t\t\t</div>\n\t\t\t<div class=\"inner-widget align-self-center col-sm\">\n\t\t\t\t<div>\n\t\t\t\t\t<div id=\"wordsFiles\" class=\"limited-height\"></div>\n\t\t\t\t\t<div id=\"actionButtons\" class=\"mt-2 text-right\" disabled>\n\t\t\t\t\t\t<i id=\"actionSpinner\" class=\"fas fa-circle-notch fa-spin text-primary align-middle\"></i>\n\t\t\t\t\t\t<button id=\"refreshButton\" type=\"button\" class=\"k-button k-secondary ml-2\" title=\"Refresh\"><i class=\"fas fa-redo-alt\"></i></button>\n\t\t\t\t\t\t<button id=\"addButton\" type=\"button\" class=\"k-button k-secondary ml-2\" title=\"Add\"><i class=\"fas fa-plus\"></i></button>\n\t\t\t\t\t\t<button id=\"editButton\" type=\"button\" class=\"k-button k-secondary ml-2\" title=\"Edit\" disabled><i class=\"fas fa-pen\"></i></button>\n\t\t\t\t\t\t<button id=\"removeButton\" type=\"button\" class=\"k-button k-secondary ml-2\" title=\"Remove\" disabled><i class=\"fas fa-times\"></i></button>\n\t\t\t\t\t\t<button id=\"assignButton\" type=\"button\" class=\"k-button k-secondary ml-2\" title=\"Assign\" disabled><i class=\"fas fa-compress-arrows-alt\"></i></button>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t`);\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize grid\n\t\tthis._wordsFilesGrid = $('#wordsFiles', $(this)).kendoGrid({\n\t\t\tresizable: true,\n\t\t\tselectable: 'row',\n\t\t\tchange: $.proxy(this._onWordsFilesGridSelectionChange, this),\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'name',\n\t\t\t\t\ttitle: 'Filename',\n\t\t\t\t\twidth: 120\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'date',\n\t\t\t\t\ttitle: 'Date',\n\t\t\t\t\twidth: 80\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'size',\n\t\t\t\t\ttitle: 'Size',\n\t\t\t\t\twidth: 80\n\t\t\t\t}\n\t\t\t],\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No files.'\n\t\t\t}\n\t\t}).data('kendoGrid');\n\n\t\t// Add listeners to button clicks\n\t\t$('#refreshButton', $(this)).on('click', $.proxy(this._onReloadClick, this));\n\t\t$('#addButton', $(this)).on('click', $.proxy(this._onAddClick, this));\n\t\t$('#editButton', $(this)).on('click', $.proxy(this._onEditClick, this));\n\t\t$('#removeButton', $(this)).on('click', $.proxy(this._onRemoveClick, this));\n\t\t$('#assignButton', $(this)).on('click', $.proxy(this._onAssignClick, this));\n\t}\n\n\tdestroy()\n\t{\n\t\t// Destroy grid\n\t\tthis._wordsFilesGrid.destroy();\n\n\t\t// Remove event listeners\n\t\t$('#refreshButton', $(this)).off('click');\n\t\t$('#addButton', $(this)).off('click');\n\t\t$('#editButton', $(this)).off('click');\n\t\t$('#removeButton', $(this)).off('click');\n\t\t$('#assignButton', $(this)).off('click');\n\n\t\t// Hide modal (which in turn destroys it)\n\t\tlet modalElement = $('#editModal', $(this));\n\n\t\tif (modalElement)\n\t\t\tmodalElement.modal('hide');\n\t}\n\n\tget enabled()\n\t{\n\t\treturn this._isEnabled;\n\t}\n\n\tset enabled(value)\n\t{\n\t\tthis._isEnabled = value;\n\n\t\t// Enable/disable buttons\n\t\t$('#actionButtons', this).attr('disabled', !value);\n\n\t\t// Hide spinner\n\t\tif (value)\n\t\t\tthis.actionSpinnerVisible = false;\n\n\t\t// Enable/disable modal\n\t\tlet modalElement = $('#editModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Disable modal close buttons\n\t\t\t$('button[data-dismiss=\"modal\"]', modalElement).attr('disabled', !value);\n\n\t\t\t// Disable save button\n\t\t\t$('#saveWordFileButton', modalElement).attr('disabled', !value);\n\n\t\t\t// Disable fieldset\n\t\t\t$('#editFieldset', modalElement).attr('disabled', !value);\n\n\t\t\t// Hide spinner\n\t\t\tif (value)\n\t\t\t\tthis.saveSpinnerVisible = false;\n\t\t}\n\t}\n\n\tset actionSpinnerVisible(value)\n\t{\n\t\tif (value)\n\t\t\t$('#actionSpinner', $(this)).show();\n\t\telse\n\t\t\t$('#actionSpinner', $(this)).hide();\n\t}\n\n\tset saveSpinnerVisible(value)\n\t{\n\t\tlet modalElement = $('#editModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\tif (value)\n\t\t\t\t$('#saveSpinner', modalElement).show();\n\t\t\telse\n\t\t\t\t$('#saveSpinner', modalElement).hide();\n\t\t}\n\t}\n\n\trefreshWordsFilesList(wordsFilesList, hideEditModal)\n\t{\n\t\tif (hideEditModal)\n\t\t{\n\t\t\tlet modalElement = $('#editModal', $(this));\n\n\t\t\tif (modalElement)\n\t\t\t\tmodalElement.modal('hide');\n\t\t}\n\n\t\tlet files = [];\n\t\tthis._existingFilenames = [];\n\n\t\tfor (let f = 0; f < wordsFilesList.size(); f++)\n\t\t{\n\t\t\tconst file = wordsFilesList.getSFSObject(f);\n\n\t\t\tconst fileObj = {};\n\t\t\tfileObj.name = file.getUtfString('name');\n\t\t\tfileObj.date = file.getUtfString('date') + ' ' + file.getUtfString('time');\n\t\t\tfileObj.size = bytesToSize(file.getLong('size'), 2);\n\n\t\t\t// Populate files list\n\t\t\tfiles.push(fileObj);\n\n\t\t\t// Save ref to existing filenames, to check them when a new file is created\n\t\t\tthis._existingFilenames.push(fileObj.name);\n\t\t}\n\n\t\t// Assign data source to grid\n\t\tthis._setWordsFilesGridDataSource(files);\n\t\tthis._onWordsFilesGridSelectionChange();\n\n\t\t// Enable\n\t\tthis.enabled = true;\n\t}\n\n\tgetSelectedWordsFileName()\n\t{\n\t\tif (this._wordsFilesGrid.select() != null)\n\t\t{\n\t\t\tlet selectedIndex = this._wordsFilesGrid.select().index();\n\t\t\treturn this._wordsFilesGrid.dataSource.at(selectedIndex).name;\n\t\t}\n\t\telse\n\t\t\treturn null;\n\t}\n\n\teditWordsFile(filename, content)\n\t{\n\t\tthis._isNewFile = false;\n\n\t\tthis.enabled = true;\n\n\t\t// Show modal\n\t\tthis._showModal();\n\n\t\t// Remove default extension from filename\n\t\tfilename = filename.substring(0, filename.lastIndexOf(this.WORDS_FILE_EXT));\n\n\t\t// Enter content filename and content in modal form\n\t\t$('#editModal #filename', $(this)).val(filename);\n\t\t$('#editModal #content', $(this)).val(content);\n\n\t\t// Set filename field as not editable and hide note\n\t\t$('#editModal #filename', $(this)).attr('disabled', true);\n\t\t$('#editModal #filenameNote', $(this)).hide();\n\t}\n\n\tgetExistingFilenames()\n\t{\n\t\treturn this._existingFilenames;\n\t}\n\n\t_setWordsFilesGridDataSource(ds)\n\t{\n\t\t// Read current horizontal scroll value\n\t   const scrollLeft = $('#wordsFiles .k-grid-content', this._wordsFilesGrid.wrapper).scrollLeft();\n\n\t   // Assign data source to grid\n\t   this._wordsFilesGrid.setDataSource(ds);\n\n\t   // Set horizontal scroll\n\t   $('#wordsFiles .k-grid-content', this._wordsFilesGrid.wrapper).scrollLeft(scrollLeft);\n\t}\n\n\t_onWordsFilesGridSelectionChange()\n\t{\n\t\t// Enable/disable buttons\n\t\tconst selectedRows = this._wordsFilesGrid.select();\n\t\t$('#editButton').attr('disabled', selectedRows.length == 0);\n\t\t$('#removeButton').attr('disabled', selectedRows.length == 0);\n\t\t$('#assignButton').attr('disabled', selectedRows.length == 0);\n\t}\n\n\t_onReloadClick()\n\t{\n\t\t// Fire event to request file content to server\n\t\tlet evt = new CustomEvent(this.REFRESH_WORDS_FILES_CLICK_EVENT, {\n\t    \tdetail: null,\n\t\t\tbubbles: false,\n\t\t\tcancelable: false\n\t\t});\n\n\t\tthis.dispatchEvent(evt);\n\t}\n\n\t_onAddClick()\n\t{\n\t\tthis._isNewFile = true;\n\n\t\t// Show modal\n\t\tthis._showModal();\n\t}\n\n\t_onEditClick()\n\t{\n\t\t// Disable buttons\n\t\tthis.enabled = false;\n\t\tthis.actionSpinnerVisible = true;\n\n\t\t// Fire event to request file content to server\n\t\tlet evt = new CustomEvent(this.EDIT_WORDS_FILE_CLICK_EVENT, {\n\t    \tdetail: this.getSelectedWordsFileName(),\n\t\t\tbubbles: false,\n\t\t\tcancelable: false\n\t\t});\n\n\t\tthis.dispatchEvent(evt);\n\t}\n\n\t_onRemoveClick()\n\t{\n\t\t// Fire event to request file removal to server\n\t\tlet evt = new CustomEvent(this.REMOVE_WORDS_FILE_CLICK_EVENT, {\n\t    \tdetail: this.getSelectedWordsFileName(),\n\t\t\tbubbles: false,\n\t\t\tcancelable: false\n\t\t});\n\n\t\tthis.dispatchEvent(evt);\n\t}\n\n\t_onAssignClick()\n\t{\n\t\t// Fire event to substitute path in configuration\n\t\tlet evt = new CustomEvent(this.ASSIGN_WORDS_FILE_CLICK_EVENT, {\n\t    \tdetail: this.CONFIG_FOLDER + this.getSelectedWordsFileName(),\n\t\t\tbubbles: false,\n\t\t\tcancelable: false\n\t\t});\n\n\t\tthis.dispatchEvent(evt);\n\t}\n\n\t_onSaveWordFileClick()\n\t{\n\t\tif (this._validator.validate())\n\t\t{\n\t\t\t// Show spinner\n\t\t\t$('#editModal #saveSpinner', $(this)).show();\n\n\t\t\t// Fire event to request file to be saved by server\n\t\t\tlet evt = new CustomEvent(this.SAVE_WORDS_FILE_CLICK_EVENT, {\n\t\t    \tdetail: {\n\t\t\t\t\tfilename: $('#editModal #filename', $(this)).val() + this.WORDS_FILE_EXT,\n\t\t\t\t\tisNew: this._isNewFile,\n\t\t\t\t\tcontent: $('#editModal #content', $(this)).val()\n\t\t\t\t},\n\t\t\t\tbubbles: false,\n\t\t\t\tcancelable: false\n\t\t\t});\n\n\t\t\tthis.dispatchEvent(evt);\n\t\t}\n\t}\n\n\t_showModal()\n\t{\n\t\t// Append modal html\n\t\t$(this).append(this._modalHtml);\n\n\t\tlet modalElement = $('#editModal', $(this));\n\n\t\t// Initialize kendo validation\n\t\tthis._validator = modalElement.find('#editFieldset').kendoValidator({\n\t\t\tvalidateOnBlur: true,\n\t\t\trules: {\n\t\t\t\trequiredFilename: $.proxy(function(input) {\n\t\t\t\t\tlet valid = true;\n\t\t\t\t\tif (input.is('[name=filename]'))\n\t\t\t\t\t\tvalid = input.val() !== '';\n\t\t\t\t\treturn valid;\n\t\t\t\t}, this),\n\t\t\t\tvalidFilename: $.proxy(function(input) {\n\t\t\t\t\tlet valid = true;\n\t\t\t\t\tif (input.is('[name=filename]'))\n\t\t\t\t\t\tvalid = (!input.val().includes('\\\\') && !input.val().includes('/'));\n\t\t\t\t\treturn valid;\n\t\t\t\t}, this)\n\t\t\t},\n\t\t\tmessages: {\n\t\t\t\trequiredFilename: 'Required',\n\t\t\t\tvalidFilename: 'Contains invalid characters (slash, backslash)'\n\t\t\t}\n\t\t}).data('kendoValidator');\n\n\t\t// Hide save spinner\n\t\t$('#saveSpinner', modalElement).hide();\n\n\t\t// Add listener to Save button click\n\t\t$('#saveWordFileButton', modalElement).on('click', $.proxy(this._onSaveWordFileClick, this));\n\n\t\t// Add listener to modal hide event\n\t\tmodalElement.on('hidden.bs.modal', $.proxy(this._destroyModal, this));\n\n\t\t// Initialize bootstrap modal\n\t\tmodalElement.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t});\n\t}\n\n\t_destroyModal()\n\t{\n\t\tlet modalElement = $('#editModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Remove listeners\n\t\t\t$('#saveWordFileButton', modalElement).off('click');\n\t\t\tmodalElement.off('hidden.bs.modal');\n\n\t\t\t// Destroy everything Kendo\n\t\t\tkendo.destroy(modalElement);\n\n\t\t\t// Dispose modal\n\t\t\tmodalElement.modal('dispose');\n\n\t\t\t// Remove html\n\t\t\tmodalElement.remove();\n\t\t\tmodalElement = null;\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('words-files-manager'))\n\twindow.customElements.define('words-files-manager', WordsFilesManager);\n","export class SidebarLayout extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\t// Attach a shadow root\n\t\tconst shadowRoot = this.attachShadow({mode: 'open'});\n\t\tshadowRoot.innerHTML = `\n\t\t\t<style>\n\t\t\t\t:host {\n\t\t\t\t\tdisplay: flex;\n\t\t\t\t\tflex-direction: row;\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 575.98px) {\n\t\t\t\t\t:host(.split-xs) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-xs) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 767.98px) {\n\t\t\t\t\t:host(.split-sm) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-sm) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 991.98px) {\n\t\t\t\t\t:host(.split-md) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-md) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 1199.98px) {\n\t\t\t\t\t:host(.split-lg) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-lg) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t.side-col::slotted(*) {\n\t\t\t\t}\n\n\t\t\t\t.main-col::slotted(*) {\n\t\t\t\t\tflex-grow: 1;\n\t\t\t\t}\n\t\t\t</style>\n\n\t\t\t<slot class=\"side-col\" name=\"side-column\"></slot>\n\t\t\t<slot class=\"main-col\" name=\"main-column\"></slot>\n\t\t`;\n\n\t\t// Set initial selection\n\t\tthis.selectedIndex = 0;\n\t}\n\n\tget selectedPanel()\n\t{\n\t\treturn this._selectedPanel;\n\t}\n\n\tset selectedPanel(element) // 'side' or 'main'\n\t{\n\t\tif (element != null && element.parentNode == this)\n\t\t{\n\t\t\tthis._selectedPanel = element;\n\n\t\t\tfor (let element of this.children)\n\t\t\t{\n\t\t\t\tif (element == this._selectedPanel)\n\t\t\t\t\telement.setAttribute('aria-selected', 'true');\n\t\t\t\telse\n\t\t\t\t\telement.removeAttribute('aria-selected');\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\tconsole.error('Element is not a child of SidebarLayout');\n\t\t}\n\t}\n\n\tget selectedIndex()\n\t{\n\t\treturn Array.from(this.children).indexOf(this._selectedPanel);\n\t}\n\n\tset selectedIndex(index)\n\t{\n\t\tif (this.children.length > 0)\n\t\t{\n\t\t\tif (this.children[index] == null)\n\t\t\t{\n\t\t\t\tconsole.error('Invalid SidebarLayout index');\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet element = this.children[index];\n\t\t\tthis.selectedPanel = element;\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('sidebar-layout'))\n\twindow.customElements.define('sidebar-layout', SidebarLayout);\n","import {BaseModule} from './base-module';\nimport {ViewStack} from '../components/view-stack';\nimport {SidebarLayout} from '../components/sidebar-layout';\nimport {ConfigInterfaceBuilder} from '../utils/uibuilder/config-interface-builder';\nimport {capitalizeFirst, filterClassName} from '../utils/utilities';\nimport {WordsFilesManager} from '../components/module-specific/words-files-manager';\n\nexport default class ZoneConfigurator extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('zoneConfig');\n\n\t\tthis.ITEM_TYPE_ZONE = 'zone';\n\t\tthis.ITEM_TYPE_ROOM = 'room';\n\n\t\t// Outgoing requests\n\t\tthis.REQ_GET_ZONES = 'getZones';\n\n\t\tthis.REQ_GET_ZONE_CONFIG = 'getZoneConfig';\n\t\tthis.REQ_SAVE_ZONE_CONFIG = 'saveZoneConfig';\n\t\tthis.REQ_NEW_ZONE_CONFIG = 'newZoneConfig';\n\t\tthis.REQ_DELETE_ZONE_CONFIG = 'delZoneConfig';\n\t\tthis.REQ_ACTIVATE_ZONE = 'actZone';\n\n\t\tthis.REQ_GET_ROOM_CONFIG = 'getRoomConfig';\n\t\tthis.REQ_SAVE_ROOM_CONFIG = 'saveRoomConfig';\n\t\tthis.REQ_NEW_ROOM_CONFIG = 'newRoomConfig';\n\t\tthis.REQ_DELETE_ROOM_CONFIG = 'delRoomConfig';\n\n\t\tthis.REQ_REFRESH_WORDS_FILE = 'refreshWordsFiles';\n\t\tthis.REQ_EDIT_WORDS_FILE = 'editWordsFile';\n\t\tthis.REQ_SAVE_WORDS_FILE = 'saveWordsFile';\n\t\tthis.REQ_DELETE_WORDS_FILE = 'delWordsFile';\n\n\t\t// Incoming responses\n\t\tthis.RESP_ZONES = 'zones';\n\n\t\tthis.RESP_ZONE_CONFIG = 'zoneConfig';\n\t\tthis.RESP_ZONE_CONFIG_UPDATE_CONFIRM = 'zoneCfgUpd';\n\t\tthis.RESP_ZONE_ADDED = 'zoneAdded';\n\t\tthis.RESP_ZONE_REFUSED = 'zoneRefused';\n\t\tthis.RESP_ZONE_DELETED = 'zoneDel';\n\t\tthis.RESP_ZONE_ACTIVATED = 'zoneAct';\n\t\tthis.RESP_ZONE_ACTIVATION_ERROR = 'zoneActErr';\n\n\t\tthis.RESP_ROOM_CONFIG = 'roomConfig';\n\t\tthis.RESP_ROOM_CONFIG_UPDATE_CONFIRM = 'roomCfgUpd';\n\t\tthis.RESP_ROOM_ADDED = 'roomAdded';\n\t\tthis.RESP_ROOM_REFUSED = 'roomRefused';\n\t\tthis.RESP_ROOM_DELETED = 'roomDel';\n\n\t\tthis.RESP_REFRESH_WORDS_FILES = 'refreshWordsFiles';\n\t\tthis.RESP_WORDS_FILE_CONTENT = 'wordsFile';\n\t\tthis.RESP_WORDS_FILE_ERROR = 'wordsFileErr';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t// This methods override those in BaseModule class.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Create interface builder instance\n\t\tthis._interfaceBuilder = new ConfigInterfaceBuilder();\n\n\t\t// Set listener for custom actions triggered by configuration interface\n\t\t$('#znc-tabNavigator').on('value-set', $.proxy(this._onConfigValueSet, this));\n\n\t\t// Initialize Zones/Rooms treeview\n\t\tthis._treeview = $('#znc-treeView').kendoTreeView({\n\t\t\tloadOnDemand: false,\n\t\t\tdataTextField: 'name',\n\t\t\ttemplate: kendo.template('<span class=\"# if (!item.active) { # inactive-list-item # } #\">#: item.name #</span>'),\n\t\t\tchange: $.proxy(this._onZoneRoomChange, this),\n\t\t}).data('kendoTreeView');\n\n\t\t// Listen to treeview double-click event\n\t\t$('#znc-treeView').on('dblclick', $.proxy(this._onTreeItemDoubleClick, this));\n\n\t\t// Request zones & rooms list to server instance\n\t\tthis.sendExtensionRequest(this.REQ_GET_ZONES);\n\n\t\t// Initialize progress bar\n\t\t$('#znc-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Add listeners to utility buttons\n\t\t$('#znc-addZoneButton').on('click', $.proxy(this._onAddZoneClick, this));\n\t\t$('#znc-addRoomButton').on('click', $.proxy(this._onAddRoomClick, this));\n\t\t$('#znc-editButton').on('click', $.proxy(this._onEditClick, this));\n\t\t$('#znc-removeButton').on('click', $.proxy(this._onRemoveClick, this));\n\t\t$('#znc-activateButton').on('click', $.proxy(this._onActivateClick, this));\n\n\t\t// Add listener to interface buttons\n\t\t$('#znc-cancelButton').on('click', $.proxy(this._onCancelClick, this));\n\t\t$('#znc-reloadButton').on('click', $.proxy(this._onReloadClick, this));\n\t\t$('#znc-submitButton').on('click', $.proxy(this._onSubmitClick, this));\n\n\t\t// Save ref to words files manager and add custom event listeners\n\t\tthis._wordsFilesManager = document.getElementById('znc-wordsFilesManager');\n\t\t$(this._wordsFilesManager).on(this._wordsFilesManager.REFRESH_WORDS_FILES_CLICK_EVENT, $.proxy(this._onWordsFileReloadClick, this));\n\t\t$(this._wordsFilesManager).on(this._wordsFilesManager.EDIT_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileEditClick, this));\n\t\t$(this._wordsFilesManager).on(this._wordsFilesManager.SAVE_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileSaveClick, this));\n\t\t$(this._wordsFilesManager).on(this._wordsFilesManager.REMOVE_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileRemoveClick, this));\n\t\t$(this._wordsFilesManager).on(this._wordsFilesManager.ASSIGN_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileAssignClick, this));\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t// Remove tree view doubleclick listener\n\t\t$('#znc-treeView').off('dblclick');\n\n\t\t// Remove listener for custom actions triggered by configuration interface\n\t\t$('#znc-tabNavigator').off('value-set');\n\n\t\t// Remove listener for zone/room activation event\n\t\t$('#znc-addZoneButton').off('click');\n\t\t$('#znc-addRoomButton').off('click');\n\t\t$('#znc-editButton').off('click');\n\t\t$('#znc-removeButton').off('click');\n\t\t$('#znc-activateButton').off('click');\n\n\t\t// Remove interface buttons click listeners\n\t\t$('#znc-cancelButton').off('click');\n\t\t$('#znc-reloadButton').off('click');\n\t\t$('#znc-submitButton').off('click');\n\n\t\t// Remove listeners to words files manager\n\t\t$(this._wordsFilesManager).off(this._wordsFilesManager.REFRESH_WORDS_FILES_CLICK_EVENT);\n\t\t$(this._wordsFilesManager).off(this._wordsFilesManager.EDIT_WORDS_FILE_CLICK_EVENT);\n\t\t$(this._wordsFilesManager).off(this._wordsFilesManager.SAVE_WORDS_FILE_CLICK_EVENT);\n\t\t$(this._wordsFilesManager).off(this._wordsFilesManager.REMOVE_WORDS_FILE_CLICK_EVENT);\n\t\t$(this._wordsFilesManager).off(this._wordsFilesManager.ASSIGN_WORDS_FILE_CLICK_EVENT);\n\n\t\t// Clear tabs container\n\t\tthis._clearTabs();\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\tconst username = data.getUtfString('user');\n\n\t\t/****** ZONES & ROOMS ******/\n\n\t\t// Zones & rooms list received\n\t\tif (command == this.RESP_ZONES)\n\t\t\tthis._populateTree(data);\n\n\t\t// Zone or room configuration data received\n\t\telse if (command == this.RESP_ZONE_CONFIG || command == this.RESP_ROOM_CONFIG)\n\t\t{\n\t\t\t// Build user interface based on received data\n\t\t\tthis._interfaceBuilder.buildInterface(data.getSFSArray('settings'), 'znc-tabNavigator', false);\n\n\t\t\t// Enable scrolling tabs (if needed)\n\t\t\tif (this._reinitTabs)\n\t\t\t{\n\t\t\t\t$('#znc-tabNavigator #tabs').scrollingTabs({\n\t\t\t\t\tbootstrapVersion: 4,\n\t\t\t\t\tscrollToTabEdge: true,\n\t\t\t\t\tenableSwiping: true,\n\t\t\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t\t\t});\n\t\t\t}\n\n\t\t\t// Enable interface\n\t\t\tthis._enableConfigInterface(true);\n\t\t}\n\n\t\t/****** ZONES ******/\n\n\t\t// Zone configuration update confirmation\n\t\telse if (command == this.RESP_ZONE_CONFIG_UPDATE_CONFIRM)\n\t\t{\n\t\t\t// If a 'name' parameter is received, it means the zone name changed, and we have to update the zones list\n\t\t\tif (data.getUtfString('zName') != null)\n\t\t\t\tthis._updateZoneNameInList(data.getInt('zId'), data.getUtfString('zName'));\n\n\t\t\t// If the current user is the updater, show a notification; otherwise, show a dialog box suggesting to reload\n\t\t\tif (username == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Enable interface\n\t\t\t\tthis._enableConfigInterface(true);\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Zone modified', `Zone settings updated successfully; changes will be applied on next <strong>server restart</strong>`);\n\n\t\t\t\t// Reset the 'modified' flag\n\t\t\t\tthis._interfaceBuilder.resetIsModified();\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// An alert box is displayed if the user is currently editing the same zone\n\t\t\t\tif (data.getInt('zId') == this._editedZoneId)\n\t\t\t\t{\n\t\t\t\t\t// Show alert\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(`Administrator ${username} has modified the Zone you are currently editing; please reload to update your view.`);\n\n\t\t\t\t\t// Disable submit button\n\t\t\t\t\t$('#znc-submitButton').attr('disabled', true);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tif (data.getUtfString('zName') != null)\n\t\t\t\t\t\tthis.shellCtrl.showNotification('Zone renamed', `Administrator ${username} has changed the name on one of the Zones`);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// New zone added\n\t\telse if (command == this.RESP_ZONE_ADDED)\n\t\t{\n\t\t\tconst zoneName = data.getSFSObject('zone').getUtfString('name');\n\n\t\t\t// If the current user is the updater, reset the interface; otherwise, just show a notification\n\t\t\tif (username == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Reset interface\n\t\t\t\tthis._onCancelClick();\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Zone added', `Zone '${zoneName}' created successfully`);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Zone added', `Administrator ${username} created Zone '${zoneName}'`);\n\t\t\t}\n\n\t\t\t// Add new zone to tree\n\t\t\tlet zonesDS = this._treeview.dataSource;\n\t\t\tzonesDS.add(this._createZoneObject(data.getSFSObject('zone')));\n\t\t\tzonesDS.sync();\n\t\t}\n\n\t\t// New zone creation refused due to invalid zone name\n\t\telse if (command == this.RESP_ZONE_REFUSED)\n\t\t{\n\t\t\t// Re-enable interface\n\t\t\tthis._enableConfigInterface(true);\n\n\t\t\t// Show warning\n\t\t\tthis.shellCtrl.showSimpleAlert('Zone configuration can\\'t be saved because another Zone with the same name already exists.', true);\n\t\t}\n\n\t\t// Existing zone deleted\n\t\telse if (command == this.RESP_ZONE_DELETED)\n\t\t{\n\t\t\t// If the current user is the deleter, reset the interface; otherwise, just show a notification\n\t\t\tif (username == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Re-enable interface\n\t\t\t\tthis._enableListInterface(true);\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Zone removed', `Zone '${data.getUtfString('zName')}' deleted successfully`);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// An alert box is displayed if the user is currently editing the same zone\n\t\t\t\tif (data.getInt('zId') == this._editedZoneId)\n\t\t\t\t{\n\t\t\t\t\t// Show alert\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(`Administrator ${username} has deleted the Zone you are currently modifying; you have to cancel your editing.`);\n\n\t\t\t\t\t// Disable submit and reload buttons\n\t\t\t\t\t$('#znc-reloadButton').attr('disabled', true);\n\t\t\t\t\t$('#znc-submitButton').attr('disabled', true);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tthis.shellCtrl.showNotification('Zone removed', `Administrator ${username} deleted Zone '${data.getUtfString('zName')}'`);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Reset selection if the currently selected item or its parent is being removed\n\t\t\tlet selectedNode = this._treeview.select();\n\t\t\tlet selectedDataItem = this._treeview.dataItem(selectedNode);\n\t\t\tif (selectedDataItem)\n\t\t\t{\n\t\t\t\tif (selectedDataItem.type == this.ITEM_TYPE_ZONE && selectedDataItem.id == data.getInt('zId'))\n\t\t\t\t\tthis._deselectTreeItem();\n\n\t\t\t\tif (selectedDataItem.type == this.ITEM_TYPE_ROOM)\n\t\t\t\t{\n\t\t\t\t\tlet parentDataItem = this._treeview.dataItem(this._treeview.parent(selectedNode));\n\n\t\t\t\t\tif (parentDataItem.id == data.getInt('zId'))\n\t\t\t\t\t\tthis._deselectTreeItem();\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Remove zone from tree\n\t\t\tlet dataItem = this._getZoneDataItemById(data.getInt('zId'));\n\t\t\tlet zonesDS = this._treeview.dataSource;\n\t\t\tzonesDS.remove(dataItem);\n\t\t\tzonesDS.sync();\n\t\t}\n\n\t\t// Zone activated\n\t\telse if (command == this.RESP_ZONE_ACTIVATED)\n\t\t{\n\t\t\t// Set zone activation status\n\t\t\tconst zoneName = this._setZoneActivationStatus(data.getInt('zId'), data.getUtfString('actRooms'), true);\n\n\t\t\t// Display notification\n\t\t\tif (username == this.smartFox.mySelf.name)\n\t\t\t\tthis.shellCtrl.showNotification('Zone activated', `Zone '${zoneName}' activated successfully`);\n\t\t\telse\n\t\t\t\tthis.shellCtrl.showNotification('Zone activated', `Administrator ${username} activated Zone '${zoneName}'`);\n\t\t}\n\n\t\t// Zone activation error\n\t\telse if (command == this.RESP_ZONE_ACTIVATION_ERROR)\n\t\t{\n\t\t\t// Set zone activation status\n\t\t\tthis._setZoneActivationStatus(data.getInt('zId'), '', false);\n\n\t\t\t// Show alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'), true);\n\t\t}\n\n\t\t/****** ROOMS ******/\n\n\t\t// Room configuration update confirmation\n\t\telse if (command == this.RESP_ROOM_CONFIG_UPDATE_CONFIRM)\n\t\t{\n\t\t\tif (data.getUtfString('rName') != null)\n\t\t\t\tthis._updateRoomNameInList(data.getInt('zId'), data.getInt('rId'), data.getUtfString('rName'));\n\n\t\t\t// If the current user is the updater, show a notification; otherwise, show a dialog box suggesting to reload\n\t\t\tif (username == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Enable interface\n\t\t\t\tthis._enableConfigInterface(true);\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Room modified', `Room settings updated successfully; changes will be applied on next <strong>server restart</strong>`);\n\n\t\t\t\t// Reset the 'modified' flag\n\t\t\t\tthis._interfaceBuilder.resetIsModified();\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// An alert box is displayed if the user is currently editing the same room\n\t\t\t\tif (data.getInt('rId') == this._editedRoomId)\n\t\t\t\t{\n\t\t\t\t\t// Show alert\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(`Administrator ${username} has modified the Room you are currently editing; please reload to update your view.`);\n\n\t\t\t\t\t// Disable submit button\n\t\t\t\t\t$('#znc-submitButton').attr('disabled', true);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tif (data.getUtfString('rName') != null)\n\t\t\t\t\t\tthis.shellCtrl.showNotification('Room renamed', `Administrator ${username} has changed the name on one of the Rooms`);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// New room added\n\t\telse if (command == this.RESP_ROOM_ADDED)\n\t\t{\n\t\t\tconst roomData = data.getSFSObject('room');\n\t\t\tconst zoneId = data.getInt('zId');\n\n\t\t\tlet zonesDS = this._treeview.dataSource;\n\t\t\tlet zoneItem = zonesDS.get(zoneId);\n\n\t\t\t// If the current user is the updater, reset the interface; otherwise, just show a notification\n\t\t\tif (username == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Reset interface\n\t\t\t\tthis._onCancelClick();\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Room added', `Room '${roomData.getUtfString('name')}' created successfully`);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Room added', `Administrator ${username} created Room '${roomData.getUtfString('name')}' in Zone '${zoneItem.name}'`);\n\t\t\t}\n\n\t\t\t// Add new room to tree\n\t\t\tzoneItem.append(this._createRoomObject(roomData, zoneId));\n\t\t\tzonesDS.sync();\n\n\t\t\t// Expand zone node where room was added\n\t\t\tthis._treeview.expand(this._treeview.select());\n\t\t}\n\n\t\t// New room creation refused due to invalid room name\n\t\telse if (command == this.RESP_ROOM_REFUSED)\n\t\t{\n\t\t\t// Re-enable interface\n\t\t\tthis._enableConfigInterface(true);\n\n\t\t\t// Show warning\n\t\t\tthis.shellCtrl.showSimpleAlert('Room configuration can\\'t be saved because another Room with the same name already exists.', true);\n\t\t}\n\n\t\t// Existing room deleted\n\t\telse if (command == this.RESP_ROOM_DELETED)\n\t\t{\n\t\t\tlet zoneItem = this._getZoneDataItemById(data.getInt('zId'));\n\t\t\tlet roomItem = this._getRoomDataItemById(data.getInt('zId'), data.getInt('rId'));\n\n\t\t\t// If the current user is the deleter, reset the interface; otherwise, just show a notification\n\t\t\tif (username == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Re-enable interface\n\t\t\t\tthis._enableListInterface(true);\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Room removed', `Room '${roomItem.name}' deleted successfully`);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// An alert box is displayed if the user is currently editing the same room\n\t\t\t\tif (data.getInt('rId') == this._editedRoomId)\n\t\t\t\t{\n\t\t\t\t\t// Show alert\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(`Administrator ${username} has deleted the Room you are currently modifying; you have to cancel your editing.`);\n\n\t\t\t\t\t// Disable submit and reload buttons\n\t\t\t\t\t$('#znc-reloadButton').attr('disabled', true);\n\t\t\t\t\t$('#znc-submitButton').attr('disabled', true);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tthis.shellCtrl.showNotification('Room removed', `Administrator ${username} deleted Room '${roomItem.name}' from Zone '${zoneItem.name}'`);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Reset selection if the currently selected item or its parent is being removed\n\t\t\tlet selectedNode = this._treeview.select();\n\t\t\tlet selectedDataItem = this._treeview.dataItem(selectedNode);\n\t\t\tif (selectedDataItem)\n\t\t\t{\n\t\t\t\tif (selectedDataItem.type == this.ITEM_TYPE_ROOM && selectedDataItem.id == data.getInt('rId'))\n\t\t\t\t\tthis._deselectTreeItem();\n\t\t\t}\n\n\t\t\t// Remove room from tree\n\t\t\tzoneItem.children.remove(roomItem);\n\t\t\tthis._treeview.dataSource.sync();\n\t\t}\n\n\t\t/****** WORDS FILES ******/\n\n\t\t// Words files list received\n\t\telse if (command == this.RESP_REFRESH_WORDS_FILES)\n\t\t{\n\t\t\tthis._wordsFilesManager.refreshWordsFilesList(data.getSFSArray('wf'), username == this.smartFox.mySelf.name);\n\n\t\t\t// If another user caused a refresh (for example deleting a file, or adding a new one) show a notification\n\t\t\tif (username != null && username != this.smartFox.mySelf.name && this._editedZoneId > -1)\n\t\t\t\tthis.shellCtrl.showNotification('Words files modified', `Administrator ${username} has added, modified or deleted a words file.`);\n\t\t}\n\n\t\t// Words file content received\n\t\telse if (command == this.RESP_WORDS_FILE_CONTENT)\n\t\t{\n\t\t\tthis._wordsFilesManager.editWordsFile(data.getUtfString('filename'), data.getText('content'));\n\t\t}\n\n\t\t// Words file error (edit/save)\n\t\telse if (command == this.RESP_WORDS_FILE_ERROR)\n\t\t{\n\t\t\t// Enable buttons\n\t\t\tthis._wordsFilesManager.enabled = true;\n\n\t\t\t// Show alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'), true);\n\t\t}\n\n\t\t// else if ()\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onTreeItemDoubleClick(e)\n\t{\n\t\t// Get event target's closest tree node\n\t\tlet treeNode = $(e.target).closest('.k-item[role=treeitem]');\n\n\t\t// Get associated data item\n\t\tlet dataItem = this._treeview.dataItem(treeNode);\n\n\t\t// Load configuration\n\t\tthis._loadConfiguration(dataItem.type);\n\t}\n\n\t_onZoneRoomChange()\n\t{\n\t\t// Reset utility buttons\n\t\tthis._setUtilityButtonsState(this._selectedItem);\n\t}\n\n\t// Utility buttons listeners\n\n\t_onAddZoneClick()\n\t{\n\t\t// Deselect list item\n\t\tthis._deselectTreeItem();\n\n\t\t// Load configuration\n\t\tthis._loadConfiguration(this.ITEM_TYPE_ZONE);\n\t}\n\n\t_onAddRoomClick()\n\t{\n\t\t// Select parent list item\n\t\tthis._selectParentTreeItem();\n\n\t\t// Load configuration\n\t\tthis._loadConfiguration(this.ITEM_TYPE_ROOM);\n\t}\n\n\t_onEditClick()\n\t{\n\t\t// Load configuration\n\t\tthis._loadConfiguration(this._selectedItem.type);\n\t}\n\n\t_onRemoveClick()\n\t{\n\t\tthis.shellCtrl.showConfirmWarning(`Are you sure you want to delete the selected ${this._selectedItem.type == this.ITEM_TYPE_ZONE ? 'Zone' : 'Room'} configuration?`, $.proxy(this._onRemoveConfirm, this));\n\t}\n\n\t_onRemoveConfirm()\n\t{\n\t\t// Disable zone/room selection list\n\t\tthis._enableListInterface(false);\n\n\t\tlet params = new SFS2X.SFSObject();\n\n\t\t// Request zone removal\n\t\tif (this._selectedItem.type == this.ITEM_TYPE_ZONE)\n\t\t{\n\t\t\tparams.putInt('zId', this._selectedItem.id);\n\t\t\tthis.sendExtensionRequest(this.REQ_DELETE_ZONE_CONFIG, params);\n\t\t}\n\t\telse\n\t\t{\n\t\t\tparams.putInt('zId', this._selectedItemParent.id);\n\t\t\tparams.putInt('rId', this._selectedItem.id);\n\t\t\tthis.sendExtensionRequest(this.REQ_DELETE_ROOM_CONFIG, params);\n\t\t}\n\t}\n\n\t_onActivateClick()\n\t{\n\t\t// Get selected data item\n\t\tif (this._selectedItem.type == this.ITEM_TYPE_ZONE)\n\t\t{\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putInt('zId', this._selectedItem.id);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_ACTIVATE_ZONE, params);\n\t\t}\n\t}\n\n\t// Configuration buttons listeners\n\n\t_onCancelClick()\n\t{\n\t\t// Enable zone/room selection lists\n\t\tthis._enableListInterface(true);\n\n\t\t// Disable configuration interface\n\t\tthis._enableConfigInterface(false);\n\n\t\t// Clear main container\n\t\tthis._resetTabsContainer(false, true);\n\n\t\t// Set isEditing flag\n\t\tthis._isEditing = false;\n\t\tthis._editedItemType = '';\n\n\t\t// Switch panel\n\t\tthis._switchPanel('znc-sidebarPanel');\n\t}\n\n\t_onReloadClick()\n\t{\n\t\t// Hide validation messages\n\t\tthis._interfaceBuilder.resetValidation();\n\n\t\t// Reload configuration\n\t\tthis._loadConfiguration(this._editedItemType, false);\n\t}\n\n\t_onSubmitClick()\n\t{\n\t\t// Check validity\n\t\tif (this._interfaceBuilder.checkIsValid())\n\t\t{\n\t\t\tlet changes = this._interfaceBuilder.getChangedData();\n\n\t\t\tif (changes.size() > 0)\n\t\t\t{\n\t\t\t\t//console.log(changes.getDump())\n\t\t\t\t//return;\n\n\t\t\t\t// In case the zone/room name changed, check it against the list (duplicate names not allowed!)\n\t\t\t\tif (this._validateName(changes))\n\t\t\t\t{\n\t\t\t\t\t// Disable configuration interface\n\t\t\t\t\tthis._enableConfigInterface(false);\n\n\t\t\t\t\t// Send settings to server instance\n\t\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\t\tparams.putSFSArray('settings', changes);\n\t\t\t\t\tparams.putBool('backup', $('#znc-backupCheck').prop('checked'));\n\t\t\t\t\tparams.putInt('zId', this._editedZoneId);\n\t\t\t\t\tparams.putInt('rId', this._editedRoomId);\n\n\t\t\t\t\tif (this._editedItemType == this.ITEM_TYPE_ZONE)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit zone settings\n\t\t\t\t\t\tif (this._editedZoneId > -1)\n\t\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SAVE_ZONE_CONFIG, params);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_NEW_ZONE_CONFIG, params);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit room settings\n\t\t\t\t\t\tif (this._editedRoomId > -1)\n\t\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SAVE_ROOM_CONFIG, params);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_NEW_ROOM_CONFIG, params);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Show alert\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(`Unable to submit configuration because the ${capitalizeFirst(this._editedItemType)} name already exists; duplicate names are not allowed.`, true);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Show alert\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit configuration changes due to an invalid value; please verify the highlighted form fields in all tabs.', true);\n\t\t}\n\t}\n\n\t_onConfigValueSet(e) // SAME METHOD DUPLICATED IN zone-monitor.js\n\t{\n\t\tconst configParam = e.target.data;\n\n\t\t// Handle extension name/type dropdowns update and update the main class dropdown datasource accordingly\n\t\tif (configParam.name == 'extension.name' || configParam.name == 'extension.type' || configParam.name == 'extension.filterClass')\n\t\t{\n\t\t\t// All involved ConfigFormItems must be available and initialized to proceed\n\t\t\tconst nameFormItem = this._interfaceBuilder.getConfigFormItem('extension.name');\n\t\t\tconst typeFormItem = this._interfaceBuilder.getConfigFormItem('extension.type');\n\t\t\tconst classFormItem = this._interfaceBuilder.getConfigFormItem('extension.file');\n\t\t\tconst filterFormItem = this._interfaceBuilder.getConfigFormItem('extension.filterClass');\n\n\t\t\tif (nameFormItem != null && typeFormItem != null && classFormItem != null && filterFormItem != null)\n\t\t\t{\n\t\t\t\tconst source = nameFormItem.data;\n\t\t\t\tlet classesList = [];\n\n\t\t\t\tlet data = source.triggerData;\n\t\t\t\tfor (let i = 0; i < data.size(); i++)\n\t\t\t\t{\n\t\t\t\t\tlet ext = data.getSFSObject(i);\n\n\t\t\t\t\tif (ext.getUtfString('name') == nameFormItem.data.value && ext.getUtfString('type') == typeFormItem.data.value)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet classes = ext.get('classesString').split(','); // We don't use \"getUTfString\" because the type is Text in case a very large list of classes is returned\n\n\t\t\t\t\t\tif (filterFormItem.data.value == true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet filteredClasses = classes.filter(filterClassName);\n\t\t\t\t\t\t\tclasses = filteredClasses;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tclassesList = classesList.concat(classes);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet currentClass = classFormItem.data.value;\n\n\t\t\t\t// If the classes list doesn't contain the current value, create an empty entry and reset the value\n\t\t\t\tif (classesList.indexOf(currentClass) < 0)\n\t\t\t\t{\n\t\t\t\t\tif (classesList.length == 0)\n\t\t\t\t\t{\n\t\t\t\t\t\tclassesList.push('');\n\t\t\t\t\t\tcurrentClass = '';\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tcurrentClass = classesList[0];\n\t\t\t\t}\n\n\t\t\t\tlet mainClassDropDown = classFormItem._innerWidget;\n\t\t\t\tmainClassDropDown.setDataSource(classesList);\n\n\t\t\t\tclassFormItem.data.value = currentClass;\n\t\t\t\tclassFormItem._setWidgetValue();\n\t\t\t}\n\t\t}\n\t}\n\n\t_onWordsFileReloadClick(evt)\n\t{\n\t\t// Send request to server\n\t\tthis.sendExtensionRequest(this.REQ_REFRESH_WORDS_FILE);\n\t}\n\n\t_onWordsFileEditClick(evt)\n\t{\n\t\t// Send request to server\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putUtfString('filename', evt.detail);\n\t\tthis.sendExtensionRequest(this.REQ_EDIT_WORDS_FILE, params);\n\t}\n\n\t_onWordsFileSaveClick(evt)\n\t{\n\t\tthis._tempWordsFileData = evt.detail;\n\n\t\t// Check if a new file is being created\n\t\tif (this._tempWordsFileData.isNew)\n\t\t{\n\t\t\t// If yes, check if name already exists\n\t\t\tif (this._wordsFilesManager.getExistingFilenames().includes(this._tempWordsFileData.filename))\n\t\t\t{\n\t\t\t\t// Show confirm dialog\n\t\t\t\tthis.shellCtrl.showConfirmWarning('A words file with the entered name already exists; do you want to proceed anyway? The existing file will be overwritten.', $.proxy(this._onWordsFileSaveConfirm, this));\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\n\t\t// Proceed\n\t\tthis._onWordsFileSaveConfirm();\n\t}\n\n\t_onWordsFileSaveConfirm()\n\t{\n\t\t// Disable words files manager buttons\n\t\tthis._wordsFilesManager.enabled = false;\n\t\tthis._wordsFilesManager.saveSpinnerVisible = true;\n\n\t\t// Send request to server\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putUtfString('filename', this._tempWordsFileData.filename);\n\t\tparams.putText('content', this._tempWordsFileData.content);\n\t\tthis.sendExtensionRequest(this.REQ_SAVE_WORDS_FILE, params);\n\t}\n\n\t_onWordsFileRemoveClick(evt)\n\t{\n\t\tthis.shellCtrl.showConfirmWarning('Are you sure you want to delete the selected words file?', $.proxy(this._onWordsFileRemoveConfirm, this));\n\t}\n\n\t_onWordsFileRemoveConfirm()\n\t{\n\t\tlet wordsFile = this._wordsFilesManager.getSelectedWordsFileName();\n\n\t\tif (wordsFile != null)\n\t\t{\n\t\t\t// Disable words files manager buttons\n \t\t\tthis._wordsFilesManager.enabled = false;\n\t\t\tthis._wordsFilesManager.actionSpinnerVisible = true;\n\n\t\t\t// Send request to server\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('filename', wordsFile);\n\t\t\tthis.sendExtensionRequest(this.REQ_DELETE_WORDS_FILE, params);\n\t\t}\n\t}\n\n\t_onWordsFileAssignClick(evt)\n\t{\n\t\tlet path = evt.detail;\n\n\t\t// Write path of the selected words file in \"wordsFilter.wordsFile\" dynamically created field\n\t\tconst wordsFileFormItem = this._interfaceBuilder.getConfigFormItem('wordsFilter.wordsFile');\n\t\twordsFileFormItem.data.value = path;\n\t\twordsFileFormItem._setWidgetValue();\n\t}\n\n\t//---------------------------------\n\t// PRIVATE METHODS\n\t//---------------------------------\n\n\t_enableListInterface(enabled)\n\t{\n\t\t$('#znc-utilButtons').attr('disabled', !enabled);\n\t\t$('#znc-treeView').attr('disabled', !enabled);\n\t}\n\n\t_setUtilityButtonsState(dataItem = null)\n\t{\n\t\tlet disable = true;\n\n\t\tif (dataItem)\n\t\t{\n\t\t\t// Enable 'activate zone' button if zone is inactive\n\t\t\t$('#znc-activateButton').attr('disabled', (dataItem.type != this.ITEM_TYPE_ZONE || dataItem.active));\n\n\t\t\tdisable = false;\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Disable 'activate zone' button\n\t\t\t$('#znc-activateButton').attr('disabled', true);\n\t\t}\n\n\t\t// Enable/disable other utility buttons\n\t\t$('#znc-addZoneButton').attr('disabled', false); // Always enabled\n\t\t$('#znc-addRoomButton').attr('disabled', disable);\n\t\t$('#znc-editButton').attr('disabled', disable);\n\t\t$('#znc-removeButton').attr('disabled', disable);\n\t}\n\n\t_enableConfigInterface(enabled)\n\t{\n\t\t$('#znc-cancelButton').attr('disabled', !enabled);\n\t\t$('#znc-reloadButton').attr('disabled', !enabled);\n\t\t$('#znc-submitButton').attr('disabled', !enabled);\n\t\t$('#znc-backupCheck').attr('disabled', !enabled);\n\n\t\tthis._interfaceBuilder.disableInterface(!enabled);\n\n\t\t// Also switch view when enabled\n\t\tif (enabled)\n\t\t\tthis._switchView('znc-main');\n\t}\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('znc-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_clearTabs()\n\t{\n\t\t// Destroy scrolling tabs\n\t\t$('#znc-tabNavigator #tabs').scrollingTabs('destroy');\n\n\t\t// Remove all tab navigator content\n\t\tthis._interfaceBuilder.destroyInterface();\n\n\t\t// Set flag to re-initialize tabs if needed\n\t\tthis._reinitTabs = true;\n\t}\n\n\t_populateTree(data)\n\t{\n\t\tlet zData = data.getSFSArray('zones');\n\n\t\tlet zonesArr = [];\n\t\tfor (let z = 0; z < zData.size(); z++)\n\t\t\tzonesArr.push( this._createZoneObject(zData.getSFSObject(z)) );\n\n\t\t// Create datasource\n\t\tlet zones = new kendo.data.HierarchicalDataSource({\n            data: zonesArr,\n\t\t\tsort: {\n\t\t\t\tfield: 'name',\n\t\t\t\tdir: 'asc'\n\t\t\t},\n            schema: {\n                model: {\n\t\t\t\t\tid: 'id',\n                    children: {\n\t\t\t\t\t\tschema: {\n\t\t\t\t\t\t\tdata: 'rooms',\n\t\t\t\t\t\t\tsort: {\n\t\t\t\t\t\t\t\tfield: 'name',\n\t\t\t\t\t\t\t\tdir: 'asc'\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n                }\n            }\n        });\n\n\t\t// Set tree view dataprovider\n\t\tthis._treeview.setDataSource(zones);\n\n\t\t// Set utility buttons state (add, remove, edit, etc)\n\t\tthis._setUtilityButtonsState();\n\t}\n\n\t_createZoneObject(zoneData)\n\t{\n\t\tlet zone = {\n\t\t\ttype: this.ITEM_TYPE_ZONE,\n\t\t\tname: zoneData.getUtfString('name'),\n\t\t\tid: zoneData.getInt('id'),\n\t\t\tactive: zoneData.getBool('act')\n\t\t}\n\n\t\t// Create rooms list dataprovider\n\t\tlet rData = zoneData.getSFSArray('rooms');\n\n\t\tlet roomsArr = [];\n\t\tfor (let r = 0; r < rData.size(); r++)\n\t\t\troomsArr.push( this._createRoomObject(rData.getSFSObject(r), zoneData.getInt('id')) );\n\n\t\tzone.rooms = roomsArr;\n\n\t\treturn zone;\n\t}\n\n\t_createRoomObject(roomData, zoneId)\n\t{\n\t\tlet room = {\n\t\t\ttype: this.ITEM_TYPE_ROOM,\n\t\t\tname: roomData.getUtfString('name'),\n\t\t\tid: roomData.getInt('id'),\n\t\t\tactive: roomData.getBool('act'),\n\t\t\tparentId: zoneId,\n\t\t\tspriteCssClass: this._getRoomListIconCssClass(roomData.getBool('act'))\n\t\t};\n\n\t\treturn room;\n\t}\n\n\t_getRoomListIconCssClass(isActive)\n\t{\n\t\treturn isActive ? 'fas fa-door-open' : 'fas fa-door-closed inactive-list-item';\n\t}\n\n\t_setZoneActivationStatus(zoneId, activeRooms, isActive)\n\t{\n\t\tlet zoneDI = this._getZoneDataItemById(zoneId);\n\n\t\tzoneDI.active = isActive;\n\n\t\tlet activeRoomsArr = activeRooms.split(',');\n\n\t\tif (zoneDI.hasChildren)\n\t\t{\n\t\t\tfor (let i = 0; i < zoneDI.children.data().length; i++)\n\t\t\t{\n\t\t\t\tlet room = zoneDI.children.data()[i];\n\t\t\t\troom.active = (isActive && activeRoomsArr.indexOf(room.name) > -1);\n\t\t\t\troom.spriteCssClass = this._getRoomListIconCssClass(room.active)\n\t\t\t}\n\t\t}\n\n\t\t// Refresh list\n\t\tthis._treeview.dataSource.sync();\n\n\t\t// Return zone name\n\t\treturn zoneDI.name;\n\t}\n\n\t_deselectTreeItem()\n\t{\n\t\tthis._treeview.select($());\n\t}\n\n\t_selectParentTreeItem()\n\t{\n\t\tlet selectedNode = this._treeview.select();\n\t\tlet selectedDataItem = this._treeview.dataItem(selectedNode);\n\n\t\tif (selectedDataItem.type == this.ITEM_TYPE_ROOM)\n\t\t{\n\t\t\tlet parentNode = this._treeview.parent(selectedNode);\n\t\t\tthis._treeview.select(parentNode);\n\t\t}\n\t}\n\n\t_loadConfiguration(type, resetTabs = true)\n\t{\n\t\t// Disable zone/room selection list\n\t\tthis._enableListInterface(false);\n\n\t\t// Disable configuration interface\n\t\tthis._enableConfigInterface(false);\n\n\t\t// Clear main container\n\t\tthis._resetTabsContainer(true, resetTabs);\n\n\t\t// Set isEditing flag\n\t\tthis._isEditing = true;\n\t\tthis._editedItemType = type;\n\n\t\t// Request zone or room configuration data to server instance\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putInt('zId', this._editedZoneId);\n\t\tparams.putInt('rId', this._editedRoomId);\n\n\t\t// If no room is selected, then we are editing a zone\n\t\tif (this._editedItemType == this.ITEM_TYPE_ZONE)\n\t\t\tthis.sendExtensionRequest(this.REQ_GET_ZONE_CONFIG, params);\n\t\telse\n\t\t\tthis.sendExtensionRequest(this.REQ_GET_ROOM_CONFIG, params);\n\n\t\t// Switch panel\n\t\tthis._switchPanel('znc-mainPanel');\n\t}\n\n\t_resetTabsContainer(isLoading, resetTabs)\n\t{\n\t\tif (resetTabs)\n\t\t\tthis._clearTabs();\n\t\telse\n\t\t\tthis._reinitTabs = false;\n\n\t\tif (!isLoading)\n\t\t\tthis._switchView('znc-select');\n\t\telse\n\t\t\tthis._switchView('znc-loading');\n\t}\n\n\t_switchPanel(panelId)\n\t{\n\t\tdocument.getElementById('znc-view').selectedPanel = document.getElementById(panelId);\n\t}\n\n\t_getZoneDataItemById(zoneId)\n\t{\n\t\tlet zonesDS = this._treeview.dataSource;\n\t\treturn zonesDS.get(zoneId);\n\t}\n\n\t_getRoomDataItemById(zoneId, roomId)\n\t{\n\t\tlet zoneDI = this._getZoneDataItemById(zoneId);\n\n\t\tif (zoneDI.hasChildren)\n\t\t\treturn zoneDI.children.get(roomId);\n\n\t\treturn null;\n\t}\n\n\t_updateZoneNameInList(zoneId, zoneName)\n\t{\n\t\tthis._getZoneDataItemById(zoneId).name = zoneName;\n\t\tthis._treeview.dataSource.sync();\n\t}\n\n\t_updateRoomNameInList(zoneId, roomId, roomName)\n\t{\n\t\tthis._getRoomDataItemById(zoneId, roomId).name = roomName;\n\t\tthis._treeview.dataSource.sync();\n\t}\n\n\t_validateName(changes)\n\t{\n\t\tconst zoneId = this._editedZoneId;\n\n\t\tfor (let i = 0; i < changes.size(); i++)\n\t\t{\n\t\t\tconst setting = changes.getSFSObject(i);\n\n\t\t\tif (setting.containsKey('name') && setting.getUtfString('name') == 'name')\n\t\t\t{\n\t\t\t\t// Get name value\n\t\t\t\tconst name = setting.getText('value');\n\n\t\t\t\t// Get data source\n\t\t\t\tlet ds = [];\n\n\t\t\t\tif (this._editedItemType == this.ITEM_TYPE_ZONE)\n\t\t\t\t\tds = this._treeview.dataSource.data();\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\tif (this._getZoneDataItemById(zoneId).hasChildren)\n\t\t\t\t\t\tds = this._getZoneDataItemById(zoneId).children.data();\n\t\t\t\t}\n\n\n\t\t\t\t// Check if name exists in data source\n\t\t\t\tfor (let j = 0; j < ds.length; j++)\n\t\t\t\t{\n\t\t\t\t\tif (ds[j].name == name)\n\t\t\t\t\t{\n\t\t\t\t\t\treturn false;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\tget _selectedItem()\n\t{\n\t\treturn this._treeview.dataItem(this._treeview.select());\n\t}\n\n\tget _selectedItemParent()\n\t{\n\t\tlet selectedNode = this._treeview.select();\n\t\tlet parentNode = this._treeview.parent(selectedNode);\n\n\t\treturn this._treeview.dataItem(parentNode);\n\t}\n\n\tget _editedZoneId()\n\t{\n\t\tif (this._isEditing && this._selectedItem)\n\t\t{\n\t\t\tif (this._selectedItem.type == this.ITEM_TYPE_ZONE)\n\t\t\t\treturn this._selectedItem.id;\n\t\t\telse\n\t\t\t\treturn this._selectedItemParent.id;\n\t\t}\n\n\t\treturn -1;\n\t}\n\n\tget _editedRoomId()\n\t{\n\t\tif (this._isEditing && this._selectedItem)\n\t\t{\n\t\t\tif (this._selectedItem.type == this.ITEM_TYPE_ROOM)\n\t\t\t\treturn this._selectedItem.id;\n\t\t}\n\n\t\treturn -1;\n\t}\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACpcA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACvHA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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":""}