rubyfox-server 2.16.0.1 → 2.16.3.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (65) hide show
  1. checksums.yaml +4 -4
  2. data/.travis.yml +2 -1
  3. data/README.md +19 -0
  4. data/lib/rubyfox/server/cli.rb +39 -5
  5. data/lib/rubyfox/server/data/lib/apache-tomcat/README.md +1 -1
  6. data/lib/rubyfox/server/data/lib/apache-tomcat/RELEASE-NOTES +2 -2
  7. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/bootstrap.jar +0 -0
  8. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina.bat +1 -0
  9. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina.sh +6 -9
  10. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon-native.tar.gz +0 -0
  11. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon.jar +0 -0
  12. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/daemon.sh +3 -3
  13. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-juli.jar +0 -0
  14. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-native.tar.gz +0 -0
  15. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/web.xml +5 -1
  16. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/annotations-api.jar +0 -0
  17. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ant.jar +0 -0
  18. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ha.jar +0 -0
  19. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ssi.jar +0 -0
  20. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-storeconfig.jar +0 -0
  21. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-tribes.jar +0 -0
  22. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina.jar +0 -0
  23. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/ecj-4.17.jar +0 -0
  24. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/el-api.jar +0 -0
  25. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper-el.jar +0 -0
  26. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper.jar +0 -0
  27. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jaspic-api.jar +0 -0
  28. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jsp-api.jar +0 -0
  29. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/servlet-api.jar +0 -0
  30. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-api.jar +0 -0
  31. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-coyote.jar +0 -0
  32. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-dbcp.jar +0 -0
  33. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-cs.jar +0 -0
  34. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-de.jar +0 -0
  35. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-es.jar +0 -0
  36. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-fr.jar +0 -0
  37. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ja.jar +0 -0
  38. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ko.jar +0 -0
  39. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-pt-BR.jar +0 -0
  40. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ru.jar +0 -0
  41. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-zh-CN.jar +0 -0
  42. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jdbc.jar +0 -0
  43. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jni.jar +0 -0
  44. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util-scan.jar +0 -0
  45. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util.jar +0 -0
  46. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-websocket.jar +0 -0
  47. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/websocket-api.jar +0 -0
  48. data/lib/rubyfox/server/data/lib/sfs2x-admin.jar +0 -0
  49. data/lib/rubyfox/server/data/lib/sfs2x-core.jar +0 -0
  50. data/lib/rubyfox/server/data/lib/sfs2x.jar +0 -0
  51. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/style.css +3 -1
  52. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/application.bundle.js +2 -2
  53. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-10.bundle.js +9 -11
  54. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12.bundle.js +604 -2
  55. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-13~module-9.bundle.js +4 -1
  56. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13.bundle.js +19 -2
  57. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-6.bundle.js +10 -12
  58. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-9.bundle.js +4 -6
  59. data/lib/rubyfox/server/data/www/ROOT/admin/modules/template.html +5 -0
  60. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-configurator.html +11 -1
  61. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-monitor.html +8 -0
  62. data/lib/rubyfox/server/data/zones/BasicExamples.zone.xml +147 -0
  63. data/lib/rubyfox/server/version.rb +1 -1
  64. metadata +9 -7
  65. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/ecj-4.15.jar +0 -0
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 71b5d2a3a58cb2cc1f18932f1e42567866e586151c6e8f9f5461d16c0c1e15b2
4
- data.tar.gz: 3c994763d0573bd68de34494b34741adbda1df6a70b05dcffdd96e5c653fcafa
3
+ metadata.gz: 9d4e28247a6ae810b198a8875a45951c88205600f99e05bd05baa882c2dd9ed9
4
+ data.tar.gz: c495ab7e93e81d6c8e24256bad9200a5b997d129d176af90ef7f6e907e3c816c
5
5
  SHA512:
6
- metadata.gz: be9ad06b978a7820d23acaa170483518e03f0e2b51c994fc74f693f7ef02e8dd020c674c525181992cecdf76e8649d4aa92344d3e0cb26031043d04278110b64
7
- data.tar.gz: ebe6fad363f2cd14a01c0b872c83a192b59c353b8388e134173c843737021688f03eabb1748e6a21e9d8d470a14ddf60291bd7c4e73b1eae4050fc235f9917ec
6
+ metadata.gz: f3ab35a69af653c13f9d01bd2dc56dc2e591eeac968a8dc55bb316c9ed55bc97eeb60ffd19aa07a3a621d7ba9bfa3f29ec2fd97db89a7274c5d72646cb388af2
7
+ data.tar.gz: 5cf41bf9088f7a67515d34099afb42d2bb6900f681946dec09935546f113ff86ea116d94ced05b7252cd2257dc0bed8bc440ed3d9014f62e325d072f5d8a73ac
@@ -1,3 +1,4 @@
1
1
  rvm:
2
- - 2.3.1
2
+ - 2.7.1
3
+ - 2.6.6
3
4
  - ruby-head
data/README.md CHANGED
@@ -19,6 +19,7 @@ See http://www.smartfoxserver.com
19
19
  smartfox configure /path/to/smartfox /path/to/template/dir
20
20
 
21
21
  smartfox start /path/to/smartfox
22
+
22
23
 
23
24
  ## Help
24
25
 
@@ -35,6 +36,24 @@ The version of this gem is tied to the real version of SmartFox Server.
35
36
 
36
37
  Example: Gem version 2.3.0.x contains SmartFox Server version 2.3.0.
37
38
 
39
+ ## Version check
40
+
41
+ To enable gem version verification, add a version file to your template folder.
42
+
43
+ Example: `.../version`
44
+ ```
45
+ 2.16.0
46
+ ```
47
+
48
+ If the version is not satisfied, the following output is printed:
49
+ ```
50
+ Configuration failed!
51
+
52
+ Your rubyfox-server version: 2.15.0.0
53
+ Needed version: ~>2.16.0
54
+ ```
55
+
56
+
38
57
  ## Contributing
39
58
 
40
59
  1. Fork it
@@ -1,9 +1,9 @@
1
- require 'thor'
2
- require 'mime/types'
1
+ require "thor"
2
+ require "mime/types"
3
3
 
4
- require 'rubyfox/server'
5
- require 'rubyfox/server/version'
6
- require 'rubyfox/server/environment'
4
+ require "rubyfox/server"
5
+ require "rubyfox/server/version"
6
+ require "rubyfox/server/environment"
7
7
 
8
8
  module Rubyfox
9
9
  module Server
@@ -15,6 +15,7 @@ module Rubyfox
15
15
  end
16
16
 
17
17
  desc "install TARGET_DIR", "Install SmartFox Server into TARGET_DIR"
18
+
18
19
  def install(target_dir)
19
20
  if File.exist?(target_dir)
20
21
  abort "Directory #{target_dir} already exists!"
@@ -24,10 +25,13 @@ module Rubyfox
24
25
  end
25
26
 
26
27
  desc "configure TARGET_DIR TEMPLATE_DIR", "Configure SmartFox Server in TARGET_DIR via TEMPLATE_DIR"
28
+
27
29
  def configure(target_dir, template_dir)
28
30
  template_dir = File.expand_path(template_dir, Dir.pwd)
29
31
  target_dir = File.expand_path(target_dir, Dir.pwd)
30
32
 
33
+ verify_version(template_dir)
34
+
31
35
  Dir["#{template_dir}/**/*"].each do |file|
32
36
  if File.file?(file)
33
37
  part = file.partition(template_dir).last
@@ -39,10 +43,14 @@ module Rubyfox
39
43
  end
40
44
  end
41
45
  end
46
+ rescue WrongVersionError => e
47
+ STDERR.puts e.message
42
48
  end
49
+
43
50
  map "config" => :configure
44
51
 
45
52
  desc "start TARGET_DIR", "Start SmartFox Server in TARGET_DIR"
53
+
46
54
  def start(target_dir)
47
55
  inside(target_dir) do
48
56
  system "sh ./sfs2x.sh"
@@ -50,6 +58,7 @@ module Rubyfox
50
58
  end
51
59
 
52
60
  desc "version", "Display version of this command"
61
+
53
62
  def version
54
63
  puts Rubyfox::Server::VERSION
55
64
  end
@@ -66,6 +75,31 @@ module Rubyfox
66
75
  types = MIME::Types.type_for(file)
67
76
  types.empty? || types.any? { |type| type.media_type == "text" }
68
77
  end
78
+
79
+ def verify_version(template_dir)
80
+ filename = "#{template_dir}/version"
81
+ if File.exist?(filename)
82
+ expected_version = File.read(filename)
83
+ version = Rubyfox::Server::VERSION
84
+
85
+ unless satisfied_version?(expected_version, version)
86
+ msg = query = <<~MSG.chomp
87
+ Configuration failed!
88
+
89
+ Your rubyfox-server version: #{version}
90
+ Needed version: ~>#{expected_version}
91
+ MSG
92
+ raise WrongVersionError.new(msg)
93
+ end
94
+ end
95
+ end
96
+
97
+ def satisfied_version?(expected_version, version)
98
+ requirement = Gem::Requirement.new("~> #{expected_version}")
99
+ requirement.satisfied_by?(Gem::Version.create(version))
100
+ end
101
+
102
+ class WrongVersionError < RuntimeError; end
69
103
  end
70
104
  end
71
105
  end
@@ -40,7 +40,7 @@ To facilitate choosing the right major Tomcat version one, we have provided a
40
40
 
41
41
  The documentation available as of the date of this release is
42
42
  included in the docs webapp which ships with tomcat. You can access that webapp
43
- by starting tomcat and visiting http://localhost:8080/docs/ in your browser.
43
+ by starting tomcat and visiting <http://localhost:8080/docs/> in your browser.
44
44
  The most up-to-date documentation for each version can be found at:
45
45
  - [Tomcat 9](https://tomcat.apache.org/tomcat-9.0-doc/)
46
46
  - [Tomcat 8](https://tomcat.apache.org/tomcat-8.5-doc/)
@@ -16,7 +16,7 @@
16
16
  ================================================================================
17
17
 
18
18
 
19
- Apache Tomcat Version 9.0.36
19
+ Apache Tomcat Version 9.0.40
20
20
  Release Notes
21
21
 
22
22
 
@@ -73,7 +73,7 @@ for use by web applications (by placing them in "lib"):
73
73
  * catalina-ssi.jar (Server-side Includes module)
74
74
  * catalina-storeconfig.jar (Generation of XML configuration from current state)
75
75
  * catalina-tribes.jar (Group communication)
76
- * ecj-4.15.jar (Eclipse JDT Java compiler)
76
+ * ecj-4.17.jar (Eclipse JDT Java compiler)
77
77
  * el-api.jar (EL 3.0 API)
78
78
  * jasper.jar (Jasper 2 Compiler and Runtime)
79
79
  * jasper-el.jar (Jasper 2 EL implementation)
@@ -266,6 +266,7 @@ goto java_dir_displayed
266
266
  echo Using JAVA_HOME: "%JAVA_HOME%"
267
267
  :java_dir_displayed
268
268
  echo Using CLASSPATH: "%CLASSPATH%"
269
+ echo Using CATALINA_OPTS: "%CATALINA_OPTS%"
269
270
 
270
271
  set _EXECJAVA=%_RUNJAVA%
271
272
  set MAINCLASS=org.apache.catalina.startup.Bootstrap
@@ -271,9 +271,9 @@ JAVA_OPTS="$JAVA_OPTS -Djava.protocol.handler.pkgs=org.apache.catalina.webresour
271
271
  # Check for the deprecated LOGGING_CONFIG
272
272
  # Only use it if CATALINA_LOGGING_CONFIG is not set and LOGGING_CONFIG starts with "-D..."
273
273
  if [ -z "$CATALINA_LOGGING_CONFIG" ]; then
274
- if [ "${LOGGING_CONFIG#*-D}" != "$LOGGING_CONFIG" ]; then
275
- CATALINA_LOGGING_CONFIG="$LOGGING_CONFIG"
276
- fi
274
+ case $LOGGING_CONFIG in
275
+ -D*) CATALINA_LOGGING_CONFIG="$LOGGING_CONFIG"
276
+ esac
277
277
  fi
278
278
 
279
279
  # Set juli LogManager config file if it is present and an override has not been issued
@@ -342,6 +342,7 @@ if [ $have_tty -eq 1 ]; then
342
342
  echo "Using JRE_HOME: $JRE_HOME"
343
343
  fi
344
344
  echo "Using CLASSPATH: $CLASSPATH"
345
+ echo "Using CATALINA_OPTS: $CATALINA_OPTS"
345
346
  if [ ! -z "$CATALINA_PID" ]; then
346
347
  echo "Using CATALINA_PID: $CATALINA_PID"
347
348
  fi
@@ -364,10 +365,6 @@ if [ "$1" = "jpda" ] ; then
364
365
  shift
365
366
  fi
366
367
 
367
- # TODO: Bugzilla 63815
368
- # This doesn't currently work (and can't be made to work) if values used in
369
- # CATALINA_OPTS and/or JAVA_OPTS require quoting. See:
370
- # https://bugs.openjdk.java.net/browse/JDK-8234808
371
368
  if [ "$1" = "debug" ] ; then
372
369
  if $os400; then
373
370
  echo "Debug command not available on OS400"
@@ -379,7 +376,7 @@ if [ "$1" = "debug" ] ; then
379
376
  echo "Using Security Manager"
380
377
  fi
381
378
  shift
382
- exec "$_RUNJDB" "$CATALINA_LOGGING_CONFIG" $LOGGING_MANAGER $JAVA_OPTS $CATALINA_OPTS \
379
+ eval exec "\"$_RUNJDB\"" "\"$CATALINA_LOGGING_CONFIG\"" $LOGGING_MANAGER "$JAVA_OPTS" "$CATALINA_OPTS" \
383
380
  -D$ENDORSED_PROP="$JAVA_ENDORSED_DIRS" \
384
381
  -classpath "$CLASSPATH" \
385
382
  -sourcepath "$CATALINA_HOME"/../../java \
@@ -390,7 +387,7 @@ if [ "$1" = "debug" ] ; then
390
387
  -Djava.io.tmpdir="$CATALINA_TMPDIR" \
391
388
  org.apache.catalina.startup.Bootstrap "$@" start
392
389
  else
393
- exec "$_RUNJDB" "$CATALINA_LOGGING_CONFIG" $LOGGING_MANAGER $JAVA_OPTS $CATALINA_OPTS \
390
+ eval exec "\"$_RUNJDB\"" "\"$CATALINA_LOGGING_CONFIG\"" $LOGGING_MANAGER "$JAVA_OPTS" "$CATALINA_OPTS" \
394
391
  -D$ENDORSED_PROP="$JAVA_ENDORSED_DIRS" \
395
392
  -classpath "$CLASSPATH" \
396
393
  -sourcepath "$CATALINA_HOME"/../../java \
@@ -152,9 +152,9 @@ fi
152
152
  # Check for the deprecated LOGGING_CONFIG
153
153
  # Only use it if CATALINA_LOGGING_CONFIG is not set and LOGGING_CONFIG starts with "-D..."
154
154
  if [ -z "$CATALINA_LOGGING_CONFIG" ]; then
155
- if [ "${LOGGING_CONFIG#*-D}" != "$LOGGING_CONFIG" ]; then
156
- CATALINA_LOGGING_CONFIG="$LOGGING_CONFIG"
157
- fi
155
+ case $LOGGING_CONFIG in
156
+ -D*) CATALINA_LOGGING_CONFIG="$LOGGING_CONFIG"
157
+ esac
158
158
  fi
159
159
 
160
160
  # Set juli LogManager config file if it is present and an override has not been issued
@@ -543,7 +543,7 @@
543
543
  <!-- this to work you will need to uncomment the .shtml mime type -->
544
544
  <!-- definition towards the bottom of this file. -->
545
545
  <!-- The contentType init param allows you to apply SSI processing to JSP -->
546
- <!-- pages, javascript, or any other content you wish. This filter -->
546
+ <!-- pages, JavaScript, or any other content you wish. This filter -->
547
547
  <!-- supports the following initialization parameters (default values are -->
548
548
  <!-- in square brackets): -->
549
549
  <!-- -->
@@ -4252,6 +4252,10 @@
4252
4252
  <extension>wad</extension>
4253
4253
  <mime-type>application/x-doom</mime-type>
4254
4254
  </mime-mapping>
4255
+ <mime-mapping>
4256
+ <extension>wasm</extension>
4257
+ <mime-type>application/wasm</mime-type>
4258
+ </mime-mapping>
4255
4259
  <mime-mapping>
4256
4260
  <extension>wav</extension>
4257
4261
  <mime-type>audio/x-wav</mime-type>
@@ -155,12 +155,13 @@ view-stack.adjust-height-100 {
155
155
  border: none !important;
156
156
  }
157
157
 
158
- .k-input, .k-textbox {
158
+ .k-input, .k-textbox, .k-textarea {
159
159
  color: #495057 !important;
160
160
  text-indent: 0 !important;
161
161
  }
162
162
 
163
163
  .k-textbox:focus,
164
+ .k-textarea:focus,
164
165
  .k-numerictextbox .k-state-focused,
165
166
  .k-numerictextbox .k-state-hover,
166
167
  .k-combobox .k-state-focused {
@@ -169,6 +170,7 @@ view-stack.adjust-height-100 {
169
170
  }
170
171
 
171
172
  .k-textbox:focus,
173
+ .k-textarea:focus,
172
174
  .k-dropdown .k-state-focused,
173
175
  .k-numerictextbox .k-state-focused,
174
176
  .k-numerictextbox .k-state-hover,
@@ -1057,7 +1057,7 @@ class ShellController
1057
1057
 
1058
1058
  this.VERSION_MAJOR = 3;
1059
1059
  this.VERSION_MINOR = 2;
1060
- this.VERSION_SUB = 1;
1060
+ this.VERSION_SUB = 4;
1061
1061
 
1062
1062
  this.DEFAULT_WS_PORT = 8080;
1063
1063
  this.DEFAULT_WSS_PORT = 8443;
@@ -2205,4 +2205,4 @@ module.exports = jQuery;
2205
2205
  /***/ })
2206
2206
 
2207
2207
  /******/ });
2208
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/application.bundle.js","sources":["webpack://application/webpack/bootstrap","webpack://application/./src/application.js","webpack://application/./src/components/view-stack.js","webpack://application/./src/managers/chat-manager.js","webpack://application/./src/managers/connection-manager.js","webpack://application/./src/managers/module-manager.js","webpack://application/./src/modules/base-module.js","webpack://application/./src/shell-controller.js","webpack://application/./src/utils/dot-properties-parse.js","webpack://application/./src/utils/event-dispatcher.js","webpack://application/./src/utils/events.js","webpack://application/./src/utils/utilities.js"],"sourcesContent":[" \t// install a JSONP callback for chunk loading\n \tfunction webpackJsonpCallback(data) {\n \t\tvar chunkIds = data[0];\n \t\tvar moreModules = data[1];\n\n\n \t\t// add \"moreModules\" to the modules object,\n \t\t// then flag all \"chunkIds\" as loaded and fire callback\n \t\tvar moduleId, chunkId, i = 0, resolves = [];\n \t\tfor(;i < chunkIds.length; i++) {\n \t\t\tchunkId = chunkIds[i];\n \t\t\tif(Object.prototype.hasOwnProperty.call(installedChunks, chunkId) && installedChunks[chunkId]) {\n \t\t\t\tresolves.push(installedChunks[chunkId][0]);\n \t\t\t}\n \t\t\tinstalledChunks[chunkId] = 0;\n \t\t}\n \t\tfor(moduleId in moreModules) {\n \t\t\tif(Object.prototype.hasOwnProperty.call(moreModules, moduleId)) {\n \t\t\t\tmodules[moduleId] = moreModules[moduleId];\n \t\t\t}\n \t\t}\n \t\tif(parentJsonpFunction) parentJsonpFunction(data);\n\n \t\twhile(resolves.length) {\n \t\t\tresolves.shift()();\n \t\t}\n\n \t};\n\n\n \t// The module cache\n \tvar installedModules = {};\n\n \t// object to store loaded and loading chunks\n \t// undefined = chunk not loaded, null = chunk preloaded/prefetched\n \t// Promise = chunk loading, 0 = chunk loaded\n \tvar installedChunks = {\n \t\t\"application\": 0\n \t};\n\n\n\n \t// script path function\n \tfunction jsonpScriptSrc(chunkId) {\n \t\treturn __webpack_require__.p + \"assets/js/core/modules/\" + ({\"module-10\":\"module-10\",\"module-11\":\"module-11\",\"module-12~module-13~module-9\":\"module-12~module-13~module-9\",\"module-12\":\"module-12\",\"module-3\":\"module-3\",\"module-6\":\"module-6\",\"vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8\":\"vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8\",\"module-1\":\"module-1\",\"module-13\":\"module-13\",\"module-4\":\"module-4\",\"module-5\":\"module-5\",\"module-7\":\"module-7\",\"module-8\":\"module-8\",\"vendors~module-0\":\"vendors~module-0\",\"module-0\":\"module-0\",\"vendors~module-9\":\"vendors~module-9\",\"module-9\":\"module-9\"}[chunkId]||chunkId) + \".bundle.js\"\n \t}\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n \t// This file contains only the entry chunk.\n \t// The chunk loading function for additional chunks\n \t__webpack_require__.e = function requireEnsure(chunkId) {\n \t\tvar promises = [];\n\n\n \t\t// JSONP chunk loading for javascript\n\n \t\tvar installedChunkData = installedChunks[chunkId];\n \t\tif(installedChunkData !== 0) { // 0 means \"already installed\".\n\n \t\t\t// a Promise means \"currently loading\".\n \t\t\tif(installedChunkData) {\n \t\t\t\tpromises.push(installedChunkData[2]);\n \t\t\t} else {\n \t\t\t\t// setup Promise in chunk cache\n \t\t\t\tvar promise = new Promise(function(resolve, reject) {\n \t\t\t\t\tinstalledChunkData = installedChunks[chunkId] = [resolve, reject];\n \t\t\t\t});\n \t\t\t\tpromises.push(installedChunkData[2] = promise);\n\n \t\t\t\t// start chunk loading\n \t\t\t\tvar script = document.createElement('script');\n \t\t\t\tvar onScriptComplete;\n\n \t\t\t\tscript.charset = 'utf-8';\n \t\t\t\tscript.timeout = 120;\n \t\t\t\tif (__webpack_require__.nc) {\n \t\t\t\t\tscript.setAttribute(\"nonce\", __webpack_require__.nc);\n \t\t\t\t}\n \t\t\t\tscript.src = jsonpScriptSrc(chunkId);\n\n \t\t\t\t// create error before stack unwound to get useful stacktrace later\n \t\t\t\tvar error = new Error();\n \t\t\t\tonScriptComplete = function (event) {\n \t\t\t\t\t// avoid mem leaks in IE.\n \t\t\t\t\tscript.onerror = script.onload = null;\n \t\t\t\t\tclearTimeout(timeout);\n \t\t\t\t\tvar chunk = installedChunks[chunkId];\n \t\t\t\t\tif(chunk !== 0) {\n \t\t\t\t\t\tif(chunk) {\n \t\t\t\t\t\t\tvar errorType = event && (event.type === 'load' ? 'missing' : event.type);\n \t\t\t\t\t\t\tvar realSrc = event && event.target && event.target.src;\n \t\t\t\t\t\t\terror.message = 'Loading chunk ' + chunkId + ' failed.\\n(' + errorType + ': ' + realSrc + ')';\n \t\t\t\t\t\t\terror.name = 'ChunkLoadError';\n \t\t\t\t\t\t\terror.type = errorType;\n \t\t\t\t\t\t\terror.request = realSrc;\n \t\t\t\t\t\t\tchunk[1](error);\n \t\t\t\t\t\t}\n \t\t\t\t\t\tinstalledChunks[chunkId] = undefined;\n \t\t\t\t\t}\n \t\t\t\t};\n \t\t\t\tvar timeout = setTimeout(function(){\n \t\t\t\t\tonScriptComplete({ type: 'timeout', target: script });\n \t\t\t\t}, 120000);\n \t\t\t\tscript.onerror = script.onload = onScriptComplete;\n \t\t\t\tdocument.head.appendChild(script);\n \t\t\t}\n \t\t}\n \t\treturn Promise.all(promises);\n \t};\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// on error function for async loading\n \t__webpack_require__.oe = function(err) { console.error(err); throw err; };\n\n \tvar jsonpArray = window[\"webpackJsonpapplication\"] = window[\"webpackJsonpapplication\"] || [];\n \tvar oldJsonpFunction = jsonpArray.push.bind(jsonpArray);\n \tjsonpArray.push = webpackJsonpCallback;\n \tjsonpArray = jsonpArray.slice();\n \tfor(var i = 0; i < jsonpArray.length; i++) webpackJsonpCallback(jsonpArray[i]);\n \tvar parentJsonpFunction = oldJsonpFunction;\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = \"./src/application.js\");\n","import {ShellController} from './shell-controller';\n\n$( document ).ready(function()\n{\n    console.info(\"SmartFoxServer 2X AdminTool ready!\");\n\n\t// Create shell controller instance\n\tthis.controller = new ShellController();\n});\n","export class ViewStack 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\n\t\t\t\t::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t      display: none !important;\n\t\t\t    }\n\t\t\t</style>\n\t\t\t<slot></slot>\n\t\t`;\n\n\t\t// Select first item\n\t\tthis.selectedIndex = 0;\n\t}\n\n\tget selectedElement()\n\t{\n\t\treturn this._selectedElement;\n\t}\n\n\tset selectedElement(element)\n\t{\n\t\tif (element != null && element.parentNode == this)\n\t\t{\n\t\t\tthis._selectedElement = element;\n\n\t\t\tfor (let element of this.children)\n\t\t\t{\n\t\t\t\tif (element == this._selectedElement)\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 ViewStack');\n\t\t}\n\t}\n\n\tget selectedIndex()\n\t{\n\t\treturn Array.from(this.children).indexOf(this._selectedElement);\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 ViewStack index\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet element = this.children[index];\n\t\t\tthis.selectedElement = element;\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('view-stack'))\n\twindow.customElements.define('view-stack', ViewStack);\n","export class ChatManager\n{\n\tconstructor(shellCtrl)\n\t{\n\t\tthis.USERVAR_MODULE = 'mod';\n\n\t\tthis.shellCtrl = shellCtrl;\n\n\t\t// TODO Implement chat manager after creating the chat UI in main shell\n\t}\n\n\t/**\n\t * Called by the shell when the user loads a new module, so that this info\n\t * can be saved in the user variables and displayed in the chat's userlist.\n\t */\n\tsetCurrentModule(moduleId)\n\t{\n\t\t// Save module id in user variables\n\t\tlet userVar = new SFS2X.SFSUserVariable(this.USERVAR_MODULE, moduleId);\n\t\tthis.shellCtrl.smartFox.send(new SFS2X.SetUserVariablesRequest([userVar]));\n\t}\n}\n","import {EventDispatcher} from '../utils/event-dispatcher';\nimport {ConnectionManagerEvent} from '../utils/events';\n\nexport class ConnectionManager extends EventDispatcher\n{\n\tconstructor(shellCtrl)\n\t{\n\t\tsuper();\n\n\t\tthis.ADMIN_ZONE_NAME = \"--=={{{ AdminZone }}}==--\";\n\t\tthis.EXTENSION_ERROR = \"error\";\n\n\t\tthis.COMMANDS_PREFIX = \"admin\";\n\t\tthis.COMMAND_RESTART = \"restart\";\n\t\tthis.COMMAND_HALT = \"halt\";\n\n\t\tthis.shellCtrl = shellCtrl;\n\t}\n\n\tget smartFox()\n\t{\n\t\treturn this._sf;\n\t}\n\n\tconnect(config, username, password)\n\t{\n\t\t// Set additional configuration options\n\t\tconfig.zone = this.ADMIN_ZONE_NAME;\n\t\tconfig.debug = false;\n\n\t\t// Create SmartFox instance\n\t\tthis._sf = new SFS2X.SmartFox(config);\n\n\t\t// Add listeners to SmartFox events useful to the shell\n\t\tthis._addSFSEventListeners();\n\n\t\t// Save reference to connection details\n\t\tthis._config = config;\n\t\tthis._username = username;\n\t\tthis._password = password;\n\n\t\t// Connect to SmartFoxServer instance\n\t\tthis._sf.connect();\n\t}\n\n\tdisconnect()\n\t{\n\t\tthis._sf.disconnect();\n\t}\n\n\trestartServer()\n\t{\n\t\tif (this._sf.isConnected)\n\t\t\tthis._sf.send(new SFS2X.ExtensionRequest(this.COMMANDS_PREFIX + \".\" + this.COMMAND_RESTART));\n\t}\n\n\thaltServer()\n\t{\n\t\tif (this._sf.isConnected)\n\t\t\tthis._sf.send(new SFS2X.ExtensionRequest(this.COMMANDS_PREFIX + \".\" + this.COMMAND_HALT));\n\t}\n\n\t/* --------- PRIVATE UTILITY METHODS --------- */\n\n\t_addSFSEventListeners()\n\t{\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.CONNECTION, this._onConnection, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.CONNECTION_LOST, this._onConnectionLost, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.LOGIN, this._onLogin, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.LOGIN_ERROR, this._onLoginError, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse, this);\n\t}\n\n\t_reset()\n\t{\n\t\t// Remove SFS event listeners\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.CONNECTION, this._onConnection);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.CONNECTION_LOST, this._onConnectionLost);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.LOGIN, this._onLogin);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.LOGIN_ERROR, this._onLoginError);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse);\n\n\t\t// Delete SmartFox class instance\n\t\tthis._sf = null;\n\n\t\t// Delete connection details\n\t\tthis._config = null;\n\t\tthis._username = null;\n\t\tthis._password = null;\n\t}\n\n\t_login()\n\t{\n\t\t// The current AdminTool version must be sent to the server during login, to check if it is up-to-date\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putInt('clientVer', this.shellCtrl.intVersion);\n\n\t\t// Login\n\t\tthis._sf.send( new SFS2X.LoginRequest(this._username, this._password, params, this._config.zone) );\n\t}\n\n\t/* --------- SMARTFOX EVENT LISTENERS --------- */\n\n\t_onConnection(evtParams)\n\t{\n\t\tif (evtParams.success)\n\t\t{\n\t\t\t// Dispatch connection event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.CONNECTION);\n\n\t\t\t// Send login request\n\t\t\tthis._login();\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Dispatch error event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.ERROR, {message: `Unable to connect to ${this._config.host}:${this._config.port}`});\n\n\t\t\t// Reset status\n\t\t\tthis._reset();\n\t\t}\n\t}\n\n\t_onConnectionLost(evtParams)\n\t{\n\t\tlet reason = evtParams.reason;\n\n\t\tif (reason != SFS2X.ClientDisconnectionReason.MANUAL)\n\t\t{\n\t\t\tvar msg;\n\n\t\t\t// Log disconnection message stating the reason\n\t\t\tif (reason == SFS2X.ClientDisconnectionReason.IDLE)\n\t\t\t\tmsg = 'inactivity';\n\t\t\telse\n\t\t\t{\n\t\t\t\tmsg = 'unknown reason';\n\n\t\t\t\tif (reason != SFS2X.ClientDisconnectionReason.UNKNOWN)\n\t\t\t\t\tmsg += ` (server reported: ${reason})`;\n\t\t\t}\n\n\t\t\t// Dispatch error event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.ERROR, {message: `A disconnection occurred due to ${msg}; please reconnect`});\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Dispatch disconnection event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.DISCONNECTION);\n\t\t}\n\n\t\t// Reset\n\t\tthis._reset();\n\t}\n\n\t_onLogin(evtParams)\n\t{\n\t\tlet data = evtParams.data;\n\n\t\tlet params = {\n\t\t\tserverVersion: data.getUtfString('serverVer'),\n\t\t\tserverName: data.getUtfString('serverName'),\n\t\t\tserverUptime: data.getIntArray('uptime'),\n\t\t\tprocControlEnabled: data.getBool('procCtrl'),\n\t\t\tallowHalt: data.getBool('allowHalt'),\n\t\t\tmodules: data.getSFSArray('modules')\n\t\t};\n\n\t\t// Dispatch login event\n\t\tthis.dispatchEvent(ConnectionManagerEvent.LOGIN, params);\n\t}\n\n\t_onLoginError(evtParams)\n\t{\n\t\t// Disconnect from server\n\t\tthis._sf.disconnect();\n\n\t\t// Dispatch error event\n\t\tthis.dispatchEvent(ConnectionManagerEvent.ERROR, {message: evtParams.errorMessage});\n\t}\n\n\t_onExtensionResponse(evtParams)\n\t{\n\t\tif (evtParams.cmd == this.EXTENSION_ERROR)\n\t\t{\n\t\t\tlet data = evtParams.params;\n\n\t\t\t// An unexpected error occurred in the Admin Tool server-side extension\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.SERVER_ERROR, {message: 'An unexpected error occurred in the Admin Tool server-side extension, please check the server-side logs; the extension reported: ' + data.getUtfString('error')});\n\t\t}\n\t}\n}\n","import {EventDispatcher} from '../utils/event-dispatcher';\nimport {ModuleManagerEvent} from '../utils/events';\nimport {toKebabCase} from '../utils/utilities';\nimport {BaseModule} from '../modules/base-module';\n\nexport class ModuleManager extends EventDispatcher\n{\n\tconstructor(shellCtrl, container)\n\t{\n\t\tsuper();\n\n\t\t// Make BaseModule class globally available, so it can be extended by custom modules\n\t\twindow.BaseModule = BaseModule\n\n\t\tthis._shellCtrl = shellCtrl;\n\t\tthis._container = container;\n\t\tthis._currentModuleId = null;\n\n\t\t// Add listener for navigation items click\n\t\tthis._container.on('click', '.nav-item', $.proxy(function(event) {\n\t\t\tevent.preventDefault();\n\n\t\t\tlet moduleId = event.currentTarget.dataset.id;\n\t\t\tlet moduleData = this._moduleConfigById[moduleId];\n\n\t\t\tif (moduleId != this._currentModuleId)\n\t\t\t{\n\t\t\t\t// Load selected module\n\t\t\t\tthis._loadModule(moduleData);\n\t\t\t}\n\n\t\t\t// Close navigation\n\t\t\tthis._shellCtrl._toggleNav(false);\n\t\t}, this));\n\n\t\t// Add listener to show tooltip\n\t\tthis._container.kendoTooltip({\n\t\t\tfilter: 'li[data-id]',\n\t\t\tcontent: $.proxy(function(event) {\n\t\t\t\tlet moduleId = event.target[0].dataset.id;\n\t\t\t\tlet moduleData = this._moduleConfigById[moduleId];\n\t\t        return `<strong>${moduleData.name}</strong><br><span>${moduleData.description}</span>`;\n\t\t    }, this)\n\t\t});\n\n\t\t/* TESTING MEMORY LEAKS IN MODULES LOADING\n\t\tADD THIS IN THE MAIN VIEW TO START THE TEST: <div><button id=\"temp\">test</button><span id=\"cnt\"></span></div>\n\t\t$('#temp').click(\n\t\t\t$.proxy(function(event)\n\t\t\t{\n\t\t\t\tthis.cnt = 0;\n\t\t\t\tif (this.timer == null)\n\t\t\t\t{\n\t\t\t\t\tthis.timer = window.setInterval($.proxy(function() {\n\n\t\t\t\t\t\t$('#cnt').text(this.cnt++)\n\t\t\t\t\t\tif (this._currentModuleId == 'Dashboard')\n\t\t\t\t\t\t\tthis._loadModule(this._moduleConfigById['ServerConfigurator']);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis._loadModule(this._moduleConfigById['Dashboard']);\n\t\t\t\t\t}, this), 500);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\twindow.clearInterval(this.timer);\n\t\t\t\t\tthis.timer = null;\n\t\t\t\t}\n\t\t\t}, this)\n\t\t);\n\t\t*/\n\t}\n\n\tinitModulesList(modulesData, loadModuleId = null)\n\t{\n\t\t// SETUP MODULES LIST\n\t\t\n\t\t// Empty container of module selection buttons\n\t\tthis._container.empty();\n\n\t\tthis._moduleConfigById = {};\n\t\tvar firstModule = null;\n\t\tvar loadModule = null;\n\n\t\tfor (let i = 0; i < modulesData.size(); i++)\n\t\t{\n\t\t\tlet moduleData = this._getModuleObject(modulesData.getSFSObject(i));\n\n\t\t\tthis._moduleConfigById[moduleData.id] = moduleData;\n\n\t\t\t// Get first module\n\t\t\tif (i == 0)\n\t\t\t\tfirstModule = moduleData;\n\n\t\t\t// Get passed module\n\t\t\tif (moduleData.id == loadModuleId)\n\t\t\t\tloadModule = moduleData;\n\n\t\t\t// Display module button\n\t\t\tlet moduleButton = this._createModuleButton(moduleData);\n\t\t\tlet element = this._container.append(moduleButton);\n\t\t}\n\n\t\t// LOAD INITIAL MODULE\n\n\t\t// If module is not passed, load the first one of the modules list\n\t\tif (loadModule == null)\n\t\t\tloadModule = firstModule;\n\n\t\tthis._loadModule(loadModule);\n\t}\n\n\tunloadModule()\n\t{\n\t\tthis._destroyCurrentModule();\n\t}\n\n\tget currentModuleId()\n\t{\n\t\treturn this._currentModuleId;\n\t}\n\n\tget currentModule()\n\t{\n\t\treturn document.querySelector('.module');\n\t}\n\n\t_getModuleObject(sfsObj)\n\t{\n\t\treturn {\n\t\t\tid: sfsObj.getUtfString('id'),\n\t\t\tname: sfsObj.getUtfString('name'),\n\t\t\tdescription: sfsObj.getUtfString('description'),\n\t\t\tisSub: sfsObj.getBool('isSub'),\n\t\t\ticon: sfsObj.containsKey('iconSvg') ? sfsObj.getUtfString('iconSvg') : '',\n\t\t\ttag: toKebabCase(sfsObj.getUtfString('id')),\n\t\t\tisCustom: sfsObj.getBool('isCustom')\n\t\t}\n\t}\n\n\t_createModuleButton(moduleData)\n\t{\n\t\treturn `\n\t\t\t<li class=\"nav-item\" data-id=\"${moduleData.id}\">\n\t\t\t\t<div class=\"module-icon\">${moduleData.icon}</div>\n\t\t\t\t<label>${moduleData.name}</label>\n\t\t\t</li>\n\t\t`;\n\t}\n\n\t_loadModule(moduleData)\n\t{\n\t\t// Load the HTML file of a module\n\t\t$('<module/>').load(`modules/${moduleData.tag}.html`, $.proxy(function(html, status) {\n\n\t\t\tif (status != 'error')\n\t\t\t{\n\t\t\t\tif (!moduleData.isCustom)\n\t\t\t\t{\n\t\t\t\t\t// Load the JS file of a standard module\n\t\t\t\t\timport(/* webpackChunkName: \"module-\" */ `../modules/${moduleData.tag}.js`).then(module => {\n\t\t\t\t\t\tthis._onModuleControllerLoadSuccess(moduleData, html, module);\n\t\t\t\t\t})\n\t\t\t\t\t.catch(error => {\n\t\t\t\t\t\tthis._onModuleControllerLoadError(moduleData, error);\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// Load the JS file of a custom module\n\t\t\t\t\timport(/* webpackIgnore: true */`../custom-modules/${moduleData.tag}.js`).then(module => {\n\t\t\t\t\t\tthis._onModuleControllerLoadSuccess(moduleData, html, module);\n\t\t\t\t\t})\n\t\t\t\t\t.catch(error => {\n\t\t\t\t\t\tthis._onModuleControllerLoadError(moduleData, error);\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Dispatch error event\n\t\t\t\tthis.dispatchEvent(ModuleManagerEvent.MODULE_LOAD_ERROR, {message: `${moduleData.name} module's view (html) not found.`});\n\t\t\t}\n\t\t}, this));\n\t}\n\n\t_onModuleControllerLoadSuccess(moduleData, html, module)\n\t{\n\t\t// Destroy current module\n\t\tthis._destroyCurrentModule();\n\n\t\t// Define loaded module (if necessary)\n\t\tif (!window.customElements.get(moduleData.tag + '-module'))\n\t\t\twindow.customElements.define(moduleData.tag + '-module', module.default);\n\n\t\t// Append new module\n\t\t$('div.module-loader').append(html);\n\n\t\t// Initialize module\n\t\tthis.currentModule.initialize(moduleData, this._shellCtrl);\n\n\t\t// Save current module id\n\t\tthis._currentModuleId = moduleData.id;\n\n\t\t// Dispatch event\n\t\tthis.dispatchEvent(ModuleManagerEvent.MODULE_LOADED, {moduleData: moduleData});\n\t}\n\n\t_onModuleControllerLoadError(moduleData, error)\n\t{\n\t\t// Log error details\n\t\tthis._shellCtrl.logMessage(error, 'warn');\n\n\t\t// Dispatch error event\n\t\tthis.dispatchEvent(ModuleManagerEvent.MODULE_LOAD_ERROR, {message: `${moduleData.name} custom module's controller (js) couldn't be loaded.`});\n\t}\n\n\t_destroyCurrentModule()\n\t{\n\t\t// Get a reference to current module\n\t\tconst mod = this.currentModule;\n\n\t\t// Call destroy method on module's class\n\t\tif (mod != null)\n\t\t\tmod.destroy();\n\n\t\t// Empty module container\n\t\t$('div.module-loader').empty();\n\n\t\tthis._currentModuleId = null;\n\t}\n}\n","export class BaseModule extends HTMLElement\n{\n\tconstructor(commandsPrefix)\n\t{\n\t    super();\n\n\t\tthis._commandsPrefix = commandsPrefix;\n\t}\n\n\tget shellCtrl()\n\t{\n\t\treturn this._shellCtrl;\n\t}\n\n\tget smartFox()\n\t{\n\t\treturn this._shellCtrl.smartFox;\n\t}\n\n\tget idData()\n\t{\n\t\treturn this._idData;\n\t}\n\n\t//---------------------------------\n\t// OVERRIDABLE METHODS\n\t//---------------------------------\n\n\t/**\n\t * Called by the modules manager after loading the module.\n\t * In case it is overridden, super must always be called!\n\t */\n\tinitialize(idData, shellController)\n\t{\n\t\tthis._idData = idData;\n\t\tthis._shellCtrl = shellController;\n\n\t\t// Add listener to Admin extension messages\n\t\tthis.smartFox.addEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse, this);\n\t}\n\n\t/**\n\t * Called by the modules manager before unloading the module.\n\t * In case it is overridden, super must always be called!\n\t */\n\tdestroy()\n\t{\n\t\t// Remove listener to Admin extension messages\n\t\tthis.smartFox.removeEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse);\n\n\t\t// Destroy all Kendo widgets\n\t\tkendo.destroy($('.module'));\n\t}\n\n\t/**\n\t * Called by the onExtensionResponse listener below.\n\t * Must be overridden.\n\t */\n\tonExtensionCommand(cmd, data)\n\t{\n\t\t// Nothing to do\n\t}\n\n\t/**\n\t * Called by the main shell whenever the server uptime changes.\n\t * Can be overridden to display the uptime inside a module or make calculations on the server uptime.\n\t */\n\tonUptimeUpdated(values)\n\t{\n\t\t// Nothing to do\n\t}\n\n\t//---------------------------------\n\t// PUBLIC METHODS\n\t//---------------------------------\n\n\t/**\n\t * Send a request to Admin extension.\n\t */\n\tsendExtensionRequest(command, data = null)\n\t{\n\t\tif (data == null)\n\t\t\tdata = new SFS2X.SFSObject();\n\n\t\tthis.smartFox.send(new SFS2X.ExtensionRequest(`${this._commandsPrefix}.${command}`, data));\n\t}\n\n\t//---------------------------------\n\t// PRIVATE METHODS\n\t//---------------------------------\n\n\t_onExtensionResponse(evtParams)\n\t{\n\t\t// Filter server responses\n\t\tlet commands = evtParams.cmd.split('.');\n\t\tlet data = evtParams.params;\n\n\t\tif (commands[0] == this._commandsPrefix)\n\t\t{\n\t\t\tif (commands.length > 1)\n\t\t\t\tthis.onExtensionCommand(commands[1], data)\n\t\t}\n\t}\n}\n","import {ViewStack} from './components/view-stack';\nimport {ModuleManager} from './managers/module-manager';\nimport {ModuleManagerEvent} from './utils/events';\nimport {ConnectionManager} from './managers/connection-manager';\nimport {ConnectionManagerEvent} from './utils/events';\nimport {ChatManager} from './managers/chat-manager';\nimport {parse} from './utils/dot-properties-parse';\n\nexport class ShellController\n{\n\tconstructor()\n\t{\n\t\t// Set constants and variables\n\n\t\tthis.VERSION_MAJOR = 3;\n\t\tthis.VERSION_MINOR = 2;\n\t\tthis.VERSION_SUB = 1;\n\n\t\tthis.DEFAULT_WS_PORT = 8080;\n\t\tthis.DEFAULT_WSS_PORT = 8443;\n\n\t\tthis.DEFAULT_USERNAME = 'sfsadmin';\n\t\tthis.DEFAULT_PASSWORD = 'sfsadmin';\n\n\t\tthis._loginValidator = null;\n\n\t\t// Display version\n\t\t$('.admin-version').text('v' + this.stringVersion);\n\n\t\t// Create modules manager instance and add event listeners\n\t\tthis._modManager = new ModuleManager(this, $('.nav-main'));\n\t\tthis._modManager.addEventListener(ModuleManagerEvent.MODULE_LOADED, this._onModuleLoaded, this);\n\t\tthis._modManager.addEventListener(ModuleManagerEvent.MODULE_LOAD_ERROR, this._onModuleLoadError, this);\n\n\t\t// Create connection manager instance and add event listeners\n\t\tthis._connManager = new ConnectionManager(this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.CONNECTION, this._onConnection, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.LOGIN, this._onLogin, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.DISCONNECTION, this._onDisconnection, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.ERROR, this._onConnManagerError, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.SERVER_ERROR, this._onServerError, this);\n\n\t\t// Create admin chat manager\n\t\tthis._chatManager = new ChatManager(this);\n\n\t\t/* --------------------------- */\n\n\t\t// Initialize views\n\t\tthis._initViews();\n\n\t\t// Show login view\n\t\tthis._switchShellView('login-view');\n\t}\n\n\t//---------------------------------\n\t// VIEW INITIALIZERS\n\t//---------------------------------\n\n\t/**\n\t * Initialize login view.\n\t */\n\t_initLoginView()\n\t{\n\t\t// Set default input values (password never saved)\n\t\tlet host = window.location.hostname;\n\t\tlet port = this.DEFAULT_WS_PORT;\n\t\tlet encrypt = false;\n\t\tlet user = 'sfsadmin';\n\t\tlet remember = false;\n\n\t\t// Load \"last-server\" cookie\n\t\tlet data = Cookies.getJSON('last-server')\n        if (data)\n\t\t{\n\t\t\thost = data.host;\n            port = data.port;\n\t\t\tencrypt = data.encrypt;\n\t\t\tuser = data.user;\n\t\t\tremember = true;\n        }\n\n\t\t// Retrieve host from GET parameter\n\t\tlet getHost = this._getUrlParameter('host');\n\t\tif (getHost)\n\t\t\thost = getHost;\n\n\t\t// Set input values\n\t\t$('#loginHost').val(host);\n\t\t$('#loginUsername').val(user);\n\t\t$('#loginEncrypt').prop('checked', encrypt);\n\t\t$('#rememberLogin').prop('checked', remember);\n\n\t\t// Initialize numeric input\n\t\t$('#loginPort').kendoNumericTextBox({\n\t\t\tformat: '#####',\n\t\t\tvalue: port\n\t\t});\n\n\t\t// Initialize the Kendo UI Validator on the \"form\" container\n\t\t// (NOTE: does NOT have to be an actual <form> tag)\n\t\tthis._loginValidator = $('#loginForm').kendoValidator({\n\t\t\tvalidateOnBlur: true\n\t\t}).data('kendoValidator');\n\n\t\t// Add listener to validate the form when the Connect button is clicked\n\t\t$('#loginButton').click($.proxy(this._onConnectClick, this));\n\n\t\t// Add listener to submit form on Enter key press\n\t\t$('#loginForm').keyup(function(event) {\n\t\t\tif (event.key !== 'Enter') return;\n\n\t\t\t$('#loginButton').click();\n\t\t\tevent.preventDefault();\n\t\t});\n\n\t\t// Add listener to encrypt checkbox\n\t\t$('#loginEncrypt').change($.proxy(function(event) {\n\t\t\tlet port = this.DEFAULT_WS_PORT;\n\n\t\t\tif ($('#loginEncrypt').prop('checked'))\n\t\t\t\tport = this.DEFAULT_WSS_PORT;\n\n\t\t\t$('#loginPort').data('kendoNumericTextBox').value(port);\n\t\t}, this));\n\n\t\t// Hide error message container\n\t\t$('#login-error').hide();\n\t}\n\n\t/**\n\t * Initialize module view.\n\t */\n\t_initModuleView()\n\t{\n\t\t// Add listeners to open/close menu buttons\n\t\t$('.nav-open').click($.proxy(function(event) {\n\t\t\tevent.preventDefault();\n\t\t\tthis._toggleNav(true);\n\t\t}, this));\n\n\t\t$('.nav-close, .nav-overlay').click($.proxy(function(event) {\n\t\t\tevent.preventDefault();\n\t\t\tthis._toggleNav(false);\n\t\t}, this));\n\n\t\t// Add listeners to service buttons\n\t\t$('#restart-button').click($.proxy(this._onRestartClick, this));\n\t\t$('#halt-button').click($.proxy(this._onHaltClick, this));\n\t\t$('#help-button').click($.proxy(this._onHelpClick, this));\n\t\t$('#disconnect-button').click($.proxy(this._onDisconnectClick, this));\n\n\t\t// Add listener to show tooltip on service buttons hover\n\t\t$('.nav-service').kendoTooltip({\n\t\t\tfilter: 'li[title]'\n\t\t});\n\n\t\t// Add listener to scroll main view to top if status bar is clicked\n\t\t$('#status-bar').click(function(event) {\n\t\t\tevent.preventDefault();\n\t\t\t$('main').animate({ scrollTop: 0 }, 'fast');\n\t\t});\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t/**\n\t * Validate login form and connect+login to SmartFoxServer.\n\t */\n\t_onConnectClick(event)\n\t{\n\t\t// Hide any previous error message\n\t\t$('#login-error').hide();\n\t\t$('#login-error').text('');\n\n\t\t// Validate login form\n\t\tif (this._loginValidator.validate())\n\t\t{\n\t\t\t// Disable login form\n\t\t\tthis._enableLoginForm(false);\n\n\t\t\t// Retrieve connection details\n\t\t\tlet config = {};\n\t\t\tconfig.host = $('#loginHost').val().trim();\n\t\t\tconfig.port = $('#loginPort').data('kendoNumericTextBox').value();\n\t\t\tconfig.useSSL = $('#loginEncrypt').prop('checked');\n\n\t\t\tlet username = $('#loginUsername').val().trim();\n\t\t\tlet password = $('#loginPassword').val();\n\n\t\t\t// Save input values to cookie (except password)...\n\t\t\t// ...or clear previously saved cookie\n\t\t\tif ($('#rememberLogin').prop('checked'))\n\t\t\t{\n\t\t\t\tCookies.set('last-server', {\n\t\t\t\t\thost: config.host,\n\t\t\t\t\tport: config.port,\n\t\t\t\t\tencrypt: config.useSSL,\n\t\t\t\t\tuser: username\n\n\t\t\t\t}, {expires: 30});\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\tCookies.remove('last-server');\n\t\t\t}\n\n\t\t\t// Connect to SFS2X & login\n\t\t\tthis._connManager.connect(config, username, password);\n\t\t}\n\t}\n\n\t/**\n\t * Restart SmartFoxServer.\n\t */\n\t_onRestartClick(event)\n\t{\n\t\tlet message = 'Are you sure you want to stop and restart this instance of SmartFoxServer 2X?';\n\t\tthis.showConfirmWarning(message, $.proxy(this._onRestartConfirmDialogConfirm, this));\n\t}\n\n\t/**\n\t * Halt SmartFoxServer.\n\t */\n\t_onHaltClick(event)\n\t{\n\t\tlet message = 'Are you sure you want to stop this instance of SmartFoxServer 2X?<br>You won\\'t be able to restart it using the Administration Tool.';\n\t\tthis.showConfirmWarning(message, $.proxy(this._onHaltConfirmDialogConfirm, this));\n\t}\n\n\t/**\n\t * Open online documentation.\n\t */\n\t_onHelpClick(event)\n\t{\n\t\t// Open online doc\n\t\twindow.open(`http://docs2x.smartfoxserver.com/GettingStarted/admintool-${this._modManager.currentModuleId}`, '_blank');\n\t}\n\n\t/**\n\t * Disconnect from server.\n\t */\n\t_onDisconnectClick(event)\n\t{\n\t\tthis._connManager.disconnect();\n\t}\n\n\t//------------------------------------\n\t// CONNECTION MANAGER EVENT LISTENERS\n\t//------------------------------------\n\n\t_onConnection(evtParams)\n\t{\n\t\t// Log message\n\t\tthis.logMessage(`Connection to ${this._connManager.smartFox.config.host}:${this._connManager.smartFox.config.port} established`);\n\t}\n\n\t_onLogin(evtParams)\n\t{\n\t\t// Log message\n\t\tthis.logMessage(`Successful login to ${this._connManager.smartFox.config.host}:${this._connManager.smartFox.config.port} performed`);\n\n\t\t// Get last loaded module from cookies\n\t\tlet loadModuleId = null;\n\t\tlet data = Cookies.getJSON('last-module')\n        if (data)\n\t\t\tloadModuleId = data.id;\n\n\t\t// Init the modules list with the configuration returned by the server and: load last saved module, or first module in the list, or passed module id\n\t\tthis._modManager.initModulesList(evtParams.modules, loadModuleId);\n\n\t\t// Save current uptime\n\t\tthis._uptime = evtParams.serverUptime;\n\n\t\t// Show/hide Halt and Restart buttons depending if:\n\t\t// 1) this feature is supported for the server operating system\n\t\t// 2) the administrator who just logged in has the permission to execute these actions\n\t\tthis._showHaltRestartButtons(evtParams.procControlEnabled && evtParams.allowHalt);\n\n\t\t// Switch to modules view\n\t\tthis._goToModulesView(evtParams.serverVersion, evtParams.serverName);\n\n\t\t// If default username and password have been used...\n\t\tif ($('#loginUsername').val() == this.DEFAULT_USERNAME && $('#loginPassword').val() == this.DEFAULT_PASSWORD)\n\t\t{\n\t\t\t// ...show alert\n\t\t\tthis.showSimpleAlert('You are using the default administration profile which is highly insecure, please make sure to immediately change the password.');\n\n\t\t\t// ...show non-removable message in alert bar\n\t\t\t$('#alert-bar').show();\n\t\t\t$('#alert-bar').text('You are using the default administration profile which is highly insecure, please change the password.');\n\t\t}\n\t}\n\n\t/**\n\t * Listener called when the user disconnected voluntarily.\n\t */\n\t_onDisconnection(evtParams)\n\t{\n\t\t// Remove any popup or alert\n\t\tthis.removeDialog();\n\n\t\t// Switch back to login view\n\t\tthis._goToLoginView();\n\n\t\t// Hide navigation if open\n\t\tthis._toggleNav(false);\n\t}\n\n\t/**\n\t * Listener called when an error caused a disconnection.\n\t */\n\t_onConnManagerError(evtParams)\n\t{\n\t\t// Remove any popup or alert\n\t\tthis.removeDialog();\n\n\t\t// Log system message\n\t\tthis.logMessage(evtParams.message, 'warn');\n\n\t\t// Switch back to login view\n\t\tthis._goToLoginView();\n\n\t\t// Display error in login view\n\t\t$('#login-error').text(evtParams.message);\n\t\t$('#login-error').show();\n\t}\n\n\t/**\n\t * Listener called when an unexpected server-side error occurs.\n\t */\n\t_onServerError(evtParams)\n\t{\n\t\t// Show an alert\n\t\tthis.showSimpleAlert(evtParams.message);\n\t}\n\n\t//---------------------------------\n\t// OTHER EVENT LISTENERS\n\t//---------------------------------\n\n\t_onModuleLoaded(evtParams)\n\t{\n\t\tconst moduleData = evtParams.moduleData;\n\n\t\t// Save last loaded module to cookies\n\t\tCookies.set('last-module', {\n\t\t\tid: moduleData.id\n\t\t}, {expires: 30});\n\n\t\t// Display module title\n\t\t$('#module-title').show();\n\t\t$('#module-title-label').text(moduleData.name);\n\n\t\t// Tell the chat manager which module has been loaded\n\t\tthis._chatManager.setCurrentModule(moduleData.id);\n\n\t\t// Assign the .selected class to the selected navigation item\n\t\t$('.nav-main').find(`[data-id='${moduleData.id}']`).addClass('selected').siblings('.selected').removeClass('selected');\n\t}\n\n\t_onModuleLoadError(evtParams)\n\t{\n\t\t// Show an alert\n\t\tthis.showSimpleAlert(evtParams.message);\n\t}\n\n\t_onRestartConfirmDialogConfirm()\n\t{\n\t\t// Send restart server request\n\t\tthis._connManager.restartServer();\n\t}\n\n\t_onHaltConfirmDialogConfirm()\n\t{\n\t\t// Send halt server request\n\t\tthis._connManager.haltServer();\n\t}\n\n\t//---------------------------------\n\t// PUBLIC METHODS\n\t// This members are used by the sub-managers (i.e. ConnectionManager)\n\t// or the modules to communicate with this shell controller.\n\t//---------------------------------\n\n\tget smartFox()\n\t{\n\t\treturn this._connManager.smartFox;\n\t}\n\n\tget intVersion()\n\t{\n\t\tvar version = this.VERSION_MAJOR;\n\t\tversion += (this.VERSION_MINOR < 10 ? '0' : '') + this.VERSION_MINOR;\n\t\tversion += (this.VERSION_SUB < 10 ? '0' : '') + this.VERSION_SUB;\n\n\t\treturn Number(version);\n\t}\n\n\tget stringVersion()\n\t{\n\t\treturn this.VERSION_MAJOR + '.' + this.VERSION_MINOR + '.' + this.VERSION_SUB;\n\t}\n\n\tlogMessage(message, type = 'log')\n\t{\n\t\tswitch (type) {\n\t\t\tcase 'info':\n\t\t\t\tconsole.info('[ ADMINTOOL | INFO ] ' + message);\n\t\t\t\tbreak;\n\t\t\tcase 'warn':\n\t\t\t\tconsole.warn('[ ADMINTOOL | WARN ] ' + message);\n\t\t\t\tbreak;\n\t\t\tcase 'error':\n\t\t\t\tconsole.error('[ ADMINTOOL | ERROR ] ' + message);\n\t\t\t\tbreak;\n\t\t\tdefault:\n\t\t\t\tconsole.log('[ ADMINTOOL | INFO ] ' + message);\n\t\t}\n\t}\n\n\tremoveDialog()\n\t{\n\t\t// Hide any showing modal\n\t\t$('.modal').modal('hide');\n\n\t\t// Hide any showing toast\n\t\t$('.toast').toast('hide');\n\n\t\t// Remove shell's dialog\n\t\tif (this._dialog != null)\n\t\t{\n\t\t\tthis._dialog.close();\n\t\t\tthis._dialog.destroy();\n\t\t\tthis._dialog = null;\n\t\t}\n\n\t\t// Enable the following if other Kendo dialogs are used in modules\n\t\t/*\n\t\t// Remove any other dialog (inner to module)\n\t\t$('.k-dialog-content').each(function(index) {\n\t\t\t// Confirm dialog\n\t\t\tlet confirm = $(this).data('kendoConfirm');\n\t\t\tif (confirm)\n\t\t\t{\n\t\t\t\tconfirm.close();\n\t\t\t\tconfirm.destroy();\n\t\t\t}\n\t\t});\n\t\t*/\n\t}\n\n\t/**\n\t * Show simple alert.\n\t */\n\tshowSimpleAlert(text, isWarning = true)\n\t{\n\t\t// Create and show dialog\n\t\tthis._dialog = kendo.alert(text);\n\t\tthis._dialog.title(isWarning ? 'Warning' : 'Information');\n\n\t\t// Set custom class for styling\n\t\tthis._dialog.wrapper.addClass(isWarning ? 'is-warning' : 'is-info');\n\n\t\t// Log message too\n\t\t// (we encapsule the text in a span and extract the text again to remove inner html tags)\n\t\tlet html = $('<span>' + text + '</span>');\n\t\tthis.logMessage(html.text(), isWarning ? 'warn' : 'info');\n\t}\n\n\t/**\n\t * Show confirm alert.\n\t */\n\tshowConfirmWarning(text, confirmHandler)\n\t{\n\t\t// Create dialog\n\t\tthis._dialog = $('<div></div>').kendoConfirm({\n\t      title: 'Warning',\n\t      content: text,\n\t\t  actions: [\n\t          {\n\t              text: 'OK',\n\t              primary: true,\n\t              action: confirmHandler\n\t          },\n\t      ]\n\t  \t}).data('kendoConfirm');\n\n\t\t// Set custom class for styling\n\t\tthis._dialog.wrapper.addClass('is-warning');\n\n\t\t// Show dialog\n\t\tthis._dialog.open();\n\t}\n\n\tshowNotification(title, message)\n\t{\n\t\t// Display notification\n\t\tlet toast = $(`\n\t\t\t<div class=\"toast\" role=\"alert\" aria-live=\"assertive\" aria-atomic=\"true\" data-delay=\"4000\">\n\t\t\t\t<div class=\"toast-header\">\n\t\t\t\t\t<strong class=\"mr-auto\">${title}</strong>\n\t\t\t\t\t<button type=\"button\" class=\"ml-2 mb-1 close\" data-dismiss=\"toast\" aria-label=\"Close\">\n\t\t\t\t\t\t<span aria-hidden=\"true\">&times;</span>\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t\t<div class=\"toast-body\">${message}</div>\n\t\t\t</div>\n\t\t`);\n\n\t\t$('.toast-container').append(toast);\n\t\ttoast.toast('show');\n\n\t\t// Log message too\n\t\t// (we encapsule the text in a span and extract the text again to remove inner html tags)\n\t\tlet html = $('<span>' + title + ' - ' + message + '</span>');\n\t\tthis.logMessage(html.text(), 'info');\n\t}\n\n\tupdateModuleTitle(title, asSuffix = false)\n\t{\n\t\t$('#module-title-label').text( (asSuffix ? $('#module-title-label').text() : '') + title );\n\t}\n\n\t//---------------------------------\n\t// PRIVATE METHODS\n\t//---------------------------------\n\n\t_initViews()\n\t{\n\t\t// Initialize login view\n\t\tthis._initLoginView();\n\n\t\t// Initialize module view\n\t\tthis._initModuleView();\n\t}\n\n\t/**\n\t * Switch view in top view stack.\n\t * @param  {string} viewId Identifier of the viewstask child to display.\n\t */\n\t_switchShellView(viewId)\n\t{\n\t\tdocument.getElementById('shell').selectedElement = document.getElementById(viewId);\n\n\t\t// (jQuery alternative)\n\t\t// $('#shell')[0].selectedElement = $('#shell').children('#' + viewId)[0];\n\t}\n\n\t/**\n\t * Enable/disable login form.\n\t */\n\t_enableLoginForm(enable)\n\t{\n\t\t// Enable/disable fieldset (works for all non-kendo inputs)\n\t\t$('#loginForm fieldset').prop('disabled', !enable);\n\n\t\t// Enable/disable numeric textbox\n\t\t$('#loginPort').data('kendoNumericTextBox').enable(enable);\n\t}\n\n\t_toggleNav(bool)\n\t{\n\t\t$('.nav-container, .nav-overlay').toggleClass('is-visible', bool);\n\t\t$('.module-container').toggleClass('scale-down', bool);\n\t}\n\n\t_getUrlParameter(param)\n\t{\n\t\tlet pageURL = window.location.search.substring(1);\n\t\tlet urlVariables = pageURL.split('&');\n\n\t\tfor (var i = 0; i < urlVariables.length; i++) {\n\t\t\tlet paramName = urlVariables[i].split('=');\n\t\t\tif (paramName[0] == param && paramName[1] != '')\n\t\t\t\treturn paramName[1];\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t_goToLoginView()\n\t{\n\t\t// Clear password field\n\t\t$('#loginPassword').val('');\n\n\t\t// Switch to login view\n\t\tthis._switchShellView('login-view');\n\n\t\t// Unload current module\n\t\tthis._modManager.unloadModule();\n\n\t\t// Enable login form\n\t\tthis._enableLoginForm(true);\n\n\t\t// Remove uptime updater\n\t\tclearInterval(this._uptimeTimer);\n\t}\n\n\t_goToModulesView(serverVersion, serverName)\n\t{\n\t\t// Hide module title\n\t\t$('#module-title').hide();\n\t\t$('#module-title-label').text('');\n\n\t\t// Hide alert bar\n\t\t$('#alert-bar').hide();\n\n\t\t// Show server version in the header\n\t\t$('#sfs-version-value').text(serverVersion);\n\n\t\t// Set server name, IP and port in module's title bar\n\t\tlet host = `${this._connManager.smartFox.config.host}:${this._connManager.smartFox.config.port}` + (serverName != '' ? ` [${serverName}]` : '');\n\t\t$('#host-label').text(host);\n\n\t\t// Save current SFS version\n\t\tthis._currentSfsVersion = serverVersion;\n\n\t\t// Check new SFS version availability\n\t\tthis._checkAvailableSfsVersion();\n\n\t\t// Start uptime updater\n\t\tthis._uptimeTimer = setInterval($.proxy(this._updateUptime, this), 1000);\n\n\t\t// Switch to modules view\n\t\tthis._switchShellView('module-view');\n\t}\n\n\t_showHaltRestartButtons(show)\n\t{\n\t\tif (show)\n\t\t{\n\t\t\t$('#restart-button').show();\n\t\t\t$('#halt-button').show();\n\t\t}\n\t\telse\n\t\t{\n\t\t\t$('#restart-button').hide();\n\t\t\t$('#halt-button').hide();\n\t\t}\n\t}\n\n\t_checkAvailableSfsVersion()\n\t{\n\t\t// Remove class to hide update button style and notification icon\n\t\t$('#sfs-version').removeClass('is-active');\n\n\t\t// Remove update button click listeners\n\t\t$('#sfs-version-button').off('click');\n\n\t\t// Load file containing latest SFS version info\n\t\t$.ajax({\n\t\t\ttype: 'GET',\n\t\t\turl: 'https://www.smartfoxserver.com/downloads/sfs2x/latestVersion.txt',\n        \tdataType: 'text',\n\t\t\tsuccess: $.proxy(this._onLatestSfsVersionInfoLoaded, this),\n\t\t\terror: $.proxy(function() {\n\t\t\t\tthis.logMessage('Unable to check new server version availability on SmartFoxServer website', 'warn');\n\t\t\t}, this)\n\t\t});\n\t}\n\n\t_onLatestSfsVersionInfoLoaded(data)\n\t{\n\t\t// Parse returned data\n\t\tconst v = parse(data);\n\n\t\tif (v.version != null && this._isSfsVersionNewer(v.version, this._currentSfsVersion))\n\t\t{\n\t\t\tthis.tempflag = true;\n\t\t\tthis.logMessage('An updated version of SmartFoxServer 2X is available for download', 'info');\n\n\t\t\t// Set upgrade dialog details\n\t\t\tthis._sfsUpdateDetails = v;\n\n\t\t\t// Add listener to show SFS version update modal\n\t\t\t$('#sfs-version-button').click(function() {\n\t\t\t\t$('#serverUpdateModal').modal({\n\t\t\t\t\tbackdrop: 'static',\n\t\t\t\t\tkeyboard: false,\n\t\t\t\t});\n\t\t\t});\n\n\t\t\t// Set class to show update button style and notification icon\n\t\t\t$('#sfs-version').addClass('is-active');\n\n\t\t\t// Update modal content\n\t\t\tconst newVer = this._sfsUpdateDetails;\n\t\t\tconst currVer = this._currentSfsVersion;\n\n\t\t\tlet modal = $('#serverUpdateModal');\n\n\t\t\t// Update modal content\n\t\t\t$('#newVersLb', modal).text(newVer.version);\n\t\t\t$('#updTypeLb', modal).text(newVer.isPatch ? 'patch' : 'full installer');\n\t\t\t$('#reqVersLb', modal).text(newVer.isPatch ? ' (requires version ' + newVer.requires + ' or later)' : '');\n\t\t\t$('#currVersLb', modal).text(currVer);\n\t\t\t$('#serverUpdateModalLink', modal).attr('href', newVer.url);\n\t\t}\n\t}\n\n\t_isSfsVersionNewer(availableVer, currentVer)\n\t{\n\t\tconst MAJ = 0;\n\t\tconst MIN = 1;\n\t\tconst SUB = 2;\n\n\t\tconst available = availableVer.split('.');\n\t\tconst current = currentVer.split('.');\n\n\t\t// Check major version\n\t\tif (available[MAJ] > current[MAJ])\n\t\t\treturn true;\n\t\telse if (available[MAJ] == current[MAJ])\n\t\t{\n\t\t\t// Check minor version\n\t\t\tif (available[MIN] > current[MIN])\n\t\t\t\treturn true;\n\t\t\telse if (available[MIN] == current[MIN])\n\t\t\t{\n\t\t\t\t// Check sub version\n\t\t\t\tif (available[SUB] > current[SUB])\n\t\t\t\t\treturn true;\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t_updateUptime()\n\t{\n\t\tlet days = this._uptime[0];\n\t\tlet hours = this._uptime[1];\n\t\tlet minutes = this._uptime[2];\n\t\tlet seconds = this._uptime[3] + 1;\n\n\t\tif (seconds > 59)\n\t\t{\n\t\t\tseconds = 0;\n\t\t\tminutes += 1;\n\n\t\t\tif (minutes > 59)\n\t\t\t{\n\t\t\t\tminutes = 0;\n\t\t\t\thours += 1;\n\n\t\t\t\tif (hours > 23)\n\t\t\t\t{\n\t\t\t\t\thours = 0;\n\t\t\t\t\tdays += 1;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tthis._uptime[3] = seconds;\n\t\tthis._uptime[2] = minutes;\n\t\tthis._uptime[1] = hours;\n\t\tthis._uptime[0] = days;\n\n\t\t// Send updated uptime to current module (if loaded)\n\t\tlet module = this._modManager.currentModule;\n\t\tif (module != null)\n\t\t\tmodule.onUptimeUpdated(this._uptime);\n\t}\n}\n","const atComment = (src, offset) => {\n  const ch = src[offset]\n  return ch === '#' || ch === '!'\n}\n\nconst atLineEnd = (src, offset) => {\n  const ch = src[offset]\n  return !ch || ch === '\\r' || ch === '\\n'\n}\n\nconst endOfIndent = (src, offset) => {\n  let ch = src[offset]\n  while (ch === '\\t' || ch === '\\f' || ch === ' ') {\n    offset += 1\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfComment = (src, offset) => {\n  let ch = src[offset]\n  while (ch && ch !== '\\r' && ch !== '\\n') {\n    offset += 1\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfKey = (src, offset) => {\n  let ch = src[offset]\n  while (ch && ch !== '\\r' && ch !== '\\n' && ch !== '\\t' && ch !== '\\f' && ch !== ' ' && ch !== ':' && ch !== '=') {\n    if (ch === '\\\\') {\n      if (src[offset + 1] === '\\n') {\n        offset = endOfIndent(src, offset + 2)\n      } else {\n        offset += 2\n      }\n    } else {\n      offset += 1\n    }\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfSeparator = (src, offset) => {\n  let ch = src[offset]\n  let hasEqSign = false\n  loop: while (ch === '\\t' || ch === '\\f' || ch === ' ' || ch === '=' || ch === ':' || ch === '\\\\') {\n    switch (ch) {\n      case '\\\\':\n        if (src[offset + 1] !== '\\n') break loop\n        offset = endOfIndent(src, offset + 2)\n        break\n      case '=':\n      case ':':\n        if (hasEqSign) break loop\n        hasEqSign = true\n        // fallthrough\n      default:\n        offset += 1\n    }\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfValue = (src, offset) => {\n  let ch = src[offset]\n  while (ch && ch !== '\\r' && ch !== '\\n') {\n    offset += ch === '\\\\' ? 2 : 1\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst unescape = (str) => str.replace(/\\\\(u[0-9a-fA-F]{4}|\\r?\\n[ \\t\\f]*|.)?/g, (match, code) => {\n  switch (code && code[0]) {\n    case 'f': return '\\f'\n    case 'n': return '\\n'\n    case 'r': return '\\r'\n    case 't': return '\\t'\n    case 'u':\n      const c = parseInt(code.substr(1), 16)\n      return isNaN(c) ? code : String.fromCharCode(c)\n    case '\\r':\n    case '\\n':\n    case undefined:\n      return ''\n    default:\n      return code\n  }\n})\n\n/**\n * Splits the input string into an array of logical lines\n *\n * Key-value pairs are [key, value] arrays with string values. Escape sequences\n * in keys and values are parsed. Empty lines are included as empty strings, and\n * comments as strings that start with '#' or '! characters. Leading whitespace\n * is not included.\n *\n * @see https://docs.oracle.com/javase/9/docs/api/java/util/Properties.html#load(java.io.Reader)\n *\n * @param {string} src\n * @returns Array<string | string[]]>\n */\nfunction parseLines (src) {\n  const lines = []\n  for (i = 0; i < src.length; ++i) {\n    if (src[i] === '\\n' && src[i - 1] === '\\r') i += 1\n    if (!src[i]) break\n    const keyStart = endOfIndent(src, i)\n    if (atLineEnd(src, keyStart)) {\n      lines.push('')\n      i = keyStart\n      continue\n    }\n    if (atComment(src, keyStart)) {\n      const commentEnd = endOfComment(src, keyStart)\n      lines.push(src.slice(keyStart, commentEnd))\n      i = commentEnd\n      continue\n    }\n    const keyEnd = endOfKey(src, keyStart)\n    const key = unescape(src.slice(keyStart, keyEnd))\n    const valueStart = endOfSeparator(src, keyEnd)\n    if (atLineEnd(src, valueStart)) {\n      lines.push([key, ''])\n      i = valueStart\n      continue\n    }\n    const valueEnd = endOfValue(src, valueStart)\n    const value = unescape(src.slice(valueStart, valueEnd))\n    lines.push([key, value])\n    i = valueEnd\n  }\n  return lines\n}\n\n/**\n * Parses an input string read from a .properties file into a JavaScript Object\n *\n * If the second `path` parameter is true, dots '.' in keys will result in a\n * multi-level object (use a string value to customise). If a parent level is\n * directly assigned a value while it also has a child with an assigned value,\n * the parent value will be assigned to its empty string '' key. Repeated keys\n * will take the last assigned value. Key order is not guaranteed, but is likely\n * to match the order of the input lines.\n *\n * @param {string} src\n * @param {boolean | string} [path=false]\n */\nfunction parse (src, path) {\n  const pathSep = typeof path === 'string' ? path : '.'\n  return parseLines(src).reduce((res, line) => {\n    if (Array.isArray(line)) {\n      const [key, value] = line\n      if (path) {\n        const keyPath = key.split(pathSep)\n        let parent = res\n        while (keyPath.length >= 2) {\n          const p = keyPath.shift()\n          if (!parent[p]) {\n            parent[p] = {}\n          } else if (typeof parent[p] !== 'object') {\n            parent[p] = { '': parent[p] }\n          }\n          parent = parent[p]\n        }\n        const leaf = keyPath[0]\n        if (typeof parent[leaf] === 'object') {\n          parent[leaf][''] = value\n        } else {\n          parent[leaf] = value\n        }\n      } else {\n        res[key] = value\n      }\n    }\n    return res\n  }, {})\n}\n\nmodule.exports = { parse, parseLines }\n","export class EventDispatcher {\n\tconstructor() {\n\t\tthis._listenersByEvent = {};\n\t}\n\n\t/**\n\t * Registers an event listener function that will receive notification of an event.\n\t *\n\t * <p>If you no longer need an event listener, remove it by calling the <em>removeEventListener()</em> method, or memory issues could arise.\n\t * In fact event listeners are not automatically removed from memory.</p>\n\t *\n\t * @param\t{string} evtType\tThe type of event to listen to.\n\t * @param\t{function} callback\tThe listener function that processes the event. This function should accept an object as its only parameter, which in turn contains the event parameters.\n\t * @param\t{object} scope\t\tThe object that acts as a context for the event listener: it is the object that acts as a \"parent scope\" for the callback function, thus providing context (i.e. access to variables and other mehtods) to the function itself.\n\t */\n\taddEventListener(evtType, callback, scope)\n\t{\n\t\tif (!this._listenersByEvent[evtType])\n\t\t\tthis._listenersByEvent[evtType] = [];\n\n\t\tthis._listenersByEvent[evtType].push({callback:callback, scope:scope});\n\t}\n\n\t/**\n\t * Removes an event listener.\n\t *\n\t * @param\t{string} evtType\tThe type of event to remove.\n\t * @param\t{function} callback\tThe listener function to be removed.\n\t */\n\tremoveEventListener(evtType, callback)\n\t{\n\t\tconst listeners = this._listenersByEvent[evtType];\n\n\t\tif (listeners)\n\t\t{\n\t\t\tfor (let i = 0; i < listeners.length; i++)\n\t\t\t{\n\t\t\t\tif (listeners[i].callback === callback)\n\t\t\t\t{\n\t\t\t\t\tlisteners.splice(i, 1);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tdispatchEvent(evtType, evtObj)\n\t{\n\t\tconst listeners = this._listenersByEvent[evtType];\n\n\t\tif (listeners)\n\t\t{\n\t\t\tfor (let listener of listeners)\n\t\t\t\tlistener.callback.call(listener.scope, evtObj);\n\t\t}\n\t}\n}\n","export const ConnectionManagerEvent = Object.freeze({\n\tCONNECTION: 'connection',\n\tLOGIN: 'login',\n\tDISCONNECTION: 'disconnection',\n\tERROR: 'error',\n\tSERVER_ERROR: 'serverError',\n});\n\nexport const ModuleManagerEvent = Object.freeze({\n\tMODULE_LOADED: 'module-loaded',\n\tMODULE_LOAD_ERROR: 'module-load-error',\n});\n","export const toKebabCase = (str) =>\n  str &&\n  str\n    .match(/[A-Z]{2,}(?=[A-Z][a-z]+[0-9]*|\\b)|[A-Z]?[a-z]+[0-9]*|[A-Z]|[0-9]+/g)\n    .map(x => x.toLowerCase())\n    .join('-');\n\nexport function bytesToSize(bytes, decimals = 1, zeroUnit = '', suffix = '') {\n\tif (bytes === 0)\n\t\treturn '0 ' + zeroUnit + (zeroUnit != '' ? suffix : '');\n\n\tif (bytes < 1) // Can happen in chart axis labels!\n\t \treturn `${bytes} Bytes${suffix}`;\n\n    const k = 1000;\n    const dm = decimals < 0 ? 0 : decimals;\n    const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];\n\n    const i = Math.floor(Math.log(bytes) / Math.log(k));\n\n    return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i] + suffix;\n}\n\nexport function kBytesToSize(kBytes, decimals = 1, zeroUnit = '', suffix = '') {\n\treturn bytesToSize(kBytes * 1000, decimals, zeroUnit, suffix);\n}\n\nexport function capitalizeFirst(string) {\n\treturn string.charAt(0).toUpperCase() + string.slice(1);\n}\n\nexport function filterClassName(element, index, array)\n{\n\tif (element.endsWith('.py'))\n\t\treturn (element.endsWith('Extension.py'));\n\telse if (element.endsWith('.js'))\n\t\treturn (element.endsWith('Extension.js'));\n\telse\n\t\treturn (element.endsWith('Extension'));\n}\n\nexport function roundToDecimals(value, decimals) {\n  return Number(Math.round(value + 'e' + decimals) + 'e-' + decimals);\n}\n\nexport function scaleBytes(bytes, decimals = 1) {\n\tlet obj = {};\n\n\tconst sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];\n\n\tif (bytes > 0)\n\t{\n\t\tconst k = 1000;\n\t\tconst dm = decimals < 0 ? 0 : decimals;\n\t\tconst i = Math.floor(Math.log(bytes) / Math.log(k));\n\n\t\tobj.value = parseFloat((bytes / Math.pow(k, i)).toFixed(dm));\n\t\tobj.unit = sizes[i];\n\t}\n\telse\n\t{\n\t\tobj.value = 0;\n\t\tobj.unit = sizes[0];\n\t}\n\n\treturn obj;\n}\n"],"mappings":";;;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;;;;;;;;;;;;;ACrMA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACRA;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;;;;;;;;;;;;;ACvEA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACrBA;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;;;;;;;;;;;;;AC/LA;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;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACtOA;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;;;;;;;;;;;;;;ACvGA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC7vBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACxLA;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;;;;;;;;;;;;;ACxDA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACXA;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;;;;;;;;;;;;;;;;A","sourceRoot":""}
2208
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/application.bundle.js","sources":["webpack://application/webpack/bootstrap","webpack://application/./src/application.js","webpack://application/./src/components/view-stack.js","webpack://application/./src/managers/chat-manager.js","webpack://application/./src/managers/connection-manager.js","webpack://application/./src/managers/module-manager.js","webpack://application/./src/modules/base-module.js","webpack://application/./src/shell-controller.js","webpack://application/./src/utils/dot-properties-parse.js","webpack://application/./src/utils/event-dispatcher.js","webpack://application/./src/utils/events.js","webpack://application/./src/utils/utilities.js"],"sourcesContent":[" \t// install a JSONP callback for chunk loading\n \tfunction webpackJsonpCallback(data) {\n \t\tvar chunkIds = data[0];\n \t\tvar moreModules = data[1];\n\n\n \t\t// add \"moreModules\" to the modules object,\n \t\t// then flag all \"chunkIds\" as loaded and fire callback\n \t\tvar moduleId, chunkId, i = 0, resolves = [];\n \t\tfor(;i < chunkIds.length; i++) {\n \t\t\tchunkId = chunkIds[i];\n \t\t\tif(Object.prototype.hasOwnProperty.call(installedChunks, chunkId) && installedChunks[chunkId]) {\n \t\t\t\tresolves.push(installedChunks[chunkId][0]);\n \t\t\t}\n \t\t\tinstalledChunks[chunkId] = 0;\n \t\t}\n \t\tfor(moduleId in moreModules) {\n \t\t\tif(Object.prototype.hasOwnProperty.call(moreModules, moduleId)) {\n \t\t\t\tmodules[moduleId] = moreModules[moduleId];\n \t\t\t}\n \t\t}\n \t\tif(parentJsonpFunction) parentJsonpFunction(data);\n\n \t\twhile(resolves.length) {\n \t\t\tresolves.shift()();\n \t\t}\n\n \t};\n\n\n \t// The module cache\n \tvar installedModules = {};\n\n \t// object to store loaded and loading chunks\n \t// undefined = chunk not loaded, null = chunk preloaded/prefetched\n \t// Promise = chunk loading, 0 = chunk loaded\n \tvar installedChunks = {\n \t\t\"application\": 0\n \t};\n\n\n\n \t// script path function\n \tfunction jsonpScriptSrc(chunkId) {\n \t\treturn __webpack_require__.p + \"assets/js/core/modules/\" + ({\"module-10\":\"module-10\",\"module-11\":\"module-11\",\"module-12~module-13~module-9\":\"module-12~module-13~module-9\",\"module-12\":\"module-12\",\"module-3\":\"module-3\",\"module-6\":\"module-6\",\"vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8\":\"vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8\",\"module-1\":\"module-1\",\"module-13\":\"module-13\",\"module-4\":\"module-4\",\"module-5\":\"module-5\",\"module-7\":\"module-7\",\"module-8\":\"module-8\",\"vendors~module-0\":\"vendors~module-0\",\"module-0\":\"module-0\",\"vendors~module-9\":\"vendors~module-9\",\"module-9\":\"module-9\"}[chunkId]||chunkId) + \".bundle.js\"\n \t}\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n \t// This file contains only the entry chunk.\n \t// The chunk loading function for additional chunks\n \t__webpack_require__.e = function requireEnsure(chunkId) {\n \t\tvar promises = [];\n\n\n \t\t// JSONP chunk loading for javascript\n\n \t\tvar installedChunkData = installedChunks[chunkId];\n \t\tif(installedChunkData !== 0) { // 0 means \"already installed\".\n\n \t\t\t// a Promise means \"currently loading\".\n \t\t\tif(installedChunkData) {\n \t\t\t\tpromises.push(installedChunkData[2]);\n \t\t\t} else {\n \t\t\t\t// setup Promise in chunk cache\n \t\t\t\tvar promise = new Promise(function(resolve, reject) {\n \t\t\t\t\tinstalledChunkData = installedChunks[chunkId] = [resolve, reject];\n \t\t\t\t});\n \t\t\t\tpromises.push(installedChunkData[2] = promise);\n\n \t\t\t\t// start chunk loading\n \t\t\t\tvar script = document.createElement('script');\n \t\t\t\tvar onScriptComplete;\n\n \t\t\t\tscript.charset = 'utf-8';\n \t\t\t\tscript.timeout = 120;\n \t\t\t\tif (__webpack_require__.nc) {\n \t\t\t\t\tscript.setAttribute(\"nonce\", __webpack_require__.nc);\n \t\t\t\t}\n \t\t\t\tscript.src = jsonpScriptSrc(chunkId);\n\n \t\t\t\t// create error before stack unwound to get useful stacktrace later\n \t\t\t\tvar error = new Error();\n \t\t\t\tonScriptComplete = function (event) {\n \t\t\t\t\t// avoid mem leaks in IE.\n \t\t\t\t\tscript.onerror = script.onload = null;\n \t\t\t\t\tclearTimeout(timeout);\n \t\t\t\t\tvar chunk = installedChunks[chunkId];\n \t\t\t\t\tif(chunk !== 0) {\n \t\t\t\t\t\tif(chunk) {\n \t\t\t\t\t\t\tvar errorType = event && (event.type === 'load' ? 'missing' : event.type);\n \t\t\t\t\t\t\tvar realSrc = event && event.target && event.target.src;\n \t\t\t\t\t\t\terror.message = 'Loading chunk ' + chunkId + ' failed.\\n(' + errorType + ': ' + realSrc + ')';\n \t\t\t\t\t\t\terror.name = 'ChunkLoadError';\n \t\t\t\t\t\t\terror.type = errorType;\n \t\t\t\t\t\t\terror.request = realSrc;\n \t\t\t\t\t\t\tchunk[1](error);\n \t\t\t\t\t\t}\n \t\t\t\t\t\tinstalledChunks[chunkId] = undefined;\n \t\t\t\t\t}\n \t\t\t\t};\n \t\t\t\tvar timeout = setTimeout(function(){\n \t\t\t\t\tonScriptComplete({ type: 'timeout', target: script });\n \t\t\t\t}, 120000);\n \t\t\t\tscript.onerror = script.onload = onScriptComplete;\n \t\t\t\tdocument.head.appendChild(script);\n \t\t\t}\n \t\t}\n \t\treturn Promise.all(promises);\n \t};\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// on error function for async loading\n \t__webpack_require__.oe = function(err) { console.error(err); throw err; };\n\n \tvar jsonpArray = window[\"webpackJsonpapplication\"] = window[\"webpackJsonpapplication\"] || [];\n \tvar oldJsonpFunction = jsonpArray.push.bind(jsonpArray);\n \tjsonpArray.push = webpackJsonpCallback;\n \tjsonpArray = jsonpArray.slice();\n \tfor(var i = 0; i < jsonpArray.length; i++) webpackJsonpCallback(jsonpArray[i]);\n \tvar parentJsonpFunction = oldJsonpFunction;\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = \"./src/application.js\");\n","import {ShellController} from './shell-controller';\n\n$( document ).ready(function()\n{\n    console.info(\"SmartFoxServer 2X AdminTool ready!\");\n\n\t// Create shell controller instance\n\tthis.controller = new ShellController();\n});\n","export class ViewStack 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\n\t\t\t\t::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t      display: none !important;\n\t\t\t    }\n\t\t\t</style>\n\t\t\t<slot></slot>\n\t\t`;\n\n\t\t// Select first item\n\t\tthis.selectedIndex = 0;\n\t}\n\n\tget selectedElement()\n\t{\n\t\treturn this._selectedElement;\n\t}\n\n\tset selectedElement(element)\n\t{\n\t\tif (element != null && element.parentNode == this)\n\t\t{\n\t\t\tthis._selectedElement = element;\n\n\t\t\tfor (let element of this.children)\n\t\t\t{\n\t\t\t\tif (element == this._selectedElement)\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 ViewStack');\n\t\t}\n\t}\n\n\tget selectedIndex()\n\t{\n\t\treturn Array.from(this.children).indexOf(this._selectedElement);\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 ViewStack index\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet element = this.children[index];\n\t\t\tthis.selectedElement = element;\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('view-stack'))\n\twindow.customElements.define('view-stack', ViewStack);\n","export class ChatManager\n{\n\tconstructor(shellCtrl)\n\t{\n\t\tthis.USERVAR_MODULE = 'mod';\n\n\t\tthis.shellCtrl = shellCtrl;\n\n\t\t// TODO Implement chat manager after creating the chat UI in main shell\n\t}\n\n\t/**\n\t * Called by the shell when the user loads a new module, so that this info\n\t * can be saved in the user variables and displayed in the chat's userlist.\n\t */\n\tsetCurrentModule(moduleId)\n\t{\n\t\t// Save module id in user variables\n\t\tlet userVar = new SFS2X.SFSUserVariable(this.USERVAR_MODULE, moduleId);\n\t\tthis.shellCtrl.smartFox.send(new SFS2X.SetUserVariablesRequest([userVar]));\n\t}\n}\n","import {EventDispatcher} from '../utils/event-dispatcher';\nimport {ConnectionManagerEvent} from '../utils/events';\n\nexport class ConnectionManager extends EventDispatcher\n{\n\tconstructor(shellCtrl)\n\t{\n\t\tsuper();\n\n\t\tthis.ADMIN_ZONE_NAME = \"--=={{{ AdminZone }}}==--\";\n\t\tthis.EXTENSION_ERROR = \"error\";\n\n\t\tthis.COMMANDS_PREFIX = \"admin\";\n\t\tthis.COMMAND_RESTART = \"restart\";\n\t\tthis.COMMAND_HALT = \"halt\";\n\n\t\tthis.shellCtrl = shellCtrl;\n\t}\n\n\tget smartFox()\n\t{\n\t\treturn this._sf;\n\t}\n\n\tconnect(config, username, password)\n\t{\n\t\t// Set additional configuration options\n\t\tconfig.zone = this.ADMIN_ZONE_NAME;\n\t\tconfig.debug = false;\n\n\t\t// Create SmartFox instance\n\t\tthis._sf = new SFS2X.SmartFox(config);\n\n\t\t// Add listeners to SmartFox events useful to the shell\n\t\tthis._addSFSEventListeners();\n\n\t\t// Save reference to connection details\n\t\tthis._config = config;\n\t\tthis._username = username;\n\t\tthis._password = password;\n\n\t\t// Connect to SmartFoxServer instance\n\t\tthis._sf.connect();\n\t}\n\n\tdisconnect()\n\t{\n\t\tthis._sf.disconnect();\n\t}\n\n\trestartServer()\n\t{\n\t\tif (this._sf.isConnected)\n\t\t\tthis._sf.send(new SFS2X.ExtensionRequest(this.COMMANDS_PREFIX + \".\" + this.COMMAND_RESTART));\n\t}\n\n\thaltServer()\n\t{\n\t\tif (this._sf.isConnected)\n\t\t\tthis._sf.send(new SFS2X.ExtensionRequest(this.COMMANDS_PREFIX + \".\" + this.COMMAND_HALT));\n\t}\n\n\t/* --------- PRIVATE UTILITY METHODS --------- */\n\n\t_addSFSEventListeners()\n\t{\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.CONNECTION, this._onConnection, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.CONNECTION_LOST, this._onConnectionLost, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.LOGIN, this._onLogin, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.LOGIN_ERROR, this._onLoginError, this);\n\t\tthis._sf.addEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse, this);\n\t}\n\n\t_reset()\n\t{\n\t\t// Remove SFS event listeners\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.CONNECTION, this._onConnection);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.CONNECTION_LOST, this._onConnectionLost);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.LOGIN, this._onLogin);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.LOGIN_ERROR, this._onLoginError);\n\t\tthis._sf.removeEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse);\n\n\t\t// Delete SmartFox class instance\n\t\tthis._sf = null;\n\n\t\t// Delete connection details\n\t\tthis._config = null;\n\t\tthis._username = null;\n\t\tthis._password = null;\n\t}\n\n\t_login()\n\t{\n\t\t// The current AdminTool version must be sent to the server during login, to check if it is up-to-date\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putInt('clientVer', this.shellCtrl.intVersion);\n\n\t\t// Login\n\t\tthis._sf.send( new SFS2X.LoginRequest(this._username, this._password, params, this._config.zone) );\n\t}\n\n\t/* --------- SMARTFOX EVENT LISTENERS --------- */\n\n\t_onConnection(evtParams)\n\t{\n\t\tif (evtParams.success)\n\t\t{\n\t\t\t// Dispatch connection event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.CONNECTION);\n\n\t\t\t// Send login request\n\t\t\tthis._login();\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Dispatch error event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.ERROR, {message: `Unable to connect to ${this._config.host}:${this._config.port}`});\n\n\t\t\t// Reset status\n\t\t\tthis._reset();\n\t\t}\n\t}\n\n\t_onConnectionLost(evtParams)\n\t{\n\t\tlet reason = evtParams.reason;\n\n\t\tif (reason != SFS2X.ClientDisconnectionReason.MANUAL)\n\t\t{\n\t\t\tvar msg;\n\n\t\t\t// Log disconnection message stating the reason\n\t\t\tif (reason == SFS2X.ClientDisconnectionReason.IDLE)\n\t\t\t\tmsg = 'inactivity';\n\t\t\telse\n\t\t\t{\n\t\t\t\tmsg = 'unknown reason';\n\n\t\t\t\tif (reason != SFS2X.ClientDisconnectionReason.UNKNOWN)\n\t\t\t\t\tmsg += ` (server reported: ${reason})`;\n\t\t\t}\n\n\t\t\t// Dispatch error event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.ERROR, {message: `A disconnection occurred due to ${msg}; please reconnect`});\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Dispatch disconnection event\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.DISCONNECTION);\n\t\t}\n\n\t\t// Reset\n\t\tthis._reset();\n\t}\n\n\t_onLogin(evtParams)\n\t{\n\t\tlet data = evtParams.data;\n\n\t\tlet params = {\n\t\t\tserverVersion: data.getUtfString('serverVer'),\n\t\t\tserverName: data.getUtfString('serverName'),\n\t\t\tserverUptime: data.getIntArray('uptime'),\n\t\t\tprocControlEnabled: data.getBool('procCtrl'),\n\t\t\tallowHalt: data.getBool('allowHalt'),\n\t\t\tmodules: data.getSFSArray('modules')\n\t\t};\n\n\t\t// Dispatch login event\n\t\tthis.dispatchEvent(ConnectionManagerEvent.LOGIN, params);\n\t}\n\n\t_onLoginError(evtParams)\n\t{\n\t\t// Disconnect from server\n\t\tthis._sf.disconnect();\n\n\t\t// Dispatch error event\n\t\tthis.dispatchEvent(ConnectionManagerEvent.ERROR, {message: evtParams.errorMessage});\n\t}\n\n\t_onExtensionResponse(evtParams)\n\t{\n\t\tif (evtParams.cmd == this.EXTENSION_ERROR)\n\t\t{\n\t\t\tlet data = evtParams.params;\n\n\t\t\t// An unexpected error occurred in the Admin Tool server-side extension\n\t\t\tthis.dispatchEvent(ConnectionManagerEvent.SERVER_ERROR, {message: 'An unexpected error occurred in the Admin Tool server-side extension, please check the server-side logs; the extension reported: ' + data.getUtfString('error')});\n\t\t}\n\t}\n}\n","import {EventDispatcher} from '../utils/event-dispatcher';\nimport {ModuleManagerEvent} from '../utils/events';\nimport {toKebabCase} from '../utils/utilities';\nimport {BaseModule} from '../modules/base-module';\n\nexport class ModuleManager extends EventDispatcher\n{\n\tconstructor(shellCtrl, container)\n\t{\n\t\tsuper();\n\n\t\t// Make BaseModule class globally available, so it can be extended by custom modules\n\t\twindow.BaseModule = BaseModule\n\n\t\tthis._shellCtrl = shellCtrl;\n\t\tthis._container = container;\n\t\tthis._currentModuleId = null;\n\n\t\t// Add listener for navigation items click\n\t\tthis._container.on('click', '.nav-item', $.proxy(function(event) {\n\t\t\tevent.preventDefault();\n\n\t\t\tlet moduleId = event.currentTarget.dataset.id;\n\t\t\tlet moduleData = this._moduleConfigById[moduleId];\n\n\t\t\tif (moduleId != this._currentModuleId)\n\t\t\t{\n\t\t\t\t// Load selected module\n\t\t\t\tthis._loadModule(moduleData);\n\t\t\t}\n\n\t\t\t// Close navigation\n\t\t\tthis._shellCtrl._toggleNav(false);\n\t\t}, this));\n\n\t\t// Add listener to show tooltip\n\t\tthis._container.kendoTooltip({\n\t\t\tfilter: 'li[data-id]',\n\t\t\tcontent: $.proxy(function(event) {\n\t\t\t\tlet moduleId = event.target[0].dataset.id;\n\t\t\t\tlet moduleData = this._moduleConfigById[moduleId];\n\t\t        return `<strong>${moduleData.name}</strong><br><span>${moduleData.description}</span>`;\n\t\t    }, this)\n\t\t});\n\n\t\t/* TESTING MEMORY LEAKS IN MODULES LOADING\n\t\tADD THIS IN THE MAIN VIEW TO START THE TEST: <div><button id=\"temp\">test</button><span id=\"cnt\"></span></div>\n\t\t$('#temp').click(\n\t\t\t$.proxy(function(event)\n\t\t\t{\n\t\t\t\tthis.cnt = 0;\n\t\t\t\tif (this.timer == null)\n\t\t\t\t{\n\t\t\t\t\tthis.timer = window.setInterval($.proxy(function() {\n\n\t\t\t\t\t\t$('#cnt').text(this.cnt++)\n\t\t\t\t\t\tif (this._currentModuleId == 'Dashboard')\n\t\t\t\t\t\t\tthis._loadModule(this._moduleConfigById['ServerConfigurator']);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis._loadModule(this._moduleConfigById['Dashboard']);\n\t\t\t\t\t}, this), 500);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\twindow.clearInterval(this.timer);\n\t\t\t\t\tthis.timer = null;\n\t\t\t\t}\n\t\t\t}, this)\n\t\t);\n\t\t*/\n\t}\n\n\tinitModulesList(modulesData, loadModuleId = null)\n\t{\n\t\t// SETUP MODULES LIST\n\t\t\n\t\t// Empty container of module selection buttons\n\t\tthis._container.empty();\n\n\t\tthis._moduleConfigById = {};\n\t\tvar firstModule = null;\n\t\tvar loadModule = null;\n\n\t\tfor (let i = 0; i < modulesData.size(); i++)\n\t\t{\n\t\t\tlet moduleData = this._getModuleObject(modulesData.getSFSObject(i));\n\n\t\t\tthis._moduleConfigById[moduleData.id] = moduleData;\n\n\t\t\t// Get first module\n\t\t\tif (i == 0)\n\t\t\t\tfirstModule = moduleData;\n\n\t\t\t// Get passed module\n\t\t\tif (moduleData.id == loadModuleId)\n\t\t\t\tloadModule = moduleData;\n\n\t\t\t// Display module button\n\t\t\tlet moduleButton = this._createModuleButton(moduleData);\n\t\t\tlet element = this._container.append(moduleButton);\n\t\t}\n\n\t\t// LOAD INITIAL MODULE\n\n\t\t// If module is not passed, load the first one of the modules list\n\t\tif (loadModule == null)\n\t\t\tloadModule = firstModule;\n\n\t\tthis._loadModule(loadModule);\n\t}\n\n\tunloadModule()\n\t{\n\t\tthis._destroyCurrentModule();\n\t}\n\n\tget currentModuleId()\n\t{\n\t\treturn this._currentModuleId;\n\t}\n\n\tget currentModule()\n\t{\n\t\treturn document.querySelector('.module');\n\t}\n\n\t_getModuleObject(sfsObj)\n\t{\n\t\treturn {\n\t\t\tid: sfsObj.getUtfString('id'),\n\t\t\tname: sfsObj.getUtfString('name'),\n\t\t\tdescription: sfsObj.getUtfString('description'),\n\t\t\tisSub: sfsObj.getBool('isSub'),\n\t\t\ticon: sfsObj.containsKey('iconSvg') ? sfsObj.getUtfString('iconSvg') : '',\n\t\t\ttag: toKebabCase(sfsObj.getUtfString('id')),\n\t\t\tisCustom: sfsObj.getBool('isCustom')\n\t\t}\n\t}\n\n\t_createModuleButton(moduleData)\n\t{\n\t\treturn `\n\t\t\t<li class=\"nav-item\" data-id=\"${moduleData.id}\">\n\t\t\t\t<div class=\"module-icon\">${moduleData.icon}</div>\n\t\t\t\t<label>${moduleData.name}</label>\n\t\t\t</li>\n\t\t`;\n\t}\n\n\t_loadModule(moduleData)\n\t{\n\t\t// Load the HTML file of a module\n\t\t$('<module/>').load(`modules/${moduleData.tag}.html`, $.proxy(function(html, status) {\n\n\t\t\tif (status != 'error')\n\t\t\t{\n\t\t\t\tif (!moduleData.isCustom)\n\t\t\t\t{\n\t\t\t\t\t// Load the JS file of a standard module\n\t\t\t\t\timport(/* webpackChunkName: \"module-\" */ `../modules/${moduleData.tag}.js`).then(module => {\n\t\t\t\t\t\tthis._onModuleControllerLoadSuccess(moduleData, html, module);\n\t\t\t\t\t})\n\t\t\t\t\t.catch(error => {\n\t\t\t\t\t\tthis._onModuleControllerLoadError(moduleData, error);\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// Load the JS file of a custom module\n\t\t\t\t\timport(/* webpackIgnore: true */`../custom-modules/${moduleData.tag}.js`).then(module => {\n\t\t\t\t\t\tthis._onModuleControllerLoadSuccess(moduleData, html, module);\n\t\t\t\t\t})\n\t\t\t\t\t.catch(error => {\n\t\t\t\t\t\tthis._onModuleControllerLoadError(moduleData, error);\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Dispatch error event\n\t\t\t\tthis.dispatchEvent(ModuleManagerEvent.MODULE_LOAD_ERROR, {message: `${moduleData.name} module's view (html) not found.`});\n\t\t\t}\n\t\t}, this));\n\t}\n\n\t_onModuleControllerLoadSuccess(moduleData, html, module)\n\t{\n\t\t// Destroy current module\n\t\tthis._destroyCurrentModule();\n\n\t\t// Define loaded module (if necessary)\n\t\tif (!window.customElements.get(moduleData.tag + '-module'))\n\t\t\twindow.customElements.define(moduleData.tag + '-module', module.default);\n\n\t\t// Append new module\n\t\t$('div.module-loader').append(html);\n\n\t\t// Initialize module\n\t\tthis.currentModule.initialize(moduleData, this._shellCtrl);\n\n\t\t// Save current module id\n\t\tthis._currentModuleId = moduleData.id;\n\n\t\t// Dispatch event\n\t\tthis.dispatchEvent(ModuleManagerEvent.MODULE_LOADED, {moduleData: moduleData});\n\t}\n\n\t_onModuleControllerLoadError(moduleData, error)\n\t{\n\t\t// Log error details\n\t\tthis._shellCtrl.logMessage(error, 'warn');\n\n\t\t// Dispatch error event\n\t\tthis.dispatchEvent(ModuleManagerEvent.MODULE_LOAD_ERROR, {message: `${moduleData.name} custom module's controller (js) couldn't be loaded.`});\n\t}\n\n\t_destroyCurrentModule()\n\t{\n\t\t// Get a reference to current module\n\t\tconst mod = this.currentModule;\n\n\t\t// Call destroy method on module's class\n\t\tif (mod != null)\n\t\t\tmod.destroy();\n\n\t\t// Empty module container\n\t\t$('div.module-loader').empty();\n\n\t\tthis._currentModuleId = null;\n\t}\n}\n","export class BaseModule extends HTMLElement\n{\n\tconstructor(commandsPrefix)\n\t{\n\t    super();\n\n\t\tthis._commandsPrefix = commandsPrefix;\n\t}\n\n\tget shellCtrl()\n\t{\n\t\treturn this._shellCtrl;\n\t}\n\n\tget smartFox()\n\t{\n\t\treturn this._shellCtrl.smartFox;\n\t}\n\n\tget idData()\n\t{\n\t\treturn this._idData;\n\t}\n\n\t//---------------------------------\n\t// OVERRIDABLE METHODS\n\t//---------------------------------\n\n\t/**\n\t * Called by the modules manager after loading the module.\n\t * In case it is overridden, super must always be called!\n\t */\n\tinitialize(idData, shellController)\n\t{\n\t\tthis._idData = idData;\n\t\tthis._shellCtrl = shellController;\n\n\t\t// Add listener to Admin extension messages\n\t\tthis.smartFox.addEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse, this);\n\t}\n\n\t/**\n\t * Called by the modules manager before unloading the module.\n\t * In case it is overridden, super must always be called!\n\t */\n\tdestroy()\n\t{\n\t\t// Remove listener to Admin extension messages\n\t\tthis.smartFox.removeEventListener(SFS2X.SFSEvent.EXTENSION_RESPONSE, this._onExtensionResponse);\n\n\t\t// Destroy all Kendo widgets\n\t\tkendo.destroy($('.module'));\n\t}\n\n\t/**\n\t * Called by the onExtensionResponse listener below.\n\t * Must be overridden.\n\t */\n\tonExtensionCommand(cmd, data)\n\t{\n\t\t// Nothing to do\n\t}\n\n\t/**\n\t * Called by the main shell whenever the server uptime changes.\n\t * Can be overridden to display the uptime inside a module or make calculations on the server uptime.\n\t */\n\tonUptimeUpdated(values)\n\t{\n\t\t// Nothing to do\n\t}\n\n\t//---------------------------------\n\t// PUBLIC METHODS\n\t//---------------------------------\n\n\t/**\n\t * Send a request to Admin extension.\n\t */\n\tsendExtensionRequest(command, data = null)\n\t{\n\t\tif (data == null)\n\t\t\tdata = new SFS2X.SFSObject();\n\n\t\tthis.smartFox.send(new SFS2X.ExtensionRequest(`${this._commandsPrefix}.${command}`, data));\n\t}\n\n\t//---------------------------------\n\t// PRIVATE METHODS\n\t//---------------------------------\n\n\t_onExtensionResponse(evtParams)\n\t{\n\t\t// Filter server responses\n\t\tlet commands = evtParams.cmd.split('.');\n\t\tlet data = evtParams.params;\n\n\t\tif (commands[0] == this._commandsPrefix)\n\t\t{\n\t\t\tif (commands.length > 1)\n\t\t\t\tthis.onExtensionCommand(commands[1], data)\n\t\t}\n\t}\n}\n","import {ViewStack} from './components/view-stack';\nimport {ModuleManager} from './managers/module-manager';\nimport {ModuleManagerEvent} from './utils/events';\nimport {ConnectionManager} from './managers/connection-manager';\nimport {ConnectionManagerEvent} from './utils/events';\nimport {ChatManager} from './managers/chat-manager';\nimport {parse} from './utils/dot-properties-parse';\n\nexport class ShellController\n{\n\tconstructor()\n\t{\n\t\t// Set constants and variables\n\n\t\tthis.VERSION_MAJOR = 3;\n\t\tthis.VERSION_MINOR = 2;\n\t\tthis.VERSION_SUB = 4;\n\n\t\tthis.DEFAULT_WS_PORT = 8080;\n\t\tthis.DEFAULT_WSS_PORT = 8443;\n\n\t\tthis.DEFAULT_USERNAME = 'sfsadmin';\n\t\tthis.DEFAULT_PASSWORD = 'sfsadmin';\n\n\t\tthis._loginValidator = null;\n\n\t\t// Display version\n\t\t$('.admin-version').text('v' + this.stringVersion);\n\n\t\t// Create modules manager instance and add event listeners\n\t\tthis._modManager = new ModuleManager(this, $('.nav-main'));\n\t\tthis._modManager.addEventListener(ModuleManagerEvent.MODULE_LOADED, this._onModuleLoaded, this);\n\t\tthis._modManager.addEventListener(ModuleManagerEvent.MODULE_LOAD_ERROR, this._onModuleLoadError, this);\n\n\t\t// Create connection manager instance and add event listeners\n\t\tthis._connManager = new ConnectionManager(this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.CONNECTION, this._onConnection, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.LOGIN, this._onLogin, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.DISCONNECTION, this._onDisconnection, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.ERROR, this._onConnManagerError, this);\n\t\tthis._connManager.addEventListener(ConnectionManagerEvent.SERVER_ERROR, this._onServerError, this);\n\n\t\t// Create admin chat manager\n\t\tthis._chatManager = new ChatManager(this);\n\n\t\t/* --------------------------- */\n\n\t\t// Initialize views\n\t\tthis._initViews();\n\n\t\t// Show login view\n\t\tthis._switchShellView('login-view');\n\t}\n\n\t//---------------------------------\n\t// VIEW INITIALIZERS\n\t//---------------------------------\n\n\t/**\n\t * Initialize login view.\n\t */\n\t_initLoginView()\n\t{\n\t\t// Set default input values (password never saved)\n\t\tlet host = window.location.hostname;\n\t\tlet port = this.DEFAULT_WS_PORT;\n\t\tlet encrypt = false;\n\t\tlet user = 'sfsadmin';\n\t\tlet remember = false;\n\n\t\t// Load \"last-server\" cookie\n\t\tlet data = Cookies.getJSON('last-server')\n        if (data)\n\t\t{\n\t\t\thost = data.host;\n            port = data.port;\n\t\t\tencrypt = data.encrypt;\n\t\t\tuser = data.user;\n\t\t\tremember = true;\n        }\n\n\t\t// Retrieve host from GET parameter\n\t\tlet getHost = this._getUrlParameter('host');\n\t\tif (getHost)\n\t\t\thost = getHost;\n\n\t\t// Set input values\n\t\t$('#loginHost').val(host);\n\t\t$('#loginUsername').val(user);\n\t\t$('#loginEncrypt').prop('checked', encrypt);\n\t\t$('#rememberLogin').prop('checked', remember);\n\n\t\t// Initialize numeric input\n\t\t$('#loginPort').kendoNumericTextBox({\n\t\t\tformat: '#####',\n\t\t\tvalue: port\n\t\t});\n\n\t\t// Initialize the Kendo UI Validator on the \"form\" container\n\t\t// (NOTE: does NOT have to be an actual <form> tag)\n\t\tthis._loginValidator = $('#loginForm').kendoValidator({\n\t\t\tvalidateOnBlur: true\n\t\t}).data('kendoValidator');\n\n\t\t// Add listener to validate the form when the Connect button is clicked\n\t\t$('#loginButton').click($.proxy(this._onConnectClick, this));\n\n\t\t// Add listener to submit form on Enter key press\n\t\t$('#loginForm').keyup(function(event) {\n\t\t\tif (event.key !== 'Enter') return;\n\n\t\t\t$('#loginButton').click();\n\t\t\tevent.preventDefault();\n\t\t});\n\n\t\t// Add listener to encrypt checkbox\n\t\t$('#loginEncrypt').change($.proxy(function(event) {\n\t\t\tlet port = this.DEFAULT_WS_PORT;\n\n\t\t\tif ($('#loginEncrypt').prop('checked'))\n\t\t\t\tport = this.DEFAULT_WSS_PORT;\n\n\t\t\t$('#loginPort').data('kendoNumericTextBox').value(port);\n\t\t}, this));\n\n\t\t// Hide error message container\n\t\t$('#login-error').hide();\n\t}\n\n\t/**\n\t * Initialize module view.\n\t */\n\t_initModuleView()\n\t{\n\t\t// Add listeners to open/close menu buttons\n\t\t$('.nav-open').click($.proxy(function(event) {\n\t\t\tevent.preventDefault();\n\t\t\tthis._toggleNav(true);\n\t\t}, this));\n\n\t\t$('.nav-close, .nav-overlay').click($.proxy(function(event) {\n\t\t\tevent.preventDefault();\n\t\t\tthis._toggleNav(false);\n\t\t}, this));\n\n\t\t// Add listeners to service buttons\n\t\t$('#restart-button').click($.proxy(this._onRestartClick, this));\n\t\t$('#halt-button').click($.proxy(this._onHaltClick, this));\n\t\t$('#help-button').click($.proxy(this._onHelpClick, this));\n\t\t$('#disconnect-button').click($.proxy(this._onDisconnectClick, this));\n\n\t\t// Add listener to show tooltip on service buttons hover\n\t\t$('.nav-service').kendoTooltip({\n\t\t\tfilter: 'li[title]'\n\t\t});\n\n\t\t// Add listener to scroll main view to top if status bar is clicked\n\t\t$('#status-bar').click(function(event) {\n\t\t\tevent.preventDefault();\n\t\t\t$('main').animate({ scrollTop: 0 }, 'fast');\n\t\t});\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t/**\n\t * Validate login form and connect+login to SmartFoxServer.\n\t */\n\t_onConnectClick(event)\n\t{\n\t\t// Hide any previous error message\n\t\t$('#login-error').hide();\n\t\t$('#login-error').text('');\n\n\t\t// Validate login form\n\t\tif (this._loginValidator.validate())\n\t\t{\n\t\t\t// Disable login form\n\t\t\tthis._enableLoginForm(false);\n\n\t\t\t// Retrieve connection details\n\t\t\tlet config = {};\n\t\t\tconfig.host = $('#loginHost').val().trim();\n\t\t\tconfig.port = $('#loginPort').data('kendoNumericTextBox').value();\n\t\t\tconfig.useSSL = $('#loginEncrypt').prop('checked');\n\n\t\t\tlet username = $('#loginUsername').val().trim();\n\t\t\tlet password = $('#loginPassword').val();\n\n\t\t\t// Save input values to cookie (except password)...\n\t\t\t// ...or clear previously saved cookie\n\t\t\tif ($('#rememberLogin').prop('checked'))\n\t\t\t{\n\t\t\t\tCookies.set('last-server', {\n\t\t\t\t\thost: config.host,\n\t\t\t\t\tport: config.port,\n\t\t\t\t\tencrypt: config.useSSL,\n\t\t\t\t\tuser: username\n\n\t\t\t\t}, {expires: 30});\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\tCookies.remove('last-server');\n\t\t\t}\n\n\t\t\t// Connect to SFS2X & login\n\t\t\tthis._connManager.connect(config, username, password);\n\t\t}\n\t}\n\n\t/**\n\t * Restart SmartFoxServer.\n\t */\n\t_onRestartClick(event)\n\t{\n\t\tlet message = 'Are you sure you want to stop and restart this instance of SmartFoxServer 2X?';\n\t\tthis.showConfirmWarning(message, $.proxy(this._onRestartConfirmDialogConfirm, this));\n\t}\n\n\t/**\n\t * Halt SmartFoxServer.\n\t */\n\t_onHaltClick(event)\n\t{\n\t\tlet message = 'Are you sure you want to stop this instance of SmartFoxServer 2X?<br>You won\\'t be able to restart it using the Administration Tool.';\n\t\tthis.showConfirmWarning(message, $.proxy(this._onHaltConfirmDialogConfirm, this));\n\t}\n\n\t/**\n\t * Open online documentation.\n\t */\n\t_onHelpClick(event)\n\t{\n\t\t// Open online doc\n\t\twindow.open(`http://docs2x.smartfoxserver.com/GettingStarted/admintool-${this._modManager.currentModuleId}`, '_blank');\n\t}\n\n\t/**\n\t * Disconnect from server.\n\t */\n\t_onDisconnectClick(event)\n\t{\n\t\tthis._connManager.disconnect();\n\t}\n\n\t//------------------------------------\n\t// CONNECTION MANAGER EVENT LISTENERS\n\t//------------------------------------\n\n\t_onConnection(evtParams)\n\t{\n\t\t// Log message\n\t\tthis.logMessage(`Connection to ${this._connManager.smartFox.config.host}:${this._connManager.smartFox.config.port} established`);\n\t}\n\n\t_onLogin(evtParams)\n\t{\n\t\t// Log message\n\t\tthis.logMessage(`Successful login to ${this._connManager.smartFox.config.host}:${this._connManager.smartFox.config.port} performed`);\n\n\t\t// Get last loaded module from cookies\n\t\tlet loadModuleId = null;\n\t\tlet data = Cookies.getJSON('last-module')\n        if (data)\n\t\t\tloadModuleId = data.id;\n\n\t\t// Init the modules list with the configuration returned by the server and: load last saved module, or first module in the list, or passed module id\n\t\tthis._modManager.initModulesList(evtParams.modules, loadModuleId);\n\n\t\t// Save current uptime\n\t\tthis._uptime = evtParams.serverUptime;\n\n\t\t// Show/hide Halt and Restart buttons depending if:\n\t\t// 1) this feature is supported for the server operating system\n\t\t// 2) the administrator who just logged in has the permission to execute these actions\n\t\tthis._showHaltRestartButtons(evtParams.procControlEnabled && evtParams.allowHalt);\n\n\t\t// Switch to modules view\n\t\tthis._goToModulesView(evtParams.serverVersion, evtParams.serverName);\n\n\t\t// If default username and password have been used...\n\t\tif ($('#loginUsername').val() == this.DEFAULT_USERNAME && $('#loginPassword').val() == this.DEFAULT_PASSWORD)\n\t\t{\n\t\t\t// ...show alert\n\t\t\tthis.showSimpleAlert('You are using the default administration profile which is highly insecure, please make sure to immediately change the password.');\n\n\t\t\t// ...show non-removable message in alert bar\n\t\t\t$('#alert-bar').show();\n\t\t\t$('#alert-bar').text('You are using the default administration profile which is highly insecure, please change the password.');\n\t\t}\n\t}\n\n\t/**\n\t * Listener called when the user disconnected voluntarily.\n\t */\n\t_onDisconnection(evtParams)\n\t{\n\t\t// Remove any popup or alert\n\t\tthis.removeDialog();\n\n\t\t// Switch back to login view\n\t\tthis._goToLoginView();\n\n\t\t// Hide navigation if open\n\t\tthis._toggleNav(false);\n\t}\n\n\t/**\n\t * Listener called when an error caused a disconnection.\n\t */\n\t_onConnManagerError(evtParams)\n\t{\n\t\t// Remove any popup or alert\n\t\tthis.removeDialog();\n\n\t\t// Log system message\n\t\tthis.logMessage(evtParams.message, 'warn');\n\n\t\t// Switch back to login view\n\t\tthis._goToLoginView();\n\n\t\t// Display error in login view\n\t\t$('#login-error').text(evtParams.message);\n\t\t$('#login-error').show();\n\t}\n\n\t/**\n\t * Listener called when an unexpected server-side error occurs.\n\t */\n\t_onServerError(evtParams)\n\t{\n\t\t// Show an alert\n\t\tthis.showSimpleAlert(evtParams.message);\n\t}\n\n\t//---------------------------------\n\t// OTHER EVENT LISTENERS\n\t//---------------------------------\n\n\t_onModuleLoaded(evtParams)\n\t{\n\t\tconst moduleData = evtParams.moduleData;\n\n\t\t// Save last loaded module to cookies\n\t\tCookies.set('last-module', {\n\t\t\tid: moduleData.id\n\t\t}, {expires: 30});\n\n\t\t// Display module title\n\t\t$('#module-title').show();\n\t\t$('#module-title-label').text(moduleData.name);\n\n\t\t// Tell the chat manager which module has been loaded\n\t\tthis._chatManager.setCurrentModule(moduleData.id);\n\n\t\t// Assign the .selected class to the selected navigation item\n\t\t$('.nav-main').find(`[data-id='${moduleData.id}']`).addClass('selected').siblings('.selected').removeClass('selected');\n\t}\n\n\t_onModuleLoadError(evtParams)\n\t{\n\t\t// Show an alert\n\t\tthis.showSimpleAlert(evtParams.message);\n\t}\n\n\t_onRestartConfirmDialogConfirm()\n\t{\n\t\t// Send restart server request\n\t\tthis._connManager.restartServer();\n\t}\n\n\t_onHaltConfirmDialogConfirm()\n\t{\n\t\t// Send halt server request\n\t\tthis._connManager.haltServer();\n\t}\n\n\t//---------------------------------\n\t// PUBLIC METHODS\n\t// This members are used by the sub-managers (i.e. ConnectionManager)\n\t// or the modules to communicate with this shell controller.\n\t//---------------------------------\n\n\tget smartFox()\n\t{\n\t\treturn this._connManager.smartFox;\n\t}\n\n\tget intVersion()\n\t{\n\t\tvar version = this.VERSION_MAJOR;\n\t\tversion += (this.VERSION_MINOR < 10 ? '0' : '') + this.VERSION_MINOR;\n\t\tversion += (this.VERSION_SUB < 10 ? '0' : '') + this.VERSION_SUB;\n\n\t\treturn Number(version);\n\t}\n\n\tget stringVersion()\n\t{\n\t\treturn this.VERSION_MAJOR + '.' + this.VERSION_MINOR + '.' + this.VERSION_SUB;\n\t}\n\n\tlogMessage(message, type = 'log')\n\t{\n\t\tswitch (type) {\n\t\t\tcase 'info':\n\t\t\t\tconsole.info('[ ADMINTOOL | INFO ] ' + message);\n\t\t\t\tbreak;\n\t\t\tcase 'warn':\n\t\t\t\tconsole.warn('[ ADMINTOOL | WARN ] ' + message);\n\t\t\t\tbreak;\n\t\t\tcase 'error':\n\t\t\t\tconsole.error('[ ADMINTOOL | ERROR ] ' + message);\n\t\t\t\tbreak;\n\t\t\tdefault:\n\t\t\t\tconsole.log('[ ADMINTOOL | INFO ] ' + message);\n\t\t}\n\t}\n\n\tremoveDialog()\n\t{\n\t\t// Hide any showing modal\n\t\t$('.modal').modal('hide');\n\n\t\t// Hide any showing toast\n\t\t$('.toast').toast('hide');\n\n\t\t// Remove shell's dialog\n\t\tif (this._dialog != null)\n\t\t{\n\t\t\tthis._dialog.close();\n\t\t\tthis._dialog.destroy();\n\t\t\tthis._dialog = null;\n\t\t}\n\n\t\t// Enable the following if other Kendo dialogs are used in modules\n\t\t/*\n\t\t// Remove any other dialog (inner to module)\n\t\t$('.k-dialog-content').each(function(index) {\n\t\t\t// Confirm dialog\n\t\t\tlet confirm = $(this).data('kendoConfirm');\n\t\t\tif (confirm)\n\t\t\t{\n\t\t\t\tconfirm.close();\n\t\t\t\tconfirm.destroy();\n\t\t\t}\n\t\t});\n\t\t*/\n\t}\n\n\t/**\n\t * Show simple alert.\n\t */\n\tshowSimpleAlert(text, isWarning = true)\n\t{\n\t\t// Create and show dialog\n\t\tthis._dialog = kendo.alert(text);\n\t\tthis._dialog.title(isWarning ? 'Warning' : 'Information');\n\n\t\t// Set custom class for styling\n\t\tthis._dialog.wrapper.addClass(isWarning ? 'is-warning' : 'is-info');\n\n\t\t// Log message too\n\t\t// (we encapsule the text in a span and extract the text again to remove inner html tags)\n\t\tlet html = $('<span>' + text + '</span>');\n\t\tthis.logMessage(html.text(), isWarning ? 'warn' : 'info');\n\t}\n\n\t/**\n\t * Show confirm alert.\n\t */\n\tshowConfirmWarning(text, confirmHandler)\n\t{\n\t\t// Create dialog\n\t\tthis._dialog = $('<div></div>').kendoConfirm({\n\t      title: 'Warning',\n\t      content: text,\n\t\t  actions: [\n\t          {\n\t              text: 'OK',\n\t              primary: true,\n\t              action: confirmHandler\n\t          },\n\t      ]\n\t  \t}).data('kendoConfirm');\n\n\t\t// Set custom class for styling\n\t\tthis._dialog.wrapper.addClass('is-warning');\n\n\t\t// Show dialog\n\t\tthis._dialog.open();\n\t}\n\n\tshowNotification(title, message)\n\t{\n\t\t// Display notification\n\t\tlet toast = $(`\n\t\t\t<div class=\"toast\" role=\"alert\" aria-live=\"assertive\" aria-atomic=\"true\" data-delay=\"4000\">\n\t\t\t\t<div class=\"toast-header\">\n\t\t\t\t\t<strong class=\"mr-auto\">${title}</strong>\n\t\t\t\t\t<button type=\"button\" class=\"ml-2 mb-1 close\" data-dismiss=\"toast\" aria-label=\"Close\">\n\t\t\t\t\t\t<span aria-hidden=\"true\">&times;</span>\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t\t<div class=\"toast-body\">${message}</div>\n\t\t\t</div>\n\t\t`);\n\n\t\t$('.toast-container').append(toast);\n\t\ttoast.toast('show');\n\n\t\t// Log message too\n\t\t// (we encapsule the text in a span and extract the text again to remove inner html tags)\n\t\tlet html = $('<span>' + title + ' - ' + message + '</span>');\n\t\tthis.logMessage(html.text(), 'info');\n\t}\n\n\tupdateModuleTitle(title, asSuffix = false)\n\t{\n\t\t$('#module-title-label').text( (asSuffix ? $('#module-title-label').text() : '') + title );\n\t}\n\n\t//---------------------------------\n\t// PRIVATE METHODS\n\t//---------------------------------\n\n\t_initViews()\n\t{\n\t\t// Initialize login view\n\t\tthis._initLoginView();\n\n\t\t// Initialize module view\n\t\tthis._initModuleView();\n\t}\n\n\t/**\n\t * Switch view in top view stack.\n\t * @param  {string} viewId Identifier of the viewstask child to display.\n\t */\n\t_switchShellView(viewId)\n\t{\n\t\tdocument.getElementById('shell').selectedElement = document.getElementById(viewId);\n\n\t\t// (jQuery alternative)\n\t\t// $('#shell')[0].selectedElement = $('#shell').children('#' + viewId)[0];\n\t}\n\n\t/**\n\t * Enable/disable login form.\n\t */\n\t_enableLoginForm(enable)\n\t{\n\t\t// Enable/disable fieldset (works for all non-kendo inputs)\n\t\t$('#loginForm fieldset').prop('disabled', !enable);\n\n\t\t// Enable/disable numeric textbox\n\t\t$('#loginPort').data('kendoNumericTextBox').enable(enable);\n\t}\n\n\t_toggleNav(bool)\n\t{\n\t\t$('.nav-container, .nav-overlay').toggleClass('is-visible', bool);\n\t\t$('.module-container').toggleClass('scale-down', bool);\n\t}\n\n\t_getUrlParameter(param)\n\t{\n\t\tlet pageURL = window.location.search.substring(1);\n\t\tlet urlVariables = pageURL.split('&');\n\n\t\tfor (var i = 0; i < urlVariables.length; i++) {\n\t\t\tlet paramName = urlVariables[i].split('=');\n\t\t\tif (paramName[0] == param && paramName[1] != '')\n\t\t\t\treturn paramName[1];\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t_goToLoginView()\n\t{\n\t\t// Clear password field\n\t\t$('#loginPassword').val('');\n\n\t\t// Switch to login view\n\t\tthis._switchShellView('login-view');\n\n\t\t// Unload current module\n\t\tthis._modManager.unloadModule();\n\n\t\t// Enable login form\n\t\tthis._enableLoginForm(true);\n\n\t\t// Remove uptime updater\n\t\tclearInterval(this._uptimeTimer);\n\t}\n\n\t_goToModulesView(serverVersion, serverName)\n\t{\n\t\t// Hide module title\n\t\t$('#module-title').hide();\n\t\t$('#module-title-label').text('');\n\n\t\t// Hide alert bar\n\t\t$('#alert-bar').hide();\n\n\t\t// Show server version in the header\n\t\t$('#sfs-version-value').text(serverVersion);\n\n\t\t// Set server name, IP and port in module's title bar\n\t\tlet host = `${this._connManager.smartFox.config.host}:${this._connManager.smartFox.config.port}` + (serverName != '' ? ` [${serverName}]` : '');\n\t\t$('#host-label').text(host);\n\n\t\t// Save current SFS version\n\t\tthis._currentSfsVersion = serverVersion;\n\n\t\t// Check new SFS version availability\n\t\tthis._checkAvailableSfsVersion();\n\n\t\t// Start uptime updater\n\t\tthis._uptimeTimer = setInterval($.proxy(this._updateUptime, this), 1000);\n\n\t\t// Switch to modules view\n\t\tthis._switchShellView('module-view');\n\t}\n\n\t_showHaltRestartButtons(show)\n\t{\n\t\tif (show)\n\t\t{\n\t\t\t$('#restart-button').show();\n\t\t\t$('#halt-button').show();\n\t\t}\n\t\telse\n\t\t{\n\t\t\t$('#restart-button').hide();\n\t\t\t$('#halt-button').hide();\n\t\t}\n\t}\n\n\t_checkAvailableSfsVersion()\n\t{\n\t\t// Remove class to hide update button style and notification icon\n\t\t$('#sfs-version').removeClass('is-active');\n\n\t\t// Remove update button click listeners\n\t\t$('#sfs-version-button').off('click');\n\n\t\t// Load file containing latest SFS version info\n\t\t$.ajax({\n\t\t\ttype: 'GET',\n\t\t\turl: 'https://www.smartfoxserver.com/downloads/sfs2x/latestVersion.txt',\n        \tdataType: 'text',\n\t\t\tsuccess: $.proxy(this._onLatestSfsVersionInfoLoaded, this),\n\t\t\terror: $.proxy(function() {\n\t\t\t\tthis.logMessage('Unable to check new server version availability on SmartFoxServer website', 'warn');\n\t\t\t}, this)\n\t\t});\n\t}\n\n\t_onLatestSfsVersionInfoLoaded(data)\n\t{\n\t\t// Parse returned data\n\t\tconst v = parse(data);\n\n\t\tif (v.version != null && this._isSfsVersionNewer(v.version, this._currentSfsVersion))\n\t\t{\n\t\t\tthis.tempflag = true;\n\t\t\tthis.logMessage('An updated version of SmartFoxServer 2X is available for download', 'info');\n\n\t\t\t// Set upgrade dialog details\n\t\t\tthis._sfsUpdateDetails = v;\n\n\t\t\t// Add listener to show SFS version update modal\n\t\t\t$('#sfs-version-button').click(function() {\n\t\t\t\t$('#serverUpdateModal').modal({\n\t\t\t\t\tbackdrop: 'static',\n\t\t\t\t\tkeyboard: false,\n\t\t\t\t});\n\t\t\t});\n\n\t\t\t// Set class to show update button style and notification icon\n\t\t\t$('#sfs-version').addClass('is-active');\n\n\t\t\t// Update modal content\n\t\t\tconst newVer = this._sfsUpdateDetails;\n\t\t\tconst currVer = this._currentSfsVersion;\n\n\t\t\tlet modal = $('#serverUpdateModal');\n\n\t\t\t// Update modal content\n\t\t\t$('#newVersLb', modal).text(newVer.version);\n\t\t\t$('#updTypeLb', modal).text(newVer.isPatch ? 'patch' : 'full installer');\n\t\t\t$('#reqVersLb', modal).text(newVer.isPatch ? ' (requires version ' + newVer.requires + ' or later)' : '');\n\t\t\t$('#currVersLb', modal).text(currVer);\n\t\t\t$('#serverUpdateModalLink', modal).attr('href', newVer.url);\n\t\t}\n\t}\n\n\t_isSfsVersionNewer(availableVer, currentVer)\n\t{\n\t\tconst MAJ = 0;\n\t\tconst MIN = 1;\n\t\tconst SUB = 2;\n\n\t\tconst available = availableVer.split('.');\n\t\tconst current = currentVer.split('.');\n\n\t\t// Check major version\n\t\tif (available[MAJ] > current[MAJ])\n\t\t\treturn true;\n\t\telse if (available[MAJ] == current[MAJ])\n\t\t{\n\t\t\t// Check minor version\n\t\t\tif (available[MIN] > current[MIN])\n\t\t\t\treturn true;\n\t\t\telse if (available[MIN] == current[MIN])\n\t\t\t{\n\t\t\t\t// Check sub version\n\t\t\t\tif (available[SUB] > current[SUB])\n\t\t\t\t\treturn true;\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t_updateUptime()\n\t{\n\t\tlet days = this._uptime[0];\n\t\tlet hours = this._uptime[1];\n\t\tlet minutes = this._uptime[2];\n\t\tlet seconds = this._uptime[3] + 1;\n\n\t\tif (seconds > 59)\n\t\t{\n\t\t\tseconds = 0;\n\t\t\tminutes += 1;\n\n\t\t\tif (minutes > 59)\n\t\t\t{\n\t\t\t\tminutes = 0;\n\t\t\t\thours += 1;\n\n\t\t\t\tif (hours > 23)\n\t\t\t\t{\n\t\t\t\t\thours = 0;\n\t\t\t\t\tdays += 1;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tthis._uptime[3] = seconds;\n\t\tthis._uptime[2] = minutes;\n\t\tthis._uptime[1] = hours;\n\t\tthis._uptime[0] = days;\n\n\t\t// Send updated uptime to current module (if loaded)\n\t\tlet module = this._modManager.currentModule;\n\t\tif (module != null)\n\t\t\tmodule.onUptimeUpdated(this._uptime);\n\t}\n}\n","const atComment = (src, offset) => {\n  const ch = src[offset]\n  return ch === '#' || ch === '!'\n}\n\nconst atLineEnd = (src, offset) => {\n  const ch = src[offset]\n  return !ch || ch === '\\r' || ch === '\\n'\n}\n\nconst endOfIndent = (src, offset) => {\n  let ch = src[offset]\n  while (ch === '\\t' || ch === '\\f' || ch === ' ') {\n    offset += 1\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfComment = (src, offset) => {\n  let ch = src[offset]\n  while (ch && ch !== '\\r' && ch !== '\\n') {\n    offset += 1\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfKey = (src, offset) => {\n  let ch = src[offset]\n  while (ch && ch !== '\\r' && ch !== '\\n' && ch !== '\\t' && ch !== '\\f' && ch !== ' ' && ch !== ':' && ch !== '=') {\n    if (ch === '\\\\') {\n      if (src[offset + 1] === '\\n') {\n        offset = endOfIndent(src, offset + 2)\n      } else {\n        offset += 2\n      }\n    } else {\n      offset += 1\n    }\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfSeparator = (src, offset) => {\n  let ch = src[offset]\n  let hasEqSign = false\n  loop: while (ch === '\\t' || ch === '\\f' || ch === ' ' || ch === '=' || ch === ':' || ch === '\\\\') {\n    switch (ch) {\n      case '\\\\':\n        if (src[offset + 1] !== '\\n') break loop\n        offset = endOfIndent(src, offset + 2)\n        break\n      case '=':\n      case ':':\n        if (hasEqSign) break loop\n        hasEqSign = true\n        // fallthrough\n      default:\n        offset += 1\n    }\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst endOfValue = (src, offset) => {\n  let ch = src[offset]\n  while (ch && ch !== '\\r' && ch !== '\\n') {\n    offset += ch === '\\\\' ? 2 : 1\n    ch = src[offset]\n  }\n  return offset\n}\n\nconst unescape = (str) => str.replace(/\\\\(u[0-9a-fA-F]{4}|\\r?\\n[ \\t\\f]*|.)?/g, (match, code) => {\n  switch (code && code[0]) {\n    case 'f': return '\\f'\n    case 'n': return '\\n'\n    case 'r': return '\\r'\n    case 't': return '\\t'\n    case 'u':\n      const c = parseInt(code.substr(1), 16)\n      return isNaN(c) ? code : String.fromCharCode(c)\n    case '\\r':\n    case '\\n':\n    case undefined:\n      return ''\n    default:\n      return code\n  }\n})\n\n/**\n * Splits the input string into an array of logical lines\n *\n * Key-value pairs are [key, value] arrays with string values. Escape sequences\n * in keys and values are parsed. Empty lines are included as empty strings, and\n * comments as strings that start with '#' or '! characters. Leading whitespace\n * is not included.\n *\n * @see https://docs.oracle.com/javase/9/docs/api/java/util/Properties.html#load(java.io.Reader)\n *\n * @param {string} src\n * @returns Array<string | string[]]>\n */\nfunction parseLines (src) {\n  const lines = []\n  for (i = 0; i < src.length; ++i) {\n    if (src[i] === '\\n' && src[i - 1] === '\\r') i += 1\n    if (!src[i]) break\n    const keyStart = endOfIndent(src, i)\n    if (atLineEnd(src, keyStart)) {\n      lines.push('')\n      i = keyStart\n      continue\n    }\n    if (atComment(src, keyStart)) {\n      const commentEnd = endOfComment(src, keyStart)\n      lines.push(src.slice(keyStart, commentEnd))\n      i = commentEnd\n      continue\n    }\n    const keyEnd = endOfKey(src, keyStart)\n    const key = unescape(src.slice(keyStart, keyEnd))\n    const valueStart = endOfSeparator(src, keyEnd)\n    if (atLineEnd(src, valueStart)) {\n      lines.push([key, ''])\n      i = valueStart\n      continue\n    }\n    const valueEnd = endOfValue(src, valueStart)\n    const value = unescape(src.slice(valueStart, valueEnd))\n    lines.push([key, value])\n    i = valueEnd\n  }\n  return lines\n}\n\n/**\n * Parses an input string read from a .properties file into a JavaScript Object\n *\n * If the second `path` parameter is true, dots '.' in keys will result in a\n * multi-level object (use a string value to customise). If a parent level is\n * directly assigned a value while it also has a child with an assigned value,\n * the parent value will be assigned to its empty string '' key. Repeated keys\n * will take the last assigned value. Key order is not guaranteed, but is likely\n * to match the order of the input lines.\n *\n * @param {string} src\n * @param {boolean | string} [path=false]\n */\nfunction parse (src, path) {\n  const pathSep = typeof path === 'string' ? path : '.'\n  return parseLines(src).reduce((res, line) => {\n    if (Array.isArray(line)) {\n      const [key, value] = line\n      if (path) {\n        const keyPath = key.split(pathSep)\n        let parent = res\n        while (keyPath.length >= 2) {\n          const p = keyPath.shift()\n          if (!parent[p]) {\n            parent[p] = {}\n          } else if (typeof parent[p] !== 'object') {\n            parent[p] = { '': parent[p] }\n          }\n          parent = parent[p]\n        }\n        const leaf = keyPath[0]\n        if (typeof parent[leaf] === 'object') {\n          parent[leaf][''] = value\n        } else {\n          parent[leaf] = value\n        }\n      } else {\n        res[key] = value\n      }\n    }\n    return res\n  }, {})\n}\n\nmodule.exports = { parse, parseLines }\n","export class EventDispatcher {\n\tconstructor() {\n\t\tthis._listenersByEvent = {};\n\t}\n\n\t/**\n\t * Registers an event listener function that will receive notification of an event.\n\t *\n\t * <p>If you no longer need an event listener, remove it by calling the <em>removeEventListener()</em> method, or memory issues could arise.\n\t * In fact event listeners are not automatically removed from memory.</p>\n\t *\n\t * @param\t{string} evtType\tThe type of event to listen to.\n\t * @param\t{function} callback\tThe listener function that processes the event. This function should accept an object as its only parameter, which in turn contains the event parameters.\n\t * @param\t{object} scope\t\tThe object that acts as a context for the event listener: it is the object that acts as a \"parent scope\" for the callback function, thus providing context (i.e. access to variables and other mehtods) to the function itself.\n\t */\n\taddEventListener(evtType, callback, scope)\n\t{\n\t\tif (!this._listenersByEvent[evtType])\n\t\t\tthis._listenersByEvent[evtType] = [];\n\n\t\tthis._listenersByEvent[evtType].push({callback:callback, scope:scope});\n\t}\n\n\t/**\n\t * Removes an event listener.\n\t *\n\t * @param\t{string} evtType\tThe type of event to remove.\n\t * @param\t{function} callback\tThe listener function to be removed.\n\t */\n\tremoveEventListener(evtType, callback)\n\t{\n\t\tconst listeners = this._listenersByEvent[evtType];\n\n\t\tif (listeners)\n\t\t{\n\t\t\tfor (let i = 0; i < listeners.length; i++)\n\t\t\t{\n\t\t\t\tif (listeners[i].callback === callback)\n\t\t\t\t{\n\t\t\t\t\tlisteners.splice(i, 1);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tdispatchEvent(evtType, evtObj)\n\t{\n\t\tconst listeners = this._listenersByEvent[evtType];\n\n\t\tif (listeners)\n\t\t{\n\t\t\tfor (let listener of listeners)\n\t\t\t\tlistener.callback.call(listener.scope, evtObj);\n\t\t}\n\t}\n}\n","export const ConnectionManagerEvent = Object.freeze({\n\tCONNECTION: 'connection',\n\tLOGIN: 'login',\n\tDISCONNECTION: 'disconnection',\n\tERROR: 'error',\n\tSERVER_ERROR: 'serverError',\n});\n\nexport const ModuleManagerEvent = Object.freeze({\n\tMODULE_LOADED: 'module-loaded',\n\tMODULE_LOAD_ERROR: 'module-load-error',\n});\n","export const toKebabCase = (str) =>\n  str &&\n  str\n    .match(/[A-Z]{2,}(?=[A-Z][a-z]+[0-9]*|\\b)|[A-Z]?[a-z]+[0-9]*|[A-Z]|[0-9]+/g)\n    .map(x => x.toLowerCase())\n    .join('-');\n\nexport function bytesToSize(bytes, decimals = 1, zeroUnit = '', suffix = '') {\n\tif (bytes === 0)\n\t\treturn '0 ' + zeroUnit + (zeroUnit != '' ? suffix : '');\n\n\tif (bytes < 1) // Can happen in chart axis labels!\n\t \treturn `${bytes} Bytes${suffix}`;\n\n    const k = 1000;\n    const dm = decimals < 0 ? 0 : decimals;\n    const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];\n\n    const i = Math.floor(Math.log(bytes) / Math.log(k));\n\n    return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i] + suffix;\n}\n\nexport function kBytesToSize(kBytes, decimals = 1, zeroUnit = '', suffix = '') {\n\treturn bytesToSize(kBytes * 1000, decimals, zeroUnit, suffix);\n}\n\nexport function capitalizeFirst(string) {\n\treturn string.charAt(0).toUpperCase() + string.slice(1);\n}\n\nexport function filterClassName(element, index, array)\n{\n\tif (element.endsWith('.py'))\n\t\treturn (element.endsWith('Extension.py'));\n\telse if (element.endsWith('.js'))\n\t\treturn (element.endsWith('Extension.js'));\n\telse\n\t\treturn (element.endsWith('Extension'));\n}\n\nexport function roundToDecimals(value, decimals) {\n  return Number(Math.round(value + 'e' + decimals) + 'e-' + decimals);\n}\n\nexport function scaleBytes(bytes, decimals = 1) {\n\tlet obj = {};\n\n\tconst sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];\n\n\tif (bytes > 0)\n\t{\n\t\tconst k = 1000;\n\t\tconst dm = decimals < 0 ? 0 : decimals;\n\t\tconst i = Math.floor(Math.log(bytes) / Math.log(k));\n\n\t\tobj.value = parseFloat((bytes / Math.pow(k, i)).toFixed(dm));\n\t\tobj.unit = sizes[i];\n\t}\n\telse\n\t{\n\t\tobj.value = 0;\n\t\tobj.unit = sizes[0];\n\t}\n\n\treturn obj;\n}\n"],"mappings":";;;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;;;;;;;;;;;;;ACrMA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACRA;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;;;;;;;;;;;;;ACvEA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACrBA;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;;;;;;;;;;;;;AC/LA;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;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACtOA;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;;;;;;;;;;;;;;ACvGA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC7vBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACxLA;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;;;;;;;;;;;;;ACxDA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACXA;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;;;;;;;;;;;;;;;;A","sourceRoot":""}
@@ -374,9 +374,6 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
374
374
  // Create file data source manager
375
375
  this._fileManager = new _managers_file_datasource_manager__WEBPACK_IMPORTED_MODULE_1__["FileDataSourceManager"](null, [], this._fileSeparator);
376
376
 
377
- // Retrieve HTTP port to be used for files uploading
378
- const uploadHttpPort = data.getInt('httpPort');
379
-
380
377
  // Retrieve module id sent by the server (required because multiple modules use file uploading service)
381
378
  const uploadModuleId = data.getUtfString('modId');
382
379
 
@@ -384,8 +381,9 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
384
381
  this._uploadTargetConfig = {
385
382
  sessionToken: this.smartFox.sessionToken,
386
383
  host: this.smartFox.config.host,
387
- httpPort: uploadHttpPort,
384
+ port: this.smartFox.config.port,
388
385
  moduleId: uploadModuleId,
386
+ protocol: this.smartFox.config.useSSL ? 'https' : 'http'
389
387
  };
390
388
 
391
389
  // Request Servlet files data to server instance
@@ -468,7 +466,7 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
468
466
  if (requester == this.smartFox.mySelf.name)
469
467
  {
470
468
  // Expand parent
471
- this._filesList.expand($(`#svm-fileList .file-controls[data-item-id="${parentPath}"]`).closest('tr'));
469
+ this._filesList.expand($(`#svm-fileList .file-controls[data-item-id="${$.escapeSelector(parentPath)}"]`).closest('tr'));
472
470
 
473
471
  if (!isUpload)
474
472
  {
@@ -476,7 +474,7 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
476
474
  this._addFolderModal.modal('hide');
477
475
 
478
476
  // Select upload file
479
- this._filesList.select($(`#svm-fileList .file-controls[data-item-id="${filePath}"]`).closest('tr'));
477
+ this._filesList.select($(`#svm-fileList .file-controls[data-item-id="${$.escapeSelector(filePath)}"]`).closest('tr'));
480
478
  }
481
479
 
482
480
  // Update selection
@@ -526,7 +524,7 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
526
524
 
527
525
  // Collapse parent if the last of its children was deleted
528
526
  if (parentItem && !parentItem.hasChildren)
529
- this._filesList.collapse($(`#svm-fileList .file-controls[data-item-id="${parentItem.id}"]`).closest('tr'));
527
+ this._filesList.collapse($(`#svm-fileList .file-controls[data-item-id="${$.escapeSelector(parentItem.id)}"]`).closest('tr'));
530
528
  }
531
529
 
532
530
  if (requester == this.smartFox.mySelf.name)
@@ -573,7 +571,7 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
573
571
  {
574
572
  // Hide control buttons on currently selected item
575
573
  if (this._selectedItem)
576
- $(`#svm-fileList .file-controls[data-item-id="${this._selectedItem.id}"]`).hide();
574
+ $(`#svm-fileList .file-controls[data-item-id="${$.escapeSelector(this._selectedItem.id)}"]`).hide();
577
575
 
578
576
  // Get selected item
579
577
  let selectedRows = this._filesList.select();
@@ -584,7 +582,7 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
584
582
  this._selectedItem = this._filesList.dataItem(selectedRows[0]);
585
583
 
586
584
  // Show control buttons on new selected item
587
- $(`#svm-fileList .file-controls[data-item-id="${this._selectedItem.id}"]`).show();
585
+ $(`#svm-fileList .file-controls[data-item-id="${$.escapeSelector(this._selectedItem.id)}"]`).show();
588
586
  }
589
587
  else
590
588
  this._selectedItem = null;
@@ -648,7 +646,7 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
648
646
  $('#svm-clearFilesBt').attr('disabled', true);
649
647
 
650
648
  // Set destination url
651
- const url = 'http://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.httpPort + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;
649
+ const url = this._uploadTargetConfig.protocol + '://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.port + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;
652
650
 
653
651
  e.sender.options.async.saveUrl = url;
654
652
 
@@ -763,4 +761,4 @@ class ServletManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModu
763
761
  /***/ })
764
762
 
765
763
  }]);
766
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-10.bundle.js","sources":["webpack://application/./src/managers/file-datasource-manager.js","webpack://application/./src/modules/servlet-manager.js"],"sourcesContent":["export class FileDataSourceManager\n{\n\tconstructor(libFolder, protectedFolders, fileSeparator)\n\t{\n\t\tthis._protectedFolders = protectedFolders; // Folders which can't be deleted (but their content can)\n\t\tthis._libFolder = libFolder;\n\t\tthis._fileSeparator = fileSeparator;\n\t}\n\n\tget dataSource()\n\t{\n\t\treturn this._dataSource;\n\t}\n\n\tinit()\n\t{\n\t\tthis._dataSource = new kendo.data.TreeListDataSource({\n\t\t\tdata: [],\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'id',\n\t\t\t\t\tparentId: 'parentId',\n\t\t\t\t\texpanded: false\n\t\t\t\t}\n\t\t\t},\n\t\t\tsort: { field: 'name', dir: 'asc' }\n\t\t});\n\t}\n\n\taddFile(fileObj, parentLevel = null)\n\t{\n\t\tlet file = {};\n\n\t\tfile.name = fileObj.getUtfString('name');\n\t\tfile.isDir = fileObj.getBool('isDir');\n\t\tfile.lastMod = fileObj.getLong('lastMod');\n\t\tfile.isLib = (file.isDir && file.name == this._libFolder);\n\t\tfile.isProtected = (file.isDir && this._protectedFolders.indexOf(file.name) > -1);\n\t\tfile.size = 0;\n\n\t\tif (parentLevel == null)\n\t\t\tfile.level = 0;\n\t\telse\n\t\t\tfile.level = parentLevel + 1;\n\n\t\tif (fileObj.containsKey('parent'))\n\t\t{\n\t\t\tfile.parentId = fileObj.getUtfString('parent');\n\t\t\tfile.id = file.parentId + this._fileSeparator + file.name;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tfile.parentId = null;\n\t\t\tfile.id = file.name;\n\t\t}\n\n\t\t// Add child files\n\t\tif (file.isDir)\n\t\t{\n\t\t\tlet filesArr = fileObj.getSFSArray('files');\n\n\t\t\tfor (let i = 0; i < filesArr.size(); i++)\n\t\t\t\tfile.size += this.addFile(filesArr.getSFSObject(i), file.level);\n\t\t}\n\t\telse\n\t\t\tfile.size = fileObj.getLong('size');\n\n\t\t// Add file to data source\n\t\tthis._dataSource.add(file);\n\n\t\t// Return file size\n\t\treturn file.size;\n\t}\n\n\tremoveFile(id)\n\t{\n\t\tlet fileItem = this._dataSource.get(id);\n\n\t\tif (fileItem)\n\t\t{\n\t\t\tif (fileItem.parentId)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tlet parentItem = this._dataSource.get(fileItem.parentId);\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\t\t\t}\n\n\t\t\tthis._dataSource.remove(fileItem);\n\n\t\t\t// Return parent item\n\t\t\tif (fileItem.parentId)\n\t\t\t\treturn this._dataSource.get(fileItem.parentId);\n\t\t}\n\t}\n\n\tgetFileById(id)\n\t{\n\t\treturn this._dataSource.get(id);\n\t}\n\n\taddFileToParent(fileObj, parentId)\n\t{\n\t\tlet parentItem = this._dataSource.get(parentId);\n\n\t\tif (parentItem != null && parentItem.isDir)\n\t\t{\n\t\t\tconst fileId = parentId + this._fileSeparator + fileObj.getUtfString('name');\n\t\t\tlet fileItem = this._dataSource.get(fileId);\n\n\t\t\tif (fileItem != null)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\n\t\t\t\t// Update existing item\n\t\t\t\tfileItem.name = fileObj.getUtfString('name');\n\t\t\t\tfileItem.lastMod = fileObj.getLong('lastMod');\n\t\t\t\tfileItem.size = fileObj.getLong('size');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Add new item\n\t\t\t\tthis.addFile(fileObj, parentItem.level);\n\t\t\t}\n\n\t\t\t// Update parent item size\n\t\t\tthis._updateParentSize(parentItem, fileObj.getLong('size'));\n\n\t\t\treturn fileId;\n\t\t}\n\t\telse\n\t\t\tthrow new Error(`An unexpected error occurred while adding file '${fileObj.getUtfString('name')}' (target: ${parentId}).`);\n\t}\n\n\t_updateParentSize(parentItem, value)\n\t{\n\t\tparentItem.size += value;\n\n\t\tif (parentItem.parentId)\n\t\t{\n\t\t\tlet grandParent = this._dataSource.get(parentItem.parentId);\n\t\t\tthis._updateParentSize(grandParent, value);\n\t\t}\n\t}\n}\n","import {BaseModule} from './base-module';\nimport {FileDataSourceManager} from '../managers/file-datasource-manager';\nimport {bytesToSize} from '../utils/utilities';\n\nexport default class ServletManager extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('servletMan');\n\n\t\t// Outgoing requests\n\t\tthis.REQ_INIT = 'init';\n\t\tthis.REQ_GET_SERVLETS = 'getServlets';\n\t\tthis.REQ_CREATE_FOLDER = 'createFolder';\n\t\tthis.REQ_DELETE_FILES = 'deleteSrvltFiles';\n\n\t\t// Incoming responses\n\t\tthis.RESP_LOCKED = 'lock';\n\t\tthis.RESP_INIT = 'init';\n\t\tthis.RESP_SERVLETS = 'servlets';\n\t\tthis.RESP_FILE_ADDED = 'fileAdded';\n\t\tthis.RESP_FILES_DELETED = 'filesDeleted';\n\t\tthis.RESP_ERROR = 'error';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize progress bar\n\t\t$('#svm-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Add listeners to buttons\n\t\t$('#svm-retryBt').on('click', $.proxy(this._onRetryClick, this));\n\t\t$('#svm-refreshBt').on('click', $.proxy(this._onRefreshClick, this));\n\n\t\t// Initialize files list\n\t\tthis._filesList = $('#svm-fileList').kendoTreeList({\n            dataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: true,\n            columns: [\n                {\n\t\t\t\t\tfield: 'name',\n\t\t\t\t\ttitle: 'Name',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t<div >\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t# if (expanded) { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder-open\"></i>\n\t\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder\"></i>\n\t\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t<i class=\"far fa-file\"></i>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t#: name #\n\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"file-controls flex-grow-1 text-right\" data-item-id=\"#:id#\">\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 create-folder-bt\"><i class=\"fas fa-folder-plus\"></i></button>\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 upload-files-bt\"><i class=\"fas fa-file-upload\"></i></button>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t# if (level > 0 && !isProtected) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 remove-file-bt\"><i class=\"fas fa-minus-circle\"></i></button>\n\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t</div>\n\t\t\t\t\t`),\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'size',\n\t\t\t\t\ttitle: 'Size',\n\t\t\t\t\ttemplate: function(dataItem) {\n\t\t\t\t\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\t\t\t\t\treturn kendo.template(`\n\t\t\t\t\t\t\t#: bytesToSize(size, 2, 'KB') #\n\t\t\t\t\t\t`)(dataItem);\n\t\t\t\t\t},\n\t\t\t\t\twidth: 120,\n\t\t\t\t\tminScreenWidth: 576\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'lastMod',\n\t\t\t\t\ttitle: 'Last Modified',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t#: kendo.toString(new Date(lastMod), 'dd MMM yyyy HH:mm:ss') #\n\t\t\t\t\t`),\n\t\t\t\t\twidth: 200,\n\t\t\t\t\tminScreenWidth: 768\n\t\t\t\t},\n            ],\n\t\t\tchange: $.proxy(this._onFileSelectedChange, this)\n        }).data('kendoTreeList');\n\n\t\t//-------------------------------------------\n\n\t\t// Add listeners to catch control button clicks on rows\n\t\t$('#svm-fileList').on('click', '.create-folder-bt', $.proxy(this._showAddFolderModalClick, this));\n\t\t$('#svm-fileList').on('click', '.upload-files-bt', $.proxy(this._showUploadFilesModalClick, this));\n\t\t$('#svm-fileList').on('click', '.remove-file-bt', $.proxy(this._onRemoveFileClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"add folder\" modal\n\t\tthis._addFolderModal = $('#svm-addFolderModal');\n\t\tthis._addFolderModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Add listener to modal hide event\n\t\tthis._addFolderModal.on('hidden.bs.modal', $.proxy(this._onAddFolderModalHidden, this));\n\n\t\t// Add listener to Add button click\n\t\t$('#svm-addFolderBt').on('click', $.proxy(this._onAddFolderClick, this));\n\n\t\t// Initialize kendo validation on folder name form\n\t\tthis._addFolderValidator = $('#svm-addFolderForm').kendoValidator({}).data('kendoValidator');\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"upload files\" modal\n\t\tthis._uploadFilesModal = $('#svm-uploadModal');\n\t\tthis._uploadFilesModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Initialize kendo uploader\n\t\tthis._uploader = $('#svm-uploader').kendoUpload({\n\t\t\tmultiple: true,\n\t\t\tasync: {\n\t\t\t\tsaveUrl: 'http://localhost', // This will be changed later in _onUploadStart method\n\t\t\t\tautoUpload: true,\n\t\t\t},\n\t\t\tdirectoryDrop: true,\n\t\t\tupload: $.proxy(this._onUploadStart, this),\n\t\t\tcomplete: $.proxy(this._onUploadEnd, this),\n\t\t\tlocalization: {\n\t\t\t\tselect: 'Select files...'\n\t\t\t}\n\t\t}).data('kendoUpload');\n\n\t\t// Add listener to Upload button click\n\t\t$('#svm-clearFilesBt').on('click', $.proxy(this._onClearFilesClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t$('#svm-retryBt').off('click');\n\t\t$('#svm-refreshBt').off('click');\n\n\t\t$('#svm-fileList').off('click');\n\n\t\tthis._addFolderModal.off('hidden.bs.modal');\n\t\tthis._addFolderModal.modal('dispose');\n\t\t$('#svm-addFolderBt').off('click');\n\n\t\tthis._uploadFilesModal.modal('dispose');\n\t\t$('#svm-clearFilesBt').off('click');\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Module can be enabled (no locking file exists)\n\t\tif (command == this.RESP_INIT)\n\t\t{\n\t\t\t// Retrieve file separator\n\t\t\tthis._fileSeparator = data.getUtfString('sep');\n\n\t\t\t// Create file data source manager\n\t\t\tthis._fileManager = new FileDataSourceManager(null, [], this._fileSeparator);\n\n\t\t\t// Retrieve HTTP port to be used for files uploading\n\t\t\tconst uploadHttpPort = data.getInt('httpPort');\n\n\t\t\t// Retrieve module id sent by the server (required because multiple modules use file uploading service)\n\t\t\tconst uploadModuleId = data.getUtfString('modId');\n\n\t\t\t// Set file uploading target configuration\n\t\t\tthis._uploadTargetConfig = {\n\t\t\t\tsessionToken: this.smartFox.sessionToken,\n\t\t\t\thost: this.smartFox.config.host,\n\t\t\t\thttpPort: uploadHttpPort,\n\t\t\t\tmoduleId: uploadModuleId,\n\t\t\t};\n\n\t\t\t// Request Servlet files data to server instance\n\t\t\tthis._refreshDataList();\n\t\t}\n\n\t\t/*\n\t\t * This response is returned if the file UploadsLock.txt exists in the /config folder of the server.\n\t\t * This is an additional security measure to avoid unwanted files to be uploaded by malicius users accessing the server\n\t\t * with the default credentials, in case they have not been changed by the administrator after the installation.\n\t\t * The file must be removed manually before accessing the Servlet Manager module for the first time\n\t\t */\n\t\telse if (command == this.RESP_LOCKED)\n\t\t{\n\t\t\t// Show warning\n\t\t\tthis._switchView('svm-locked');\n\t\t}\n\n\t\t// Servlet folders and files\n\t\telse if (command == this.RESP_SERVLETS)\n\t\t{\n\t\t\t// Retrieve Servlets file list\n\t\t\tlet servletsObj = data.getSFSObject('servlets');\n\n\t\t\t// Initialize manager\n\t\t\tthis._fileManager.init();\n\n\t\t\t// Add list to manager\n\t\t\tthis._fileManager.addFile(servletsObj);\n\n\t\t\t// Set TreeList data source\n\t\t\tthis._filesList.setDataSource(this._fileManager.dataSource);\n\n\t\t\t// Expand first level\n\t\t\tthis._filesList.expand($('#svm-fileList tbody>tr:eq(0)'));\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show module's main view\n\t\t\tthis._switchView('svm-main');\n\t\t}\n\n\t\t// An error occurred while managing servlet files\n\t\telse if (command == this.RESP_ERROR)\n\t\t{\n\t\t\t// Hide add folder modal\n\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t// Re-enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\t// Servlet folder or file added\n\t\telse if (command == this.RESP_FILE_ADDED)\n\t\t{\n\t\t\t// Get name of the user who added the file/folder\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the object representing the file/folder being added\n\t\t\tconst fileObj = data.getSFSObject('file');\n\n\t\t\t// Get the target folder where the new file/folder should be added\n\t\t\tconst parentPath = data.getUtfString('parent');\n\n\t\t\t// Get the flag notifying this was a file upload\n\t\t\tconst isUpload = data.getBool('isUpload');\n\n\t\t\ttry\n\t\t\t{\n\t\t\t\t// Add/update item on data source\n\t\t\t\tconst filePath = this._fileManager.addFileToParent(fileObj, parentPath);\n\n\t\t\t\t// Refresh view\n\t\t\t\tthis._filesList.refresh();\n\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t{\n\t\t\t\t\t// Expand parent\n\t\t\t\t\tthis._filesList.expand($(`#svm-fileList .file-controls[data-item-id=\"${parentPath}\"]`).closest('tr'));\n\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Hide modal\n\t\t\t\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t\t\t\t// Select upload file\n\t\t\t\t\t\tthis._filesList.select($(`#svm-fileList .file-controls[data-item-id=\"${filePath}\"]`).closest('tr'));\n\t\t\t\t\t}\n\n\t\t\t\t\t// Update selection\n\t\t\t\t\tthis._onFileSelectedChange();\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`Folder created`, `Administrator ${requester} created folder: <strong>${filePath}</strong>`);\n\t\t\t\t\telse\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`File uploaded`, `Administrator ${requester} uploaded file: <strong>${filePath}</strong>`);\n\t\t\t\t}\n\t\t\t}\n\t\t\tcatch (e)\n\t\t\t{\n\t\t\t\t// This should not happen... data source is corrupted?\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(e.message, true);\n\t\t\t}\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\t\t}\n\n\t\t// Servlet files deleted\n\t\telse if (command == this.RESP_FILES_DELETED)\n\t\t{\n\t\t\t// Get name of the user who deleted the file/s\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the list of deleted files\n\t\t\tlet files = data.getSFSArray('files');\n\n\t\t\tlet filesArr = [];\n\n\t\t\t// Update data source\n\t\t\tfor (let j = 0; j < files.size(); j++)\n\t\t\t{\n\t\t\t\tlet path = files.getUtfString(j);\n\t\t\t\tfilesArr.push(path);\n\n\t\t\t\t//------------------------\n\n\t\t\t\t// Remove item from data source; parent item is returned\n\t\t\t\tlet parentItem = this._fileManager.removeFile(path);\n\n\t\t\t\t// Collapse parent if the last of its children was deleted\n\t\t\t\tif (parentItem && !parentItem.hasChildren)\n\t\t\t\t\tthis._filesList.collapse($(`#svm-fileList .file-controls[data-item-id=\"${parentItem.id}\"]`).closest('tr'));\n\t\t\t}\n\n\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`${this._selectedItem.isDir ? 'Folder' : 'File'} deleted`, `${this._selectedItem.isDir ? 'Folder' : 'File'} '${this._selectedItem.name}' deleted successfully`);\n\n\t\t\t\tthis._selectedItem = null;\n\n\t\t\t\tthis._enableInterface(true);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`File deleted`, `Administrator ${requester} deleted the following file${filesArr.length > 1 ? 's' : ''}: <strong>${filesArr.join('<br> ')}</strong>`);\n\t\t\t}\n\n\t\t\t// Reset selection\n\t\t\tthis._onFileSelectedChange();\n\t\t}\n\n\t\t// else if ()\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onRetryClick()\n\t{\n\t\tthis._switchView('svm-init');\n\n\t\t// Re-send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\t_onRefreshClick()\n\t{\n\t\tthis._filesList.clearSelection();\n\t\tthis._refreshDataList();\n\t}\n\n\t_onFileSelectedChange()\n\t{\n\t\t// Hide control buttons on currently selected item\n\t\tif (this._selectedItem)\n\t\t\t$(`#svm-fileList .file-controls[data-item-id=\"${this._selectedItem.id}\"]`).hide();\n\n\t\t// Get selected item\n\t\tlet selectedRows = this._filesList.select();\n\n\t\tif (selectedRows.length > 0)\n\t\t{\n\t\t\t// Save ref. to selected item\n\t\t\tthis._selectedItem = this._filesList.dataItem(selectedRows[0]);\n\n\t\t\t// Show control buttons on new selected item\n\t\t\t$(`#svm-fileList .file-controls[data-item-id=\"${this._selectedItem.id}\"]`).show();\n\t\t}\n\t\telse\n\t\t\tthis._selectedItem = null;\n\t}\n\n\t_showAddFolderModalClick()\n\t{\n\t\tif (this._selectedItem && this._selectedItem.isDir)\n\t\t{\n\t\t\tthis._addFolderModal.modal('show');\n\t\t\t$('#svm-folderNameIn').focus();\n\t\t}\n\t}\n\n\t_onAddFolderClick()\n\t{\n\t\t// The parent folder could have been deleted while user is still typing the name of the new child folder\n\t\tif (!this._selectedItem)\n\t\t{\n\t\t\tthis._addFolderModal.modal('hide');\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to create folder; the parent folder doesn\\'t exist.');\n\t\t\treturn;\n\t\t}\n\n\t\tif (this._addFolderValidator.validate())\n\t\t{\n\t\t\t// Disable modal interface\n\t\t\tthis._enableAddFolderModal(false);\n\n\t\t\tlet data = new SFS2X.SFSObject();\n\t\t\tdata.putUtfString('folder', this._selectedItem.id + this._fileSeparator + $('#svm-folderNameIn').val());\n\n\t\t\t// Send request to server\n\t\t\tthis.sendExtensionRequest(this.REQ_CREATE_FOLDER, data);\n\t\t}\n\t}\n\n\t_onAddFolderModalHidden()\n\t{\n\t\t$('#svm-folderNameIn').val('');\n\t\tthis._resetAddFolderValidation();\n\n\t\t// Enable modal interface\n\t\tthis._enableAddFolderModal(true);\n\t}\n\n\t_showUploadFilesModalClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis._uploadFilesModal.modal('show');\n\t}\n\n\t_onClearFilesClick()\n\t{\n\t\tthis._uploader.clearAllFiles();\n\t}\n\n\t_onUploadStart(e)\n\t{\n\t\t// Disable clear button\n\t\t$('#svm-clearFilesBt').attr('disabled', true);\n\n\t\t// Set destination url\n\t\tconst url = 'http://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.httpPort + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;\n\n\t\te.sender.options.async.saveUrl = url;\n\n\t\t// Set payload\n\t\tconst params = new FormData();\n\t\tparams.append('__module', this._uploadTargetConfig.moduleId);\n\t\tparams.append('__target', this._selectedItem.id);\n\n\t\tfor (let f = 0; f < e.files.length; f++)\n\t\t\tparams.append('files[]', e.files[f].rawFile);\n\n\t\te.formData = params;\n\t}\n\n\t_onUploadEnd(e)\n\t{\n\t\t// Enable clear button\n\t\t$('#svm-clearFilesBt').attr('disabled', false);\n\t}\n\n\t_onFilesUploadEnd(response)\n\t{\n\t\t// Nothing to do: we have to wait the upload process completion to be signaled by the server through the dedicated Extension response\n\n\t\t//=================================================================\n\n\t\t// TODO Should we handle this response in some way? For some unknown reason we always get ok=false and status=0\n\t\t// console.log(response)\n\t\t// console.log(response.ok)\n\t\t// console.log(response.status)\n\t}\n\n\t_onRemoveFileClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis.shellCtrl.showConfirmWarning(`Are you sure you want to delete the selected ${this._selectedItem.isDir ? 'folder' : 'file'}?<br><br>Path: <strong>${this._selectedItem.id}</strong>`, $.proxy(this._onRemoveFileConfirm, this));\n\t}\n\n\t_onRemoveFileConfirm()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Request Servlet files removal\n\t\t// NOTE: for compatibility with older AdminTool, the file to be deleted is sent\n\t\t// in an array of strings, even if we can't delete more than 1 file at once in this AdminTool\n\n\t\tlet files = new SFS2X.SFSArray();\n\t\tfiles.addUtfString(this._selectedItem.id);\n\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putSFSArray('files', files);\n\n\t\tthis.sendExtensionRequest(this.REQ_DELETE_FILES, params);\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('svm-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_enableInterface(enable)\n\t{\n\t\t$('#svm-fileList').attr('disabled', !enable);\n\t\t$('#svm-refreshBt').attr('disabled', !enable);\n\t}\n\n\t_refreshDataList()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Send request to server\n\t\tthis.sendExtensionRequest(this.REQ_GET_SERVLETS)\n\t}\n\n\t_resetAddFolderValidation()\n\t{\n\t\tthis._addFolderValidator.hideMessages();\n\n\t\t// The method above doesn't remove the k-invalid classes and aria-invalid=\"true\" attributes from inputs\n\t\t// Let's do it manually\n\t\t$('#svm-addFolderForm .k-invalid').removeClass('k-invalid');\n\t\t$('#svm-addFolderForm [aria-invalid=\"true\"]').removeAttr('aria-invalid');\n\t}\n\n\t_enableAddFolderModal(enable)\n\t{\n\t\t// Disable modal close buttons\n\t\t$('#svm-addFolderModal button[data-dismiss=\"modal\"]').attr('disabled', !enable);\n\n\t\t// Disable add button\n\t\t$('#svm-addFolderBt').attr('disabled', !enable);\n\n\t\t// Disable fieldset\n\t\t$('#svm-addFolderForm').attr('disabled', !enable);\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChJA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}
764
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-10.bundle.js","sources":["webpack://application/./src/managers/file-datasource-manager.js","webpack://application/./src/modules/servlet-manager.js"],"sourcesContent":["export class FileDataSourceManager\n{\n\tconstructor(libFolder, protectedFolders, fileSeparator)\n\t{\n\t\tthis._protectedFolders = protectedFolders; // Folders which can't be deleted (but their content can)\n\t\tthis._libFolder = libFolder;\n\t\tthis._fileSeparator = fileSeparator;\n\t}\n\n\tget dataSource()\n\t{\n\t\treturn this._dataSource;\n\t}\n\n\tinit()\n\t{\n\t\tthis._dataSource = new kendo.data.TreeListDataSource({\n\t\t\tdata: [],\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'id',\n\t\t\t\t\tparentId: 'parentId',\n\t\t\t\t\texpanded: false\n\t\t\t\t}\n\t\t\t},\n\t\t\tsort: { field: 'name', dir: 'asc' }\n\t\t});\n\t}\n\n\taddFile(fileObj, parentLevel = null)\n\t{\n\t\tlet file = {};\n\n\t\tfile.name = fileObj.getUtfString('name');\n\t\tfile.isDir = fileObj.getBool('isDir');\n\t\tfile.lastMod = fileObj.getLong('lastMod');\n\t\tfile.isLib = (file.isDir && file.name == this._libFolder);\n\t\tfile.isProtected = (file.isDir && this._protectedFolders.indexOf(file.name) > -1);\n\t\tfile.size = 0;\n\n\t\tif (parentLevel == null)\n\t\t\tfile.level = 0;\n\t\telse\n\t\t\tfile.level = parentLevel + 1;\n\n\t\tif (fileObj.containsKey('parent'))\n\t\t{\n\t\t\tfile.parentId = fileObj.getUtfString('parent');\n\t\t\tfile.id = file.parentId + this._fileSeparator + file.name;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tfile.parentId = null;\n\t\t\tfile.id = file.name;\n\t\t}\n\n\t\t// Add child files\n\t\tif (file.isDir)\n\t\t{\n\t\t\tlet filesArr = fileObj.getSFSArray('files');\n\n\t\t\tfor (let i = 0; i < filesArr.size(); i++)\n\t\t\t\tfile.size += this.addFile(filesArr.getSFSObject(i), file.level);\n\t\t}\n\t\telse\n\t\t\tfile.size = fileObj.getLong('size');\n\n\t\t// Add file to data source\n\t\tthis._dataSource.add(file);\n\n\t\t// Return file size\n\t\treturn file.size;\n\t}\n\n\tremoveFile(id)\n\t{\n\t\tlet fileItem = this._dataSource.get(id);\n\n\t\tif (fileItem)\n\t\t{\n\t\t\tif (fileItem.parentId)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tlet parentItem = this._dataSource.get(fileItem.parentId);\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\t\t\t}\n\n\t\t\tthis._dataSource.remove(fileItem);\n\n\t\t\t// Return parent item\n\t\t\tif (fileItem.parentId)\n\t\t\t\treturn this._dataSource.get(fileItem.parentId);\n\t\t}\n\t}\n\n\tgetFileById(id)\n\t{\n\t\treturn this._dataSource.get(id);\n\t}\n\n\taddFileToParent(fileObj, parentId)\n\t{\n\t\tlet parentItem = this._dataSource.get(parentId);\n\n\t\tif (parentItem != null && parentItem.isDir)\n\t\t{\n\t\t\tconst fileId = parentId + this._fileSeparator + fileObj.getUtfString('name');\n\t\t\tlet fileItem = this._dataSource.get(fileId);\n\n\t\t\tif (fileItem != null)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\n\t\t\t\t// Update existing item\n\t\t\t\tfileItem.name = fileObj.getUtfString('name');\n\t\t\t\tfileItem.lastMod = fileObj.getLong('lastMod');\n\t\t\t\tfileItem.size = fileObj.getLong('size');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Add new item\n\t\t\t\tthis.addFile(fileObj, parentItem.level);\n\t\t\t}\n\n\t\t\t// Update parent item size\n\t\t\tthis._updateParentSize(parentItem, fileObj.getLong('size'));\n\n\t\t\treturn fileId;\n\t\t}\n\t\telse\n\t\t\tthrow new Error(`An unexpected error occurred while adding file '${fileObj.getUtfString('name')}' (target: ${parentId}).`);\n\t}\n\n\t_updateParentSize(parentItem, value)\n\t{\n\t\tparentItem.size += value;\n\n\t\tif (parentItem.parentId)\n\t\t{\n\t\t\tlet grandParent = this._dataSource.get(parentItem.parentId);\n\t\t\tthis._updateParentSize(grandParent, value);\n\t\t}\n\t}\n}\n","import {BaseModule} from './base-module';\nimport {FileDataSourceManager} from '../managers/file-datasource-manager';\nimport {bytesToSize} from '../utils/utilities';\n\nexport default class ServletManager extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('servletMan');\n\n\t\t// Outgoing requests\n\t\tthis.REQ_INIT = 'init';\n\t\tthis.REQ_GET_SERVLETS = 'getServlets';\n\t\tthis.REQ_CREATE_FOLDER = 'createFolder';\n\t\tthis.REQ_DELETE_FILES = 'deleteSrvltFiles';\n\n\t\t// Incoming responses\n\t\tthis.RESP_LOCKED = 'lock';\n\t\tthis.RESP_INIT = 'init';\n\t\tthis.RESP_SERVLETS = 'servlets';\n\t\tthis.RESP_FILE_ADDED = 'fileAdded';\n\t\tthis.RESP_FILES_DELETED = 'filesDeleted';\n\t\tthis.RESP_ERROR = 'error';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize progress bar\n\t\t$('#svm-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Add listeners to buttons\n\t\t$('#svm-retryBt').on('click', $.proxy(this._onRetryClick, this));\n\t\t$('#svm-refreshBt').on('click', $.proxy(this._onRefreshClick, this));\n\n\t\t// Initialize files list\n\t\tthis._filesList = $('#svm-fileList').kendoTreeList({\n            dataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: true,\n            columns: [\n                {\n\t\t\t\t\tfield: 'name',\n\t\t\t\t\ttitle: 'Name',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t<div >\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t# if (expanded) { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder-open\"></i>\n\t\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder\"></i>\n\t\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t<i class=\"far fa-file\"></i>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t#: name #\n\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"file-controls flex-grow-1 text-right\" data-item-id=\"#:id#\">\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 create-folder-bt\"><i class=\"fas fa-folder-plus\"></i></button>\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 upload-files-bt\"><i class=\"fas fa-file-upload\"></i></button>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t# if (level > 0 && !isProtected) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 remove-file-bt\"><i class=\"fas fa-minus-circle\"></i></button>\n\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t</div>\n\t\t\t\t\t`),\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'size',\n\t\t\t\t\ttitle: 'Size',\n\t\t\t\t\ttemplate: function(dataItem) {\n\t\t\t\t\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\t\t\t\t\treturn kendo.template(`\n\t\t\t\t\t\t\t#: bytesToSize(size, 2, 'KB') #\n\t\t\t\t\t\t`)(dataItem);\n\t\t\t\t\t},\n\t\t\t\t\twidth: 120,\n\t\t\t\t\tminScreenWidth: 576\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'lastMod',\n\t\t\t\t\ttitle: 'Last Modified',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t#: kendo.toString(new Date(lastMod), 'dd MMM yyyy HH:mm:ss') #\n\t\t\t\t\t`),\n\t\t\t\t\twidth: 200,\n\t\t\t\t\tminScreenWidth: 768\n\t\t\t\t},\n            ],\n\t\t\tchange: $.proxy(this._onFileSelectedChange, this)\n        }).data('kendoTreeList');\n\n\t\t//-------------------------------------------\n\n\t\t// Add listeners to catch control button clicks on rows\n\t\t$('#svm-fileList').on('click', '.create-folder-bt', $.proxy(this._showAddFolderModalClick, this));\n\t\t$('#svm-fileList').on('click', '.upload-files-bt', $.proxy(this._showUploadFilesModalClick, this));\n\t\t$('#svm-fileList').on('click', '.remove-file-bt', $.proxy(this._onRemoveFileClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"add folder\" modal\n\t\tthis._addFolderModal = $('#svm-addFolderModal');\n\t\tthis._addFolderModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Add listener to modal hide event\n\t\tthis._addFolderModal.on('hidden.bs.modal', $.proxy(this._onAddFolderModalHidden, this));\n\n\t\t// Add listener to Add button click\n\t\t$('#svm-addFolderBt').on('click', $.proxy(this._onAddFolderClick, this));\n\n\t\t// Initialize kendo validation on folder name form\n\t\tthis._addFolderValidator = $('#svm-addFolderForm').kendoValidator({}).data('kendoValidator');\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"upload files\" modal\n\t\tthis._uploadFilesModal = $('#svm-uploadModal');\n\t\tthis._uploadFilesModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Initialize kendo uploader\n\t\tthis._uploader = $('#svm-uploader').kendoUpload({\n\t\t\tmultiple: true,\n\t\t\tasync: {\n\t\t\t\tsaveUrl: 'http://localhost', // This will be changed later in _onUploadStart method\n\t\t\t\tautoUpload: true,\n\t\t\t},\n\t\t\tdirectoryDrop: true,\n\t\t\tupload: $.proxy(this._onUploadStart, this),\n\t\t\tcomplete: $.proxy(this._onUploadEnd, this),\n\t\t\tlocalization: {\n\t\t\t\tselect: 'Select files...'\n\t\t\t}\n\t\t}).data('kendoUpload');\n\n\t\t// Add listener to Upload button click\n\t\t$('#svm-clearFilesBt').on('click', $.proxy(this._onClearFilesClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t$('#svm-retryBt').off('click');\n\t\t$('#svm-refreshBt').off('click');\n\n\t\t$('#svm-fileList').off('click');\n\n\t\tthis._addFolderModal.off('hidden.bs.modal');\n\t\tthis._addFolderModal.modal('dispose');\n\t\t$('#svm-addFolderBt').off('click');\n\n\t\tthis._uploadFilesModal.modal('dispose');\n\t\t$('#svm-clearFilesBt').off('click');\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Module can be enabled (no locking file exists)\n\t\tif (command == this.RESP_INIT)\n\t\t{\n\t\t\t// Retrieve file separator\n\t\t\tthis._fileSeparator = data.getUtfString('sep');\n\n\t\t\t// Create file data source manager\n\t\t\tthis._fileManager = new FileDataSourceManager(null, [], this._fileSeparator);\n\n\t\t\t// Retrieve module id sent by the server (required because multiple modules use file uploading service)\n\t\t\tconst uploadModuleId = data.getUtfString('modId');\n\n\t\t\t// Set file uploading target configuration\n\t\t\tthis._uploadTargetConfig = {\n\t\t\t\tsessionToken: this.smartFox.sessionToken,\n\t\t\t\thost: this.smartFox.config.host,\n\t\t\t\tport: this.smartFox.config.port,\n\t\t\t\tmoduleId: uploadModuleId,\n\t\t\t\tprotocol: this.smartFox.config.useSSL ? 'https' : 'http'\n\t\t\t};\n\n\t\t\t// Request Servlet files data to server instance\n\t\t\tthis._refreshDataList();\n\t\t}\n\n\t\t/*\n\t\t * This response is returned if the file UploadsLock.txt exists in the /config folder of the server.\n\t\t * This is an additional security measure to avoid unwanted files to be uploaded by malicius users accessing the server\n\t\t * with the default credentials, in case they have not been changed by the administrator after the installation.\n\t\t * The file must be removed manually before accessing the Servlet Manager module for the first time\n\t\t */\n\t\telse if (command == this.RESP_LOCKED)\n\t\t{\n\t\t\t// Show warning\n\t\t\tthis._switchView('svm-locked');\n\t\t}\n\n\t\t// Servlet folders and files\n\t\telse if (command == this.RESP_SERVLETS)\n\t\t{\n\t\t\t// Retrieve Servlets file list\n\t\t\tlet servletsObj = data.getSFSObject('servlets');\n\n\t\t\t// Initialize manager\n\t\t\tthis._fileManager.init();\n\n\t\t\t// Add list to manager\n\t\t\tthis._fileManager.addFile(servletsObj);\n\n\t\t\t// Set TreeList data source\n\t\t\tthis._filesList.setDataSource(this._fileManager.dataSource);\n\n\t\t\t// Expand first level\n\t\t\tthis._filesList.expand($('#svm-fileList tbody>tr:eq(0)'));\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show module's main view\n\t\t\tthis._switchView('svm-main');\n\t\t}\n\n\t\t// An error occurred while managing servlet files\n\t\telse if (command == this.RESP_ERROR)\n\t\t{\n\t\t\t// Hide add folder modal\n\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t// Re-enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\t// Servlet folder or file added\n\t\telse if (command == this.RESP_FILE_ADDED)\n\t\t{\n\t\t\t// Get name of the user who added the file/folder\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the object representing the file/folder being added\n\t\t\tconst fileObj = data.getSFSObject('file');\n\n\t\t\t// Get the target folder where the new file/folder should be added\n\t\t\tconst parentPath = data.getUtfString('parent');\n\n\t\t\t// Get the flag notifying this was a file upload\n\t\t\tconst isUpload = data.getBool('isUpload');\n\n\t\t\ttry\n\t\t\t{\n\t\t\t\t// Add/update item on data source\n\t\t\t\tconst filePath = this._fileManager.addFileToParent(fileObj, parentPath);\n\n\t\t\t\t// Refresh view\n\t\t\t\tthis._filesList.refresh();\n\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t{\n\t\t\t\t\t// Expand parent\n\t\t\t\t\tthis._filesList.expand($(`#svm-fileList .file-controls[data-item-id=\"${$.escapeSelector(parentPath)}\"]`).closest('tr'));\n\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Hide modal\n\t\t\t\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t\t\t\t// Select upload file\n\t\t\t\t\t\tthis._filesList.select($(`#svm-fileList .file-controls[data-item-id=\"${$.escapeSelector(filePath)}\"]`).closest('tr'));\n\t\t\t\t\t}\n\n\t\t\t\t\t// Update selection\n\t\t\t\t\tthis._onFileSelectedChange();\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`Folder created`, `Administrator ${requester} created folder: <strong>${filePath}</strong>`);\n\t\t\t\t\telse\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`File uploaded`, `Administrator ${requester} uploaded file: <strong>${filePath}</strong>`);\n\t\t\t\t}\n\t\t\t}\n\t\t\tcatch (e)\n\t\t\t{\n\t\t\t\t// This should not happen... data source is corrupted?\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(e.message, true);\n\t\t\t}\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\t\t}\n\n\t\t// Servlet files deleted\n\t\telse if (command == this.RESP_FILES_DELETED)\n\t\t{\n\t\t\t// Get name of the user who deleted the file/s\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the list of deleted files\n\t\t\tlet files = data.getSFSArray('files');\n\n\t\t\tlet filesArr = [];\n\n\t\t\t// Update data source\n\t\t\tfor (let j = 0; j < files.size(); j++)\n\t\t\t{\n\t\t\t\tlet path = files.getUtfString(j);\n\t\t\t\tfilesArr.push(path);\n\n\t\t\t\t//------------------------\n\n\t\t\t\t// Remove item from data source; parent item is returned\n\t\t\t\tlet parentItem = this._fileManager.removeFile(path);\n\n\t\t\t\t// Collapse parent if the last of its children was deleted\n\t\t\t\tif (parentItem && !parentItem.hasChildren)\n\t\t\t\t\tthis._filesList.collapse($(`#svm-fileList .file-controls[data-item-id=\"${$.escapeSelector(parentItem.id)}\"]`).closest('tr'));\n\t\t\t}\n\n\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`${this._selectedItem.isDir ? 'Folder' : 'File'} deleted`, `${this._selectedItem.isDir ? 'Folder' : 'File'} '${this._selectedItem.name}' deleted successfully`);\n\n\t\t\t\tthis._selectedItem = null;\n\n\t\t\t\tthis._enableInterface(true);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`File deleted`, `Administrator ${requester} deleted the following file${filesArr.length > 1 ? 's' : ''}: <strong>${filesArr.join('<br> ')}</strong>`);\n\t\t\t}\n\n\t\t\t// Reset selection\n\t\t\tthis._onFileSelectedChange();\n\t\t}\n\n\t\t// else if ()\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onRetryClick()\n\t{\n\t\tthis._switchView('svm-init');\n\n\t\t// Re-send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\t_onRefreshClick()\n\t{\n\t\tthis._filesList.clearSelection();\n\t\tthis._refreshDataList();\n\t}\n\n\t_onFileSelectedChange()\n\t{\n\t\t// Hide control buttons on currently selected item\n\t\tif (this._selectedItem)\n\t\t\t$(`#svm-fileList .file-controls[data-item-id=\"${$.escapeSelector(this._selectedItem.id)}\"]`).hide();\n\n\t\t// Get selected item\n\t\tlet selectedRows = this._filesList.select();\n\n\t\tif (selectedRows.length > 0)\n\t\t{\n\t\t\t// Save ref. to selected item\n\t\t\tthis._selectedItem = this._filesList.dataItem(selectedRows[0]);\n\n\t\t\t// Show control buttons on new selected item\n\t\t\t$(`#svm-fileList .file-controls[data-item-id=\"${$.escapeSelector(this._selectedItem.id)}\"]`).show();\n\t\t}\n\t\telse\n\t\t\tthis._selectedItem = null;\n\t}\n\n\t_showAddFolderModalClick()\n\t{\n\t\tif (this._selectedItem && this._selectedItem.isDir)\n\t\t{\n\t\t\tthis._addFolderModal.modal('show');\n\t\t\t$('#svm-folderNameIn').focus();\n\t\t}\n\t}\n\n\t_onAddFolderClick()\n\t{\n\t\t// The parent folder could have been deleted while user is still typing the name of the new child folder\n\t\tif (!this._selectedItem)\n\t\t{\n\t\t\tthis._addFolderModal.modal('hide');\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to create folder; the parent folder doesn\\'t exist.');\n\t\t\treturn;\n\t\t}\n\n\t\tif (this._addFolderValidator.validate())\n\t\t{\n\t\t\t// Disable modal interface\n\t\t\tthis._enableAddFolderModal(false);\n\n\t\t\tlet data = new SFS2X.SFSObject();\n\t\t\tdata.putUtfString('folder', this._selectedItem.id + this._fileSeparator + $('#svm-folderNameIn').val());\n\n\t\t\t// Send request to server\n\t\t\tthis.sendExtensionRequest(this.REQ_CREATE_FOLDER, data);\n\t\t}\n\t}\n\n\t_onAddFolderModalHidden()\n\t{\n\t\t$('#svm-folderNameIn').val('');\n\t\tthis._resetAddFolderValidation();\n\n\t\t// Enable modal interface\n\t\tthis._enableAddFolderModal(true);\n\t}\n\n\t_showUploadFilesModalClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis._uploadFilesModal.modal('show');\n\t}\n\n\t_onClearFilesClick()\n\t{\n\t\tthis._uploader.clearAllFiles();\n\t}\n\n\t_onUploadStart(e)\n\t{\n\t\t// Disable clear button\n\t\t$('#svm-clearFilesBt').attr('disabled', true);\n\n\t\t// Set destination url\n\t\tconst url = this._uploadTargetConfig.protocol + '://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.port + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;\n\n\t\te.sender.options.async.saveUrl = url;\n\n\t\t// Set payload\n\t\tconst params = new FormData();\n\t\tparams.append('__module', this._uploadTargetConfig.moduleId);\n\t\tparams.append('__target', this._selectedItem.id);\n\n\t\tfor (let f = 0; f < e.files.length; f++)\n\t\t\tparams.append('files[]', e.files[f].rawFile);\n\n\t\te.formData = params;\n\t}\n\n\t_onUploadEnd(e)\n\t{\n\t\t// Enable clear button\n\t\t$('#svm-clearFilesBt').attr('disabled', false);\n\t}\n\n\t_onFilesUploadEnd(response)\n\t{\n\t\t// Nothing to do: we have to wait the upload process completion to be signaled by the server through the dedicated Extension response\n\n\t\t//=================================================================\n\n\t\t// TODO Should we handle this response in some way? For some unknown reason we always get ok=false and status=0\n\t\t// console.log(response)\n\t\t// console.log(response.ok)\n\t\t// console.log(response.status)\n\t}\n\n\t_onRemoveFileClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis.shellCtrl.showConfirmWarning(`Are you sure you want to delete the selected ${this._selectedItem.isDir ? 'folder' : 'file'}?<br><br>Path: <strong>${this._selectedItem.id}</strong>`, $.proxy(this._onRemoveFileConfirm, this));\n\t}\n\n\t_onRemoveFileConfirm()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Request Servlet files removal\n\t\t// NOTE: for compatibility with older AdminTool, the file to be deleted is sent\n\t\t// in an array of strings, even if we can't delete more than 1 file at once in this AdminTool\n\n\t\tlet files = new SFS2X.SFSArray();\n\t\tfiles.addUtfString(this._selectedItem.id);\n\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putSFSArray('files', files);\n\n\t\tthis.sendExtensionRequest(this.REQ_DELETE_FILES, params);\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('svm-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_enableInterface(enable)\n\t{\n\t\t$('#svm-fileList').attr('disabled', !enable);\n\t\t$('#svm-refreshBt').attr('disabled', !enable);\n\t}\n\n\t_refreshDataList()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Send request to server\n\t\tthis.sendExtensionRequest(this.REQ_GET_SERVLETS)\n\t}\n\n\t_resetAddFolderValidation()\n\t{\n\t\tthis._addFolderValidator.hideMessages();\n\n\t\t// The method above doesn't remove the k-invalid classes and aria-invalid=\"true\" attributes from inputs\n\t\t// Let's do it manually\n\t\t$('#svm-addFolderForm .k-invalid').removeClass('k-invalid');\n\t\t$('#svm-addFolderForm [aria-invalid=\"true\"]').removeAttr('aria-invalid');\n\t}\n\n\t_enableAddFolderModal(enable)\n\t{\n\t\t// Disable modal close buttons\n\t\t$('#svm-addFolderModal button[data-dismiss=\"modal\"]').attr('disabled', !enable);\n\n\t\t// Disable add button\n\t\t$('#svm-addFolderBt').attr('disabled', !enable);\n\n\t\t// Disable fieldset\n\t\t$('#svm-addFolderForm').attr('disabled', !enable);\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChJA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}