rubyfox-server 2.16.0.3 → 2.17.3.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (216) hide show
  1. checksums.yaml +4 -4
  2. data/lib/rubyfox/server/data/config/admin/admintool.xml +0 -0
  3. data/lib/rubyfox/server/data/config/admin/descriptors/config_room.txt +0 -0
  4. data/lib/rubyfox/server/data/config/admin/descriptors/config_server.txt +0 -0
  5. data/lib/rubyfox/server/data/config/admin/descriptors/config_zone.txt +0 -0
  6. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_game.txt +0 -0
  7. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_mmo.txt +0 -0
  8. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_room.txt +0 -0
  9. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_user.txt +0 -0
  10. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_zone.txt +0 -0
  11. data/lib/rubyfox/server/data/config/admin/gmc/gmc.py +0 -0
  12. data/lib/rubyfox/server/data/config/admin/icons/Analytics.png +0 -0
  13. data/lib/rubyfox/server/data/config/admin/icons/Analytics.svg +0 -0
  14. data/lib/rubyfox/server/data/config/admin/icons/BanManager.png +0 -0
  15. data/lib/rubyfox/server/data/config/admin/icons/BanManager.svg +0 -0
  16. data/lib/rubyfox/server/data/config/admin/icons/BlueBoxMonitor.png +0 -0
  17. data/lib/rubyfox/server/data/config/admin/icons/BlueBoxMonitor.svg +0 -0
  18. data/lib/rubyfox/server/data/config/admin/icons/Console.png +0 -0
  19. data/lib/rubyfox/server/data/config/admin/icons/Console.svg +0 -0
  20. data/lib/rubyfox/server/data/config/admin/icons/Dashboard.png +0 -0
  21. data/lib/rubyfox/server/data/config/admin/icons/Dashboard.svg +0 -0
  22. data/lib/rubyfox/server/data/config/admin/icons/ExtensionManager.png +0 -0
  23. data/lib/rubyfox/server/data/config/admin/icons/ExtensionManager.svg +0 -0
  24. data/lib/rubyfox/server/data/config/admin/icons/LicenseManager.png +0 -0
  25. data/lib/rubyfox/server/data/config/admin/icons/LicenseManager.svg +0 -0
  26. data/lib/rubyfox/server/data/config/admin/icons/LogViewer.png +0 -0
  27. data/lib/rubyfox/server/data/config/admin/icons/LogViewer.svg +0 -0
  28. data/lib/rubyfox/server/data/config/admin/icons/ServerConfigurator.png +0 -0
  29. data/lib/rubyfox/server/data/config/admin/icons/ServerConfigurator.svg +0 -0
  30. data/lib/rubyfox/server/data/config/admin/icons/ServletManager.png +0 -0
  31. data/lib/rubyfox/server/data/config/admin/icons/ServletManager.svg +0 -0
  32. data/lib/rubyfox/server/data/config/admin/icons/ZoneConfigurator.png +0 -0
  33. data/lib/rubyfox/server/data/config/admin/icons/ZoneConfigurator.svg +0 -0
  34. data/lib/rubyfox/server/data/config/admin/icons/ZoneMonitor.png +0 -0
  35. data/lib/rubyfox/server/data/config/admin/icons/ZoneMonitor.svg +0 -0
  36. data/lib/rubyfox/server/data/config/logParser.properties +17 -28
  37. data/lib/rubyfox/server/data/lib/BlueBox.war +0 -0
  38. data/lib/rubyfox/server/data/lib/apache-tomcat/NOTICE +1 -1
  39. data/lib/rubyfox/server/data/lib/apache-tomcat/README.md +2 -0
  40. data/lib/rubyfox/server/data/lib/apache-tomcat/RELEASE-NOTES +2 -2
  41. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/bootstrap.jar +0 -0
  42. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina.bat +2 -0
  43. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina.sh +4 -6
  44. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon-native.tar.gz +0 -0
  45. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon.jar +0 -0
  46. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-juli.jar +0 -0
  47. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-native.tar.gz +0 -0
  48. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.policy +6 -1
  49. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.properties +1 -0
  50. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/keystore.jks +0 -0
  51. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/annotations-api.jar +0 -0
  52. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ant.jar +0 -0
  53. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ha.jar +0 -0
  54. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ssi.jar +0 -0
  55. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-storeconfig.jar +0 -0
  56. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-tribes.jar +0 -0
  57. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina.jar +0 -0
  58. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/ecj-4.20.jar +0 -0
  59. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/el-api.jar +0 -0
  60. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper-el.jar +0 -0
  61. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper.jar +0 -0
  62. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jaspic-api.jar +0 -0
  63. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jsp-api.jar +0 -0
  64. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/servlet-api.jar +0 -0
  65. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-api.jar +0 -0
  66. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-coyote.jar +0 -0
  67. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-dbcp.jar +0 -0
  68. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-cs.jar +0 -0
  69. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-de.jar +0 -0
  70. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-es.jar +0 -0
  71. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-fr.jar +0 -0
  72. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ja.jar +0 -0
  73. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ko.jar +0 -0
  74. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-pt-BR.jar +0 -0
  75. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ru.jar +0 -0
  76. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-zh-CN.jar +0 -0
  77. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jdbc.jar +0 -0
  78. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jni.jar +0 -0
  79. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util-scan.jar +0 -0
  80. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util.jar +0 -0
  81. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-websocket.jar +0 -0
  82. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/websocket-api.jar +0 -0
  83. data/lib/rubyfox/server/data/lib/commons-compress-1.19.jar +0 -0
  84. data/lib/rubyfox/server/data/lib/sfs2x-admin.jar +0 -0
  85. data/lib/rubyfox/server/data/lib/sfs2x-core.jar +0 -0
  86. data/lib/rubyfox/server/data/lib/sfs2x.jar +0 -0
  87. data/lib/rubyfox/server/data/sfs2x-service +294 -159
  88. data/lib/rubyfox/server/data/sfs2x.sh +1 -1
  89. data/lib/rubyfox/server/data/www/BlueBox.war +0 -0
  90. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/_changes-to-default-themes_.txt +0 -0
  91. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/bootstrap.min.css +0 -0
  92. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/custom-bootstrap.min.css +0 -0
  93. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/css/all.min.css +0 -0
  94. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-brands-400.eot +0 -0
  95. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-brands-400.svg +0 -0
  96. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-brands-400.ttf +0 -0
  97. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-brands-400.woff +0 -0
  98. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-brands-400.woff2 +0 -0
  99. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-regular-400.eot +0 -0
  100. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-regular-400.svg +0 -0
  101. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-regular-400.ttf +0 -0
  102. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-regular-400.woff +0 -0
  103. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-regular-400.woff2 +0 -0
  104. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-solid-900.eot +0 -0
  105. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-solid-900.svg +0 -0
  106. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-solid-900.ttf +0 -0
  107. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-solid-900.woff +0 -0
  108. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fontawesome/webfonts/fa-solid-900.woff2 +0 -0
  109. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/fonts.css +0 -0
  110. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/jquery.scrolling-tabs.min.css +0 -0
  111. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/kendo.custom-bootstrap-v4.css +0 -0
  112. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/reset.css +0 -0
  113. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/style.css +3 -1
  114. data/lib/rubyfox/server/data/www/ROOT/admin/assets/data/countries.geo.json +0 -0
  115. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/FreeMono.otf +0 -0
  116. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/FreeMono.ttf +0 -0
  117. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/FreeMono.woff +0 -0
  118. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/FreeMono.woff2 +0 -0
  119. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-300.ttf +0 -0
  120. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-300.woff +0 -0
  121. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-300.woff2 +0 -0
  122. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-300italic.ttf +0 -0
  123. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-300italic.woff +0 -0
  124. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-300italic.woff2 +0 -0
  125. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-600.ttf +0 -0
  126. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-600.woff +0 -0
  127. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-600.woff2 +0 -0
  128. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-600italic.ttf +0 -0
  129. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-600italic.woff +0 -0
  130. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-600italic.woff2 +0 -0
  131. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-700.ttf +0 -0
  132. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-700.woff +0 -0
  133. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-700.woff2 +0 -0
  134. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-700italic.ttf +0 -0
  135. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-700italic.woff +0 -0
  136. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-700italic.woff2 +0 -0
  137. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-800.ttf +0 -0
  138. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-800.woff +0 -0
  139. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-800.woff2 +0 -0
  140. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-800italic.ttf +0 -0
  141. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-800italic.woff +0 -0
  142. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-800italic.woff2 +0 -0
  143. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-italic.ttf +0 -0
  144. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-italic.woff +0 -0
  145. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-italic.woff2 +0 -0
  146. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-regular.ttf +0 -0
  147. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-regular.woff +0 -0
  148. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/open-sans-v15-cyrillic_greek_cyrillic-ext_vietnamese_latin_latin-ext_greek-ext-regular.woff2 +0 -0
  149. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/roboto-condensed-v16-latin-300.ttf +0 -0
  150. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/roboto-condensed-v16-latin-300.woff +0 -0
  151. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/roboto-condensed-v16-latin-300.woff2 +0 -0
  152. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/roboto-condensed-v16-latin-700.ttf +0 -0
  153. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/roboto-condensed-v16-latin-700.woff +0 -0
  154. data/lib/rubyfox/server/data/www/ROOT/admin/assets/fonts/roboto-condensed-v16-latin-700.woff2 +0 -0
  155. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/compare.png +0 -0
  156. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/demographicsChart.png +0 -0
  157. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/demographicsCompare.png +0 -0
  158. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/demographicsFilter.png +0 -0
  159. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/demographicsMap.png +0 -0
  160. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/demographicsTable.png +0 -0
  161. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/technologyChart.png +0 -0
  162. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/technologyCompare.png +0 -0
  163. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/technologyFilter.png +0 -0
  164. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/analytics-adv/technologyTable.png +0 -0
  165. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/apple-touch-icon-180x180.png +0 -0
  166. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/favicon-16x16.png +0 -0
  167. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/favicon-32x32.png +0 -0
  168. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/favicon-64x64.png +0 -0
  169. data/lib/rubyfox/server/data/www/ROOT/admin/assets/images/logo.png +0 -0
  170. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/application.bundle.js +4 -3
  171. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-0.bundle.js +0 -0
  172. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-1.bundle.js +96 -4
  173. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-10.bundle.js +9 -11
  174. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-11.bundle.js +0 -0
  175. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12.bundle.js +604 -2
  176. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-13~module-9.bundle.js +18 -2
  177. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13.bundle.js +19 -2
  178. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-3.bundle.js +0 -0
  179. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-4.bundle.js +0 -0
  180. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-5.bundle.js +0 -0
  181. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-6.bundle.js +10 -12
  182. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-7.bundle.js +0 -0
  183. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-8.bundle.js +0 -0
  184. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-9.bundle.js +4 -6
  185. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-0.bundle.js +0 -0
  186. 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 +0 -0
  187. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-1.bundle.js +983 -0
  188. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-9.bundle.js +0 -0
  189. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/bootstrap.min.js +0 -0
  190. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/jquery-3.3.1.min.js +0 -0
  191. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/jquery.scrolling-tabs.min.js +0 -0
  192. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/js.cookie.min.js +0 -0
  193. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/kendo.all.min.js +0 -0
  194. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/popper.min.js +0 -0
  195. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/sfs2x-api-1.7.15.js +0 -0
  196. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/libs/versions-readme.txt +0 -0
  197. data/lib/rubyfox/server/data/www/ROOT/admin/index.html +0 -0
  198. data/lib/rubyfox/server/data/www/ROOT/admin/manifest.json +0 -0
  199. data/lib/rubyfox/server/data/www/ROOT/admin/modules/analytics.html +0 -0
  200. data/lib/rubyfox/server/data/www/ROOT/admin/modules/ban-manager.html +6 -0
  201. data/lib/rubyfox/server/data/www/ROOT/admin/modules/blue-box-monitor.html +0 -0
  202. data/lib/rubyfox/server/data/www/ROOT/admin/modules/console.html +0 -0
  203. data/lib/rubyfox/server/data/www/ROOT/admin/modules/dashboard.html +0 -0
  204. data/lib/rubyfox/server/data/www/ROOT/admin/modules/extension-manager.html +0 -0
  205. data/lib/rubyfox/server/data/www/ROOT/admin/modules/license-manager.html +0 -0
  206. data/lib/rubyfox/server/data/www/ROOT/admin/modules/log-viewer.html +0 -0
  207. data/lib/rubyfox/server/data/www/ROOT/admin/modules/server-configurator.html +0 -0
  208. data/lib/rubyfox/server/data/www/ROOT/admin/modules/servlet-manager.html +0 -0
  209. data/lib/rubyfox/server/data/www/ROOT/admin/modules/template.html +5 -0
  210. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-configurator.html +11 -1
  211. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-monitor.html +8 -0
  212. data/lib/rubyfox/server/data/zones/BasicExamples.zone.xml +147 -0
  213. data/lib/rubyfox/server/version.rb +1 -1
  214. metadata +9 -7
  215. data/lib/rubyfox/server/data/lib/apache-tomcat/.RELEASE-NOTES.swp +0 -0
  216. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/ecj-4.15.jar +0 -0
@@ -1,6 +1,475 @@
1
1
  /*! (c) gotoAndPlay | All rights reserved */
2
2
  (window["webpackJsonpapplication"] = window["webpackJsonpapplication"] || []).push([["module-12"],{
3
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
+
4
473
  /***/ "./src/components/sidebar-layout.js":
5
474
  /*!******************************************!*\
6
475
  !*** ./src/components/sidebar-layout.js ***!
@@ -150,6 +619,8 @@ __webpack_require__.r(__webpack_exports__);
150
619
  /* harmony import */ var _components_sidebar_layout__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../components/sidebar-layout */ "./src/components/sidebar-layout.js");
151
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");
152
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
+
153
624
 
154
625
 
155
626
 
@@ -179,6 +650,11 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
179
650
  this.REQ_NEW_ROOM_CONFIG = 'newRoomConfig';
180
651
  this.REQ_DELETE_ROOM_CONFIG = 'delRoomConfig';
181
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
+
182
658
  // Incoming responses
183
659
  this.RESP_ZONES = 'zones';
184
660
 
@@ -195,6 +671,10 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
195
671
  this.RESP_ROOM_ADDED = 'roomAdded';
196
672
  this.RESP_ROOM_REFUSED = 'roomRefused';
197
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';
198
678
  }
199
679
 
200
680
  //------------------------------------
@@ -223,7 +703,7 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
223
703
  change: $.proxy(this._onZoneRoomChange, this),
224
704
  }).data('kendoTreeView');
225
705
 
226
- // Lestend to treeview double-click event
706
+ // Listen to treeview double-click event
227
707
  $('#znc-treeView').on('dblclick', $.proxy(this._onTreeItemDoubleClick, this));
228
708
 
229
709
  // Request zones & rooms list to server instance
@@ -251,6 +731,14 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
251
731
  $('#znc-cancelButton').on('click', $.proxy(this._onCancelClick, this));
252
732
  $('#znc-reloadButton').on('click', $.proxy(this._onReloadClick, this));
253
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));
254
742
  }
255
743
 
256
744
  destroy()
@@ -276,6 +764,13 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
276
764
  $('#znc-reloadButton').off('click');
277
765
  $('#znc-submitButton').off('click');
278
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
+
279
774
  // Clear tabs container
280
775
  this._clearTabs();
281
776
  }
@@ -598,6 +1093,34 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
598
1093
  this._treeview.dataSource.sync();
599
1094
  }
600
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
+
601
1124
  // else if ()
602
1125
  }
603
1126
 
@@ -832,6 +1355,85 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
832
1355
  }
833
1356
  }
834
1357
 
1358
+ _onWordsFileReloadClick(evt)
1359
+ {
1360
+ // Send request to server
1361
+ this.sendExtensionRequest(this.REQ_REFRESH_WORDS_FILE);
1362
+ }
1363
+
1364
+ _onWordsFileEditClick(evt)
1365
+ {
1366
+ // Send request to server
1367
+ let params = new SFS2X.SFSObject();
1368
+ params.putUtfString('filename', evt.detail);
1369
+ this.sendExtensionRequest(this.REQ_EDIT_WORDS_FILE, params);
1370
+ }
1371
+
1372
+ _onWordsFileSaveClick(evt)
1373
+ {
1374
+ this._tempWordsFileData = evt.detail;
1375
+
1376
+ // Check if a new file is being created
1377
+ if (this._tempWordsFileData.isNew)
1378
+ {
1379
+ // If yes, check if name already exists
1380
+ if (this._wordsFilesManager.getExistingFilenames().includes(this._tempWordsFileData.filename))
1381
+ {
1382
+ // Show confirm dialog
1383
+ 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));
1384
+ return;
1385
+ }
1386
+ }
1387
+
1388
+ // Proceed
1389
+ this._onWordsFileSaveConfirm();
1390
+ }
1391
+
1392
+ _onWordsFileSaveConfirm()
1393
+ {
1394
+ // Disable words files manager buttons
1395
+ this._wordsFilesManager.enabled = false;
1396
+ this._wordsFilesManager.saveSpinnerVisible = true;
1397
+
1398
+ // Send request to server
1399
+ let params = new SFS2X.SFSObject();
1400
+ params.putUtfString('filename', this._tempWordsFileData.filename);
1401
+ params.putText('content', this._tempWordsFileData.content);
1402
+ this.sendExtensionRequest(this.REQ_SAVE_WORDS_FILE, params);
1403
+ }
1404
+
1405
+ _onWordsFileRemoveClick(evt)
1406
+ {
1407
+ this.shellCtrl.showConfirmWarning('Are you sure you want to delete the selected words file?', $.proxy(this._onWordsFileRemoveConfirm, this));
1408
+ }
1409
+
1410
+ _onWordsFileRemoveConfirm()
1411
+ {
1412
+ let wordsFile = this._wordsFilesManager.getSelectedWordsFileName();
1413
+
1414
+ if (wordsFile != null)
1415
+ {
1416
+ // Disable words files manager buttons
1417
+ this._wordsFilesManager.enabled = false;
1418
+ this._wordsFilesManager.actionSpinnerVisible = true;
1419
+
1420
+ // Send request to server
1421
+ let params = new SFS2X.SFSObject();
1422
+ params.putUtfString('filename', wordsFile);
1423
+ this.sendExtensionRequest(this.REQ_DELETE_WORDS_FILE, params);
1424
+ }
1425
+ }
1426
+
1427
+ _onWordsFileAssignClick(evt)
1428
+ {
1429
+ let path = evt.detail;
1430
+
1431
+ // Write path of the selected words file in "wordsFilter.wordsFile" dynamically created field
1432
+ const wordsFileFormItem = this._interfaceBuilder.getConfigFormItem('wordsFilter.wordsFile');
1433
+ wordsFileFormItem.data.value = path;
1434
+ wordsFileFormItem._setWidgetValue();
1435
+ }
1436
+
835
1437
  //---------------------------------
836
1438
  // PRIVATE METHODS
837
1439
  //---------------------------------
@@ -1181,4 +1783,4 @@ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
1181
1783
  /***/ })
1182
1784
 
1183
1785
  }]);
1184
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-12.bundle.js","sources":["webpack://application/./src/components/sidebar-layout.js","webpack://application/./src/modules/zone-configurator.js"],"sourcesContent":["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';\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\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\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// Lestend 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\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// 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// 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\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.getUtfString('classesString').split(',');\n\n\t\t\t\t\t\tif (filterFormItem.data.value == true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet filteredClasses = classes.filter(filterClassName);\n\t\t\t\t\t\t\tclasses = filteredClasses;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tclassesList = classesList.concat(classes);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet currentClass = classFormItem.data.value;\n\n\t\t\t\t// If the classes list doesn't contain the current value, create an empty entry and reset the value\n\t\t\t\tif (classesList.indexOf(currentClass) < 0)\n\t\t\t\t{\n\t\t\t\t\tif (classesList.length == 0)\n\t\t\t\t\t{\n\t\t\t\t\t\tclassesList.push('');\n\t\t\t\t\t\tcurrentClass = '';\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tcurrentClass = classesList[0];\n\t\t\t\t}\n\n\t\t\t\tlet mainClassDropDown = classFormItem._innerWidget;\n\t\t\t\tmainClassDropDown.setDataSource(classesList);\n\n\t\t\t\tclassFormItem.data.value = currentClass;\n\t\t\t\tclassFormItem._setWidgetValue();\n\t\t\t}\n\t\t}\n\t}\n\n\t//---------------------------------\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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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":""}
1786
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-12.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\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.getUtfString('classesString').split(',');\n\n\t\t\t\t\t\tif (filterFormItem.data.value == true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet filteredClasses = classes.filter(filterClassName);\n\t\t\t\t\t\t\tclasses = filteredClasses;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tclassesList = classesList.concat(classes);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet currentClass = classFormItem.data.value;\n\n\t\t\t\t// If the classes list doesn't contain the current value, create an empty entry and reset the value\n\t\t\t\tif (classesList.indexOf(currentClass) < 0)\n\t\t\t\t{\n\t\t\t\t\tif (classesList.length == 0)\n\t\t\t\t\t{\n\t\t\t\t\t\tclassesList.push('');\n\t\t\t\t\t\tcurrentClass = '';\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tcurrentClass = classesList[0];\n\t\t\t\t}\n\n\t\t\t\tlet mainClassDropDown = classFormItem._innerWidget;\n\t\t\t\tmainClassDropDown.setDataSource(classesList);\n\n\t\t\t\tclassFormItem.data.value = currentClass;\n\t\t\t\tclassFormItem._setWidgetValue();\n\t\t\t}\n\t\t}\n\t}\n\n\t_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;;;;;;A","sourceRoot":""}