mirror of
https://gitlab.com/crafty-controller/crafty-4.git
synced 2024-08-30 18:23:09 +00:00
Merge branch 'pretzel-bedrock' into 'pretzel-fixes'
Pretzel bedrock See merge request crafty-controller/crafty-commander!168
This commit is contained in:
commit
d62dee2da5
@ -132,12 +132,6 @@ def ping(ip, port):
|
|||||||
try:
|
try:
|
||||||
sock.connect((ip, port))
|
sock.connect((ip, port))
|
||||||
|
|
||||||
except socket.error:
|
|
||||||
#We'll try gathering this data on a UDP port next
|
|
||||||
try:
|
|
||||||
sock =socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
|
||||||
sock.connect((ip, port))
|
|
||||||
#If the udp port fails as well we're toast. :sadcat:
|
|
||||||
except socket.error:
|
except socket.error:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@ -171,3 +165,61 @@ def ping(ip, port):
|
|||||||
return Server(json.loads(data))
|
return Server(json.loads(data))
|
||||||
finally:
|
finally:
|
||||||
sock.close()
|
sock.close()
|
||||||
|
|
||||||
|
# For the rest of requests see wiki.vg/Protocol
|
||||||
|
def ping_bedrock(ip, port):
|
||||||
|
def read_var_int():
|
||||||
|
i = 0
|
||||||
|
j = 0
|
||||||
|
while True:
|
||||||
|
k = sock.recv(1)
|
||||||
|
if not k:
|
||||||
|
return 0
|
||||||
|
k = k[0]
|
||||||
|
i |= (k & 0x7f) << (j * 7)
|
||||||
|
j += 1
|
||||||
|
if j > 5:
|
||||||
|
raise ValueError('var_int too big')
|
||||||
|
if not k & 0x80:
|
||||||
|
return i
|
||||||
|
|
||||||
|
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
||||||
|
try:
|
||||||
|
sock.connect((ip, port))
|
||||||
|
|
||||||
|
except:
|
||||||
|
print("in except")
|
||||||
|
return False
|
||||||
|
|
||||||
|
try:
|
||||||
|
host = ip.encode('utf-8')
|
||||||
|
data = b'' # wiki.vg/Server_List_Ping
|
||||||
|
data += b'\x00' # packet ID
|
||||||
|
data += b'\x04' # protocol variant
|
||||||
|
data += struct.pack('>b', len(host)) + host
|
||||||
|
data += struct.pack('>H', port)
|
||||||
|
data += b'\x01' # next state
|
||||||
|
data = struct.pack('>b', len(data)) + data
|
||||||
|
sock.sendall(data + b'\x01\x00') # handshake + status ping
|
||||||
|
try:
|
||||||
|
length = read_var_int() # full packet length
|
||||||
|
except:
|
||||||
|
return False
|
||||||
|
if length < 10:
|
||||||
|
if length < 0:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
sock.recv(1) # packet type, 0 for pings
|
||||||
|
length = read_var_int() # string length
|
||||||
|
data = b''
|
||||||
|
while len(data) != length:
|
||||||
|
chunk = sock.recv(length - len(data))
|
||||||
|
if not chunk:
|
||||||
|
return False
|
||||||
|
|
||||||
|
data += chunk
|
||||||
|
logger.debug(f"Server reports this data on ping: {data}")
|
||||||
|
return Server(json.loads(data))
|
||||||
|
finally:
|
||||||
|
sock.close()
|
||||||
|
@ -9,7 +9,7 @@ from app.classes.models.management import Host_Stats
|
|||||||
from app.classes.models.servers import Server_Stats, servers_helper
|
from app.classes.models.servers import Server_Stats, servers_helper
|
||||||
|
|
||||||
from app.classes.shared.helpers import helper
|
from app.classes.shared.helpers import helper
|
||||||
from app.classes.minecraft.mc_ping import ping
|
from app.classes.minecraft.mc_ping import ping, ping_bedrock
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -177,6 +177,9 @@ class Stats:
|
|||||||
server_port = server['server_port']
|
server_port = server['server_port']
|
||||||
|
|
||||||
logger.debug("Pinging {internal_ip} on port {server_port}")
|
logger.debug("Pinging {internal_ip} on port {server_port}")
|
||||||
|
if servers_helper.get_server_type_by_id(server_id) == 'minecraft-bedrock':
|
||||||
|
int_mc_ping = ping_bedrock(internal_ip, int(server_port))
|
||||||
|
else:
|
||||||
int_mc_ping = ping(internal_ip, int(server_port))
|
int_mc_ping = ping(internal_ip, int(server_port))
|
||||||
|
|
||||||
ping_data = {}
|
ping_data = {}
|
||||||
@ -223,6 +226,9 @@ class Stats:
|
|||||||
server = s.get('server_name', f"ID#{server_id}")
|
server = s.get('server_name', f"ID#{server_id}")
|
||||||
|
|
||||||
logger.debug("Pinging server '{server}' on {internal_ip}:{server_port}")
|
logger.debug("Pinging server '{server}' on {internal_ip}:{server_port}")
|
||||||
|
if servers_helper.get_server_type_by_id(server_id) == 'minecraft-bedrock':
|
||||||
|
int_mc_ping = ping_bedrock(internal_ip, int(server_port))
|
||||||
|
else:
|
||||||
int_mc_ping = ping(internal_ip, int(server_port))
|
int_mc_ping = ping(internal_ip, int(server_port))
|
||||||
|
|
||||||
int_data = False
|
int_data = False
|
||||||
@ -286,6 +292,9 @@ class Stats:
|
|||||||
|
|
||||||
|
|
||||||
logger.debug(f"Pinging server '{server.name}' on {internal_ip}:{server_port}")
|
logger.debug(f"Pinging server '{server.name}' on {internal_ip}:{server_port}")
|
||||||
|
if servers_helper.get_server_type_by_id(server_id) == 'minecraft-bedrock':
|
||||||
|
int_mc_ping = ping_bedrock(internal_ip, int(server_port))
|
||||||
|
else:
|
||||||
int_mc_ping = ping(internal_ip, int(server_port))
|
int_mc_ping = ping(internal_ip, int(server_port))
|
||||||
|
|
||||||
int_data = False
|
int_data = False
|
||||||
|
@ -100,6 +100,7 @@ class helper_servers:
|
|||||||
server_file: str,
|
server_file: str,
|
||||||
server_log_file: str,
|
server_log_file: str,
|
||||||
server_stop: str,
|
server_stop: str,
|
||||||
|
server_type: str,
|
||||||
server_port=25565):
|
server_port=25565):
|
||||||
return Servers.insert({
|
return Servers.insert({
|
||||||
Servers.server_name: name,
|
Servers.server_name: name,
|
||||||
@ -113,7 +114,8 @@ class helper_servers:
|
|||||||
Servers.log_path: server_log_file,
|
Servers.log_path: server_log_file,
|
||||||
Servers.server_port: server_port,
|
Servers.server_port: server_port,
|
||||||
Servers.stop_command: server_stop,
|
Servers.stop_command: server_stop,
|
||||||
Servers.backup_path: backup_path
|
Servers.backup_path: backup_path,
|
||||||
|
Servers.type: server_type
|
||||||
}).execute()
|
}).execute()
|
||||||
|
|
||||||
|
|
||||||
@ -121,6 +123,11 @@ class helper_servers:
|
|||||||
def get_server_obj(server_id):
|
def get_server_obj(server_id):
|
||||||
return Servers.get_by_id(server_id)
|
return Servers.get_by_id(server_id)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_server_type_by_id(server_id):
|
||||||
|
server_type = Servers.select().where(Servers.server_id == server_id).get()
|
||||||
|
return server_type.type
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def update_server(server_obj):
|
def update_server(server_obj):
|
||||||
return server_obj.save()
|
return server_obj.save()
|
||||||
|
@ -7,7 +7,6 @@ import tempfile
|
|||||||
from distutils import dir_util
|
from distutils import dir_util
|
||||||
from typing import Union
|
from typing import Union
|
||||||
from peewee import DoesNotExist
|
from peewee import DoesNotExist
|
||||||
import platform
|
|
||||||
|
|
||||||
from app.classes.controllers.crafty_perms_controller import Crafty_Perms_Controller
|
from app.classes.controllers.crafty_perms_controller import Crafty_Perms_Controller
|
||||||
from app.classes.controllers.management_controller import Management_Controller
|
from app.classes.controllers.management_controller import Management_Controller
|
||||||
@ -297,7 +296,8 @@ class Controller:
|
|||||||
# download the jar
|
# download the jar
|
||||||
server_jar_obj.download_jar(server, version, full_jar_path)
|
server_jar_obj.download_jar(server, version, full_jar_path)
|
||||||
|
|
||||||
new_id = self.register_server(name, server_id, server_dir, backup_path, server_command, server_file, server_log_file, server_stop, port, type='minecraft-java')
|
new_id = self.register_server(name, server_id, server_dir, backup_path, server_command, server_file, server_log_file, server_stop,
|
||||||
|
port, server_type='minecraft-java')
|
||||||
return new_id
|
return new_id
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@ -396,10 +396,12 @@ class Controller:
|
|||||||
server_stop = "stop"
|
server_stop = "stop"
|
||||||
|
|
||||||
new_id = self.register_server(server_name, server_id, new_server_dir, backup_path, server_command, server_jar,
|
new_id = self.register_server(server_name, server_id, new_server_dir, backup_path, server_command, server_jar,
|
||||||
server_log_file, server_stop, port, type='minecraft-java')
|
server_log_file, server_stop, port, server_type='minecraft-java')
|
||||||
return new_id
|
return new_id
|
||||||
|
|
||||||
#---------------------------------------------------BEDROCK IMPORTS---------------------------------------------------
|
#************************************************************************************************
|
||||||
|
# BEDROCK IMPORTS
|
||||||
|
#************************************************************************************************
|
||||||
|
|
||||||
def import_bedrock_server(self, server_name: str, server_path: str, server_exe: str, port: int):
|
def import_bedrock_server(self, server_name: str, server_path: str, server_exe: str, port: int):
|
||||||
server_id = helper.create_uuid()
|
server_id = helper.create_uuid()
|
||||||
@ -484,7 +486,9 @@ class Controller:
|
|||||||
server_log_file, server_stop, port, server_type='minecraft-bedrock')
|
server_log_file, server_stop, port, server_type='minecraft-bedrock')
|
||||||
return new_id
|
return new_id
|
||||||
|
|
||||||
#---------------------------------------------------END BEDROCK IMPORTS---------------------------------------------------
|
#************************************************************************************************
|
||||||
|
# BEDROCK IMPORTS END
|
||||||
|
#************************************************************************************************
|
||||||
|
|
||||||
def rename_backup_dir(self, old_server_id, new_server_id, new_uuid):
|
def rename_backup_dir(self, old_server_id, new_server_id, new_uuid):
|
||||||
server_data = self.servers.get_server_data_by_id(old_server_id)
|
server_data = self.servers.get_server_data_by_id(old_server_id)
|
||||||
|
@ -292,7 +292,7 @@ class ServerHandler(BaseHandler):
|
|||||||
port = bleach.clean(self.get_argument('port', ''))
|
port = bleach.clean(self.get_argument('port', ''))
|
||||||
import_type = bleach.clean(self.get_argument('create_type', ''))
|
import_type = bleach.clean(self.get_argument('create_type', ''))
|
||||||
import_server_path = bleach.clean(self.get_argument('server_path', ''))
|
import_server_path = bleach.clean(self.get_argument('server_path', ''))
|
||||||
import_server_jar = bleach.clean(self.get_argument('server_jar', ''))
|
import_server_exe = bleach.clean(self.get_argument('server_jar', ''))
|
||||||
server_parts = server.split("|")
|
server_parts = server.split("|")
|
||||||
captured_roles = []
|
captured_roles = []
|
||||||
for role in user_roles:
|
for role in user_roles:
|
||||||
@ -310,7 +310,7 @@ class ServerHandler(BaseHandler):
|
|||||||
self.redirect("/panel/error?error=Server path or Server Jar not found!")
|
self.redirect("/panel/error?error=Server path or Server Jar not found!")
|
||||||
return
|
return
|
||||||
|
|
||||||
new_server_id = self.controller.import_jar_server(server_name, import_server_path,import_server_jar, min_mem, max_mem, port)
|
new_server_id = self.controller.import_bedrock_server(server_name, import_server_path,import_server_exe, port)
|
||||||
self.controller.management.add_to_audit_log(exec_user['user_id'],
|
self.controller.management.add_to_audit_log(exec_user['user_id'],
|
||||||
f"imported a jar server named \"{server_name}\"", # Example: Admin imported a server named "old creative"
|
f"imported a jar server named \"{server_name}\"", # Example: Admin imported a server named "old creative"
|
||||||
new_server_id,
|
new_server_id,
|
||||||
@ -323,7 +323,7 @@ class ServerHandler(BaseHandler):
|
|||||||
self.redirect("/panel/error?error=Temp path not found!")
|
self.redirect("/panel/error?error=Temp path not found!")
|
||||||
return
|
return
|
||||||
|
|
||||||
new_server_id = self.controller.import_zip_server(server_name, zip_path, import_server_jar, min_mem, max_mem, port)
|
new_server_id = self.controller.import_bedrock_zip_server(server_name, zip_path, import_server_exe, port)
|
||||||
if new_server_id == "false":
|
if new_server_id == "false":
|
||||||
self.redirect("/panel/error?error=Zip file not accessible! You can fix this permissions issue with" +
|
self.redirect("/panel/error?error=Zip file not accessible! You can fix this permissions issue with" +
|
||||||
f"sudo chown -R crafty:crafty {import_server_path} And sudo chmod 2775 -R {import_server_path}")
|
f"sudo chown -R crafty:crafty {import_server_path} And sudo chmod 2775 -R {import_server_path}")
|
||||||
|
@ -5,7 +5,8 @@
|
|||||||
<i class="fas fa-file-signature"></i>{{ translate('serverDetails', 'terminal', data['lang']) }}</a>
|
<i class="fas fa-file-signature"></i>{{ translate('serverDetails', 'terminal', data['lang']) }}</a>
|
||||||
</li>
|
</li>
|
||||||
{% end %}
|
{% end %}
|
||||||
{% if data['permissions']['Logs'] in data['user_permissions'] %}
|
<!--Bedrock servers don't have logs so we'll only show it if we know it's not a bedrock server.-->
|
||||||
|
{% if data['permissions']['Logs'] in data['user_permissions'] and data['server_data']['type'] != 'minecraft-bedrock'%}
|
||||||
<li class="nav-item term-nav-item">
|
<li class="nav-item term-nav-item">
|
||||||
<a class="nav-link {% if data['active_link'] == 'logs' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=logs" role="tab" aria-selected="false">
|
<a class="nav-link {% if data['active_link'] == 'logs' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=logs" role="tab" aria-selected="false">
|
||||||
<i class="fas fa-file-signature"></i>{{ translate('serverDetails', 'logs', data['lang']) }}</a>
|
<i class="fas fa-file-signature"></i>{{ translate('serverDetails', 'logs', data['lang']) }}</a>
|
||||||
|
@ -312,7 +312,7 @@
|
|||||||
});
|
});
|
||||||
setTimeout(function(){
|
setTimeout(function(){
|
||||||
dialog.modal('hide');
|
dialog.modal('hide');
|
||||||
}, 2000);
|
}, 5000);
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "POST",
|
type: "POST",
|
||||||
headers: {'X-XSRFToken': token},
|
headers: {'X-XSRFToken': token},
|
||||||
|
@ -0,0 +1,650 @@
|
|||||||
|
{% extends ../base.html %}
|
||||||
|
|
||||||
|
{% block title %}Crafty Controller - {{ translate('serverWizard', 'newServer', data['lang']) }}{% end %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
|
||||||
|
<div class="content-wrapper">
|
||||||
|
<div class="content-wrapper">
|
||||||
|
<ul class="nav nav-tabs col-md-12 tab-simple-styled " role="tablist">
|
||||||
|
<li class="nav-item term-nav-item">
|
||||||
|
<a class="nav-link active" href="/server/step1" role="tab" aria-selected="false">
|
||||||
|
<i class="fas fa-file-signature"></i>Minecraft-Java</a>
|
||||||
|
</li>
|
||||||
|
<li class="nav-item term-nav-item">
|
||||||
|
<a class="nav-link" href="/server/bedrock_step1" role="tab" aria-selected="false">
|
||||||
|
<i class="fas fa-file-signature"></i>Minecraft-Bedrock</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
<div class="d-none" id="overlay" onclick="hide(event)"></div>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-6 grid-margin stretch-card">
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
|
||||||
|
<h4>{{ translate('serverWizard', 'newServer', data['lang']) }}</h4>
|
||||||
|
<br />
|
||||||
|
<p class="card-description">
|
||||||
|
|
||||||
|
<form method="post" class="server-wizard" onSubmit="wait_msg()">
|
||||||
|
{% raw xsrf_form_html() %}
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server_type">{{ translate('serverWizard', 'serverType', data['lang']) }}</label>
|
||||||
|
<select required class="form-control form-control-lg select-css" id="server_type" name="server_type" onchange="serverTypeChange(this)">
|
||||||
|
<option value="">{{ translate('serverWizard', 'selectType', data['lang']) }}</option>
|
||||||
|
{% for s in data['server_types'] %}
|
||||||
|
<option value="{{ s }}">{{ s.capitalize() }}</option>
|
||||||
|
{% end %}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server_version">{{ translate('serverWizard', 'serverVersion', data['lang']) }}</label>
|
||||||
|
<select required class="form-control form-control-lg select-css" id="server" name="server">
|
||||||
|
<option value="">{{ translate('serverWizard', 'selectVersion', data['lang']) }}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server_name">{{ translate('serverWizard', 'serverName', data['lang']) }}</label>
|
||||||
|
<input type="text" class="form-control" id="server_name" name="server_name" placeholder="{{ translate('serverWizard', 'myNewServer', data['lang']) }}" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<br />
|
||||||
|
<h4 class="card-title">{{ translate('serverWizard', 'quickSettings', data['lang']) }} <small style="text-transform: none;"> - {{ translate('serverWizard', 'quickSettingsDescription', data['lang']) }}</small></h4>
|
||||||
|
<hr>
|
||||||
|
<div class="row">
|
||||||
|
|
||||||
|
<div class="col-sm-3">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="min_memory1">{{ translate('serverWizard', 'minMem', data['lang']) }} <small> - {{ translate('serverWizard', 'sizeInGB', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="min_memory1" name="min_memory" value="1" step="0.5" min="0.5" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-3 offset-1">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="max_memory1">{{ translate('serverWizard', 'maxMem', data['lang']) }} <small> - {{ translate('serverWizard', 'sizeInGB', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="max_memory1" name="max_memory" value="2" step="0.5" min="0.5" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-3 offset-1">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="port1">{{ translate('serverWizard', 'serverPort', data['lang']) }} <small> - {{ translate('serverWizard', 'defaultPort', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="port1" name="port" value="25565" step="1" min="1" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<div id="accordion-1">
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-header p-2" id="Role-1">
|
||||||
|
<p class="mb-0 p-0" data-toggle="collapse" data-target="#collapseRole-1" aria-expanded="true" aria-controls="collapseRole-1">
|
||||||
|
<i class="fas fa-chevron-down"></i> {{ translate('serverWizard', 'addRole', data['lang']) }} <small style="text-transform: none;"> - {{ translate('serverWizard', 'autoCreate', data['lang']) }}</small>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div id="collapseRole-1" class="collapse" aria-labelledby="Role-1" data-parent="">
|
||||||
|
<div class="card-body scroll">
|
||||||
|
<div class="form-group">
|
||||||
|
{% for r in data['roles'] %}
|
||||||
|
<span class="d-block menu-option"><label><input name="{{ r['role_id'] }}" type="checkbox">
|
||||||
|
{{ r['role_name'].capitalize() }}</label></span>
|
||||||
|
{% end %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<button type="submit" class="btn btn-primary mr-2">{{ translate('serverWizard', 'buildServer', data['lang']) }}</button>
|
||||||
|
<button type="reset" class="btn btn-danger mr-2">{{ translate('serverWizard', 'resetForm', data['lang']) }}</button>
|
||||||
|
|
||||||
|
</form>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-6 grid-margin stretch-card">
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
|
||||||
|
<h4>{{ translate('serverWizard', 'importServer', data['lang']) }}</h4>
|
||||||
|
<br />
|
||||||
|
<p class="card-description">
|
||||||
|
|
||||||
|
<form method="post" class="server-wizard" onSubmit="wait_msg(true)">
|
||||||
|
{% raw xsrf_form_html() %}
|
||||||
|
<input type="hidden" value="import_jar" name="create_type">
|
||||||
|
<div class="row">
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server_name">{{ translate('serverWizard', 'serverName', data['lang']) }}</label>
|
||||||
|
<input type="text" class="form-control" id="server_name" name="server_name" value="" placeholder="{{ translate('serverWizard', 'myNewServer', data['lang']) }}" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server">{{ translate('serverWizard', 'serverPath', data['lang']) }} <small>{{ translate('serverWizard', 'absoluteServerPath', data['lang']) }}</small></label>
|
||||||
|
<input type="text" class="form-control" id="server_path" name="server_path" placeholder="/var/opt/server" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server_jar">{{ translate('serverWizard', 'serverJar', data['lang']) }}</label>
|
||||||
|
<input type="text" class="form-control" id="server_jar" name="server_jar" value="" placeholder="paper.jar" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<br />
|
||||||
|
<h4 class="card-title">{{ translate('serverWizard', 'quickSettings', data['lang']) }} <small style="text-transform: none;"> - {{ translate('serverWizard', 'quickSettingsDescription', data['lang']) }}</small></h4>
|
||||||
|
<hr>
|
||||||
|
<div class="row">
|
||||||
|
|
||||||
|
<div class="col-sm-3">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="min_memory2">{{ translate('serverWizard', 'minMem', data['lang']) }} <small> - {{ translate('serverWizard', 'sizeInGB', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="min_memory2" name="min_memory" value="1" step="0.5" min="0.5" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-3 offset-1">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="max_memory2">{{ translate('serverWizard', 'maxMem', data['lang']) }} <small> - {{ translate('serverWizard', 'sizeInGB', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="max_memory2" name="max_memory" value="2" step="0.5" min="0.5" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-3 offset-1">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="port2">{{ translate('serverWizard', 'serverPort', data['lang']) }} <small> - {{ translate('serverWizard', 'defaultPort', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="port2" name="port" value="25565" step="1" min="1" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<div id="accordion-2">
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-header p-2" id="Role-2">
|
||||||
|
<p class="mb-0 p-0" data-toggle="collapse" data-target="#collapseRole-2" aria-expanded="true" aria-controls="collapseRole-2">
|
||||||
|
<i class="fas fa-chevron-down"></i> {{ translate('serverWizard', 'addRole', data['lang']) }} <small style="text-transform: none;"> - {{ translate('serverWizard', 'autoCreate', data['lang']) }}</small>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div id="collapseRole-2" class="collapse" aria-labelledby="Role-2" data-parent="">
|
||||||
|
<div class="card-body scroll">
|
||||||
|
<div class="form-group">
|
||||||
|
{% for r in data['roles'] %}
|
||||||
|
<span class="d-block menu-option"><label><input name="{{ r['role_id'] }}" type="checkbox">
|
||||||
|
{{ r['role_name'].capitalize() }}</label></span>
|
||||||
|
{% end %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button type="submit" class="btn btn-primary mr-2">{{ translate('serverWizard', 'importServerButton', data['lang']) }}</button>
|
||||||
|
<button type="reset" class="btn btn-danger mr-2">{{ translate('serverWizard', 'resetForm', data['lang']) }}</button>
|
||||||
|
|
||||||
|
</form>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-13 grid-margin stretch-card">
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
|
||||||
|
<h4>{{ translate('serverWizard', 'importZip', data['lang']) }}</h4>
|
||||||
|
<br />
|
||||||
|
<p class="card-description">
|
||||||
|
|
||||||
|
<form name="zip" method="post" class="server-wizard" onSubmit="wait_msg(true)">
|
||||||
|
{% raw xsrf_form_html() %}
|
||||||
|
<input type="hidden" value="import_zip" name="create_type">
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-9">
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server_name">{{ translate('serverWizard', 'serverName', data['lang']) }}</label>
|
||||||
|
<input type="text" class="form-control" id="server_name" name="server_name" value="" placeholder="{{ translate('serverWizard', 'myNewServer', data['lang']) }}" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server">{{ translate('serverWizard', 'zipPath', data['lang']) }} <small>{{ translate('serverWizard', 'absoluteZipPath', data['lang']) }}</small></label>
|
||||||
|
<input type="text" class="form-control" id="server_path" name="server_path" placeholder="/var/opt/server.zip" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server">{{ translate('serverWizard', 'selectRoot', data['lang']) }} <small>{{ translate('serverWizard', 'explainRoot', data['lang']) }}</small></label>
|
||||||
|
<br>
|
||||||
|
<button class="btn btn-primary mr-2" id="root_files_button" type="button">{{ translate('serverWizard', 'clickRoot', data['lang']) }}</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="server_jar">{{ translate('serverWizard', 'serverJar', data['lang']) }}</label>
|
||||||
|
<input type="text" class="form-control" id="server_jar" name="server_jar" value="" placeholder="paper.jar" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<div class="col-sm-3">
|
||||||
|
<h4 class="card-title">{{ translate('serverWizard', 'quickSettings', data['lang']) }} <small style="text-transform: none;"> - {{ translate('serverWizard', 'quickSettingsDescription', data['lang']) }}</small></h4>
|
||||||
|
<hr>
|
||||||
|
<div class="row">
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="min_memory3">{{ translate('serverWizard', 'minMem', data['lang']) }} <small> - {{ translate('serverWizard', 'sizeInGB', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="min_memory3" name="min_memory" value="1" step="0.5" min="0.5" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="max_memory3">{{ translate('serverWizard', 'maxMem', data['lang']) }} <small> - {{ translate('serverWizard', 'sizeInGB', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="max_memory3" name="max_memory" value="2" step="0.5" min="0.5" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="port3">{{ translate('serverWizard', 'serverPort', data['lang']) }} <small> - {{ translate('serverWizard', 'defaultPort', data['lang']) }}</small></label>
|
||||||
|
<input type="number" class="form-control" id="port3" name="port" value="25565" step="1" min="1" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<div id="accordion-3">
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-header p-2" id="Role-3">
|
||||||
|
<p class="mb-0 p-0" data-toggle="collapse" data-target="#collapseRole-3" aria-expanded="true" aria-controls="collapseRole-3">
|
||||||
|
<i class="fas fa-chevron-down"></i> {{ translate('serverWizard', 'addRole', data['lang']) }} <small style="text-transform: none;"> - {{ translate('serverWizard', 'autoCreate', data['lang']) }}</small>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div id="collapseRole-3" class="collapse" aria-labelledby="Role-3" data-parent="">
|
||||||
|
<div class="card-body scroll">
|
||||||
|
<div class="form-group">
|
||||||
|
{% for r in data['roles'] %}
|
||||||
|
<span class="d-block menu-option"><label><input name="{{ r['role_id'] }}" type="checkbox">
|
||||||
|
{{ r['role_name'].capitalize() }}</label></span>
|
||||||
|
{% end %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-12" style="visibility: hidden;">
|
||||||
|
<div class="form-group">
|
||||||
|
<input type="text" class="form-control" id="zip_root_path" name="zip_root_path">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal fade" id="dir_select" tabindex="-1" role="dialog" aria-labelledby="dir_select" aria-hidden="true">
|
||||||
|
<div class="modal-dialog" role="document">
|
||||||
|
<div class="modal-content">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h5 class="modal-title" id="exampleModalLongTitle">{{ translate('serverWizard', 'selectZipDir', data['lang']) }}</h5>
|
||||||
|
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
|
||||||
|
<span aria-hidden="true">×</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body">
|
||||||
|
<div class="tree-ctx-item" id="main-tree-div" data-path="" style="overflow: scroll; max-height:75%;">
|
||||||
|
<input type="radio" id="main-tree-input" name="root_path" value="" checked>
|
||||||
|
<span id="main-tree" class="files-tree-title tree-caret-down root-dir" data-path="">
|
||||||
|
<i class="far fa-folder"></i>
|
||||||
|
<i class="far fa-folder-open"></i>
|
||||||
|
{{ translate('serverFiles', 'files', data['lang']) }}
|
||||||
|
</span>
|
||||||
|
</input>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<button type="button" class="btn btn-secondary" data-dismiss="modal">{{ translate('serverWizard', 'close', data['lang']) }}</button>
|
||||||
|
<button type="button" id="modal-okay" data-dismiss="modal" class="btn btn-primary">{{ translate('serverWizard', 'save', data['lang']) }}</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button id="zip_submit" type="submit" title="You must select server root dir first" disabled class="btn btn-primary mr-2">{{ translate('serverWizard', 'importServerButton', data['lang']) }}</button>
|
||||||
|
<button type="reset" class="btn btn-danger mr-2">{{ translate('serverWizard', 'resetForm', data['lang']) }}</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<style>
|
||||||
|
.scroll {
|
||||||
|
max-height: 12em;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
.menu-btn {
|
||||||
|
font-size: 0.9em;
|
||||||
|
padding: 2px 10px;
|
||||||
|
}
|
||||||
|
.menu {
|
||||||
|
padding-top: 10px;
|
||||||
|
z-index: 200;
|
||||||
|
margin-top: 4px;
|
||||||
|
position: absolute;
|
||||||
|
background-color: #2a2c44;
|
||||||
|
}
|
||||||
|
.menu-option {
|
||||||
|
padding: 6px 20px 6px;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
#overlay {
|
||||||
|
position: absolute;
|
||||||
|
top: 0px;
|
||||||
|
left: 0px;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
z-index: 100;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<style>
|
||||||
|
/* Remove default bullets */
|
||||||
|
.tree-view,
|
||||||
|
.tree-nested {
|
||||||
|
list-style-type: none;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Style the items */
|
||||||
|
.tree-item,
|
||||||
|
.files-tree-title {
|
||||||
|
cursor: pointer;
|
||||||
|
user-select: none; /* Prevent text selection */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Create the caret/arrow with a unicode, and style it */
|
||||||
|
.tree-caret .fa-folder {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
.tree-caret .fa-folder-open {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Rotate the caret/arrow icon when clicked on (using JavaScript) */
|
||||||
|
.tree-caret-down .fa-folder {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.tree-caret-down .fa-folder-open {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Hide the nested list */
|
||||||
|
.tree-nested {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% block js%}
|
||||||
|
<script>
|
||||||
|
document.getElementById("root_files_button").addEventListener("click", function(){
|
||||||
|
if(document.forms["zip"]["server_path"].value != ""){
|
||||||
|
if(document.getElementById('root_files_button').classList.contains('clicked')){
|
||||||
|
document.getElementById('main-tree-div').innerHTML = '<input type="radio" id="main-tree-input" name="root_path" value="" checked><span id="main-tree" class="files-tree-title tree-caret-down root-dir" data-path=""><i class="far fa-folder"></i><i class="far fa-folder-open"></i>{{ translate('serverFiles', 'files', data['lang']) }}</span></input>'
|
||||||
|
}else{
|
||||||
|
document.getElementById('root_files_button').classList.add('clicked')
|
||||||
|
}
|
||||||
|
path = document.forms["zip"]["server_path"].value;
|
||||||
|
console.log(document.forms["zip"]["server_path"].value)
|
||||||
|
var token = getCookie("_xsrf");
|
||||||
|
var dialog = bootbox.dialog({
|
||||||
|
message: '<p class="text-center mb-0"><i class="fa fa-spin fa-cog"></i> Please wait while we gather your files...</p>',
|
||||||
|
closeButton: false
|
||||||
|
});
|
||||||
|
setTimeout(function(){
|
||||||
|
dialog.modal('hide');
|
||||||
|
}, 5000);
|
||||||
|
$.ajax({
|
||||||
|
type: "POST",
|
||||||
|
headers: {'X-XSRFToken': token},
|
||||||
|
url: '/ajax/unzip_server?id=-1&path='+path,
|
||||||
|
});
|
||||||
|
}else{
|
||||||
|
bootbox.alert("You must input a path before selecting this button");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
function dropDown(event) {
|
||||||
|
event.target.parentElement.children[1].classList.remove("d-none");
|
||||||
|
document.getElementById("overlay").classList.remove("d-none");
|
||||||
|
}
|
||||||
|
function hide(event) {
|
||||||
|
var items = document.getElementsByClassName('menu');
|
||||||
|
for (let i = 0; i < items.length; i++) {
|
||||||
|
items[i].classList.add("d-none");
|
||||||
|
}
|
||||||
|
document.getElementById("overlay").classList.add("d-none");
|
||||||
|
}
|
||||||
|
$( document ).ready(function() {
|
||||||
|
console.log('ready');
|
||||||
|
var forms = $('form.server-wizard');
|
||||||
|
forms.each(function(i, formEl) {
|
||||||
|
var form = $(formEl);
|
||||||
|
var min = form.find('[name=min_memory]');
|
||||||
|
var max = form.find('[name=max_memory]');
|
||||||
|
console.log(form, min, max)
|
||||||
|
min.change(function(){
|
||||||
|
check_sizes(max, min, 'min');
|
||||||
|
});
|
||||||
|
max.change(function(){
|
||||||
|
check_sizes(max, min, 'max');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function wait_msg(importing){
|
||||||
|
bootbox.alert({
|
||||||
|
title: importing ? '{% raw translate("serverWizard", "importing", data['lang']) %}' : '{% raw translate("serverWizard", "downloading", data['lang']) %}',
|
||||||
|
message: '<i class="fas fa-cloud-download"></i> {% raw translate("serverWizard", "bePatient", data['lang']) %}',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function show_file_tree(){
|
||||||
|
$("#dir_select").modal();
|
||||||
|
}
|
||||||
|
|
||||||
|
function check_sizes(a, b, changed){
|
||||||
|
max_mem = parseFloat(a.val());
|
||||||
|
min_mem = parseFloat(b.val());
|
||||||
|
if (max_mem < min_mem && changed === 'min'){
|
||||||
|
a.val(min_mem)
|
||||||
|
}
|
||||||
|
if (max_mem < min_mem && changed === 'max'){
|
||||||
|
b.val(max_mem)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTreeView(path) {
|
||||||
|
document.getElementById('zip_submit').disabled = false;
|
||||||
|
path = path
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
type: "GET",
|
||||||
|
url: '/ajax/get_zip_tree?id=-1&path='+path,
|
||||||
|
dataType: 'text',
|
||||||
|
success: function(data){
|
||||||
|
console.log("got response:");
|
||||||
|
console.log(data);
|
||||||
|
|
||||||
|
dataArr = data.split('\n');
|
||||||
|
serverDir = dataArr.shift(); // Remove & return first element (server directory)
|
||||||
|
text = dataArr.join('\n');
|
||||||
|
|
||||||
|
try{
|
||||||
|
document.getElementById('main-tree-div').innerHTML += text;
|
||||||
|
document.getElementById('main-tree').parentElement.classList.add("clicked");
|
||||||
|
}catch{
|
||||||
|
document.getElementById('files-tree').innerHTML = text;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
document.getElementsByClassName('files-tree-title')[0].setAttribute('data-path', serverDir);
|
||||||
|
document.getElementsByClassName('files-tree-title')[0].setAttribute('data-name', 'Files');
|
||||||
|
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getToggleMain(event) {
|
||||||
|
path = event.target.parentElement.getAttribute('data-path');
|
||||||
|
document.getElementById("files-tree").classList.toggle("d-block");
|
||||||
|
document.getElementById(path+"span").classList.toggle("tree-caret-down");
|
||||||
|
document.getElementById(path+"span").classList.toggle("tree-caret");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function getDirView(event) {
|
||||||
|
path = event.target.parentElement.getAttribute('data-path');
|
||||||
|
|
||||||
|
if (document.getElementById(path).classList.contains('clicked')){
|
||||||
|
|
||||||
|
var toggler = document.getElementById(path+"span");
|
||||||
|
|
||||||
|
if (toggler.classList.contains('files-tree-title')){
|
||||||
|
document.getElementById(path+"ul").classList.toggle("d-block");
|
||||||
|
document.getElementById(path+"span").classList.toggle("tree-caret-down");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}else{
|
||||||
|
$.ajax({
|
||||||
|
type: "GET",
|
||||||
|
url: '/ajax/get_zip_dir?id=-1&path='+path,
|
||||||
|
dataType: 'text',
|
||||||
|
success: function(data){
|
||||||
|
console.log("got response:");
|
||||||
|
|
||||||
|
dataArr = data.split('\n');
|
||||||
|
serverDir = dataArr.shift(); // Remove & return first element (server directory)
|
||||||
|
text = dataArr.join('\n');
|
||||||
|
|
||||||
|
try{
|
||||||
|
document.getElementById(path+"span").classList.add('tree-caret-down');
|
||||||
|
document.getElementById(path).innerHTML += text;
|
||||||
|
document.getElementById(path).classList.add("clicked");
|
||||||
|
}catch{
|
||||||
|
console.log("Bad")
|
||||||
|
}
|
||||||
|
|
||||||
|
var toggler = document.getElementById(path);
|
||||||
|
|
||||||
|
if (toggler.classList.contains('files-tree-title')){
|
||||||
|
document.getElementById(path+"span").addEventListener("click", function caretListener() {
|
||||||
|
document.getElementById(path+"ul").classList.toggle("d-block");
|
||||||
|
document.getElementById(path+"span").classList.toggle("tree-caret-down");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (webSocket) {
|
||||||
|
webSocket.on('send_temp_path', function (data) {
|
||||||
|
document.getElementById('main-tree-input').setAttribute('value', data.path)
|
||||||
|
getTreeView(data.path);
|
||||||
|
show_file_tree();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
</script>
|
||||||
|
<script type="text/javascript">
|
||||||
|
//<![CDATA[
|
||||||
|
// array of possible countries in the same order as they appear in the country selection list
|
||||||
|
|
||||||
|
function decodeHtmlCharCodes(str) {
|
||||||
|
return str.replace(""", "\"");
|
||||||
|
}
|
||||||
|
|
||||||
|
function convertHtmlJsonToJavacriptArray(str) {
|
||||||
|
var result = []
|
||||||
|
str = decodeHtmlCharCodes(str)
|
||||||
|
for(var i in str)
|
||||||
|
result.push([i, str [i]]);
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
var text = '{% raw data["js_server_types"] %}';
|
||||||
|
var serverTypesLists = JSON.parse(text);
|
||||||
|
//convertHtmlJsonToJavacriptArray('{{ data["js_server_types"] }}')
|
||||||
|
/* CountryChange() is called from the onchange event of a select element.
|
||||||
|
* param selectObj - the select object which fired the on change event.
|
||||||
|
*/
|
||||||
|
function serverTypeChange(selectObj) {
|
||||||
|
// get the index of the selected option
|
||||||
|
var idx = selectObj.selectedIndex;
|
||||||
|
// get the value of the selected option
|
||||||
|
var which = selectObj.options[idx].value;
|
||||||
|
// use the selected option value to retrieve the list of items from the serverTypesLists array
|
||||||
|
cList = serverTypesLists[which];
|
||||||
|
// get the country select element via its known id
|
||||||
|
var cSelect = document.getElementById("server");
|
||||||
|
// remove the current options from the country select
|
||||||
|
var len=cSelect.options.length;
|
||||||
|
while (cSelect.options.length > 0) {
|
||||||
|
cSelect.remove(0);
|
||||||
|
}
|
||||||
|
var newOption;
|
||||||
|
// create new options ordered by descending
|
||||||
|
for (var i=(cList.length)-1; i>=0; i--) {
|
||||||
|
newOption = document.createElement("option");
|
||||||
|
newOption.value = which+"|"+cList[i]; // assumes option string and value are the same
|
||||||
|
newOption.text=cList[i];
|
||||||
|
// add the new option
|
||||||
|
try {
|
||||||
|
cSelect.add(newOption); // this will fail in DOM browsers but is needed for IE
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
cSelect.appendChild(newOption);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//]]>
|
||||||
|
</script>
|
||||||
|
{% end %}
|
Loading…
Reference in New Issue
Block a user