mirror of
https://gitlab.com/crafty-controller/crafty-4.git
synced 2024-08-30 18:23:09 +00:00
524 lines
17 KiB
HTML
Executable File
524 lines
17 KiB
HTML
Executable File
<!DOCTYPE html>
|
|
<html lang="{{ data['lang_page'] }}">
|
|
|
|
<head>
|
|
<!-- Required meta tags -->
|
|
<meta charset="utf-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
|
|
{% block meta %}{% end %}
|
|
<title>{% block title %}{{ _('Default') }}{% end %}</title>
|
|
|
|
<!-- plugins:css -->
|
|
<link rel="stylesheet" href="/static/assets/vendors/mdi/css/materialdesignicons.min.css">
|
|
<link rel="stylesheet" href="/static/assets/vendors/flag-icon-css/css/flag-icon.min.css">
|
|
<link rel="stylesheet" href="/static/assets/vendors/ti-icons/css/themify-icons.css">
|
|
<link rel="stylesheet" href="/static/assets/vendors/typicons/typicons.css">
|
|
<link rel="stylesheet" href="/static/assets/vendors/fontawesome5/css/all.css">
|
|
<link rel="stylesheet" type="text/css"
|
|
href="https://cdn.datatables.net/v/bs4/dt-1.10.22/fh-3.1.7/r-2.2.6/sc-2.0.3/sp-1.2.2/datatables.min.css" />
|
|
<link rel="stylesheet" href="/static/assets/vendors/css/vendor.bundle.base.css">
|
|
<link rel="stylesheet" href="/static/assets/css/crafty.css">
|
|
|
|
<!-- endinject -->
|
|
|
|
<!-- Plugin css for this page-->
|
|
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.12.4/jquery.min.js"></script>
|
|
<!-- End Plugin css for this page-->
|
|
|
|
<!-- Layout styles -->
|
|
<link rel="stylesheet" href="/static/assets/css/dark/style.css">
|
|
<!-- End Layout styles -->
|
|
|
|
<link rel="shortcut icon" type="image/svg+xml" href="/static/assets/images/logo_small.svg">
|
|
<link rel="alternate icon" href="/static/assets/images/favicon.png" />
|
|
<link rel="stylesheet" href="/static/assets/css/crafty.css">
|
|
|
|
<!-- Alpine.js - The modern jQuery alternative -->
|
|
<script defer src="https://unpkg.com/alpinejs@3.x.x/dist/cdn.min.js"></script>
|
|
<!-- End Alpine.js -->
|
|
|
|
</head>
|
|
|
|
<body class="dark-theme">
|
|
<div class="container-scroller">
|
|
<!-- partial:partials/_navbar.html -->
|
|
<nav class="navbar default-layout col-lg-12 col-12 p-0 fixed-top d-flex flex-row">
|
|
<div class="text-center navbar-brand-wrapper d-flex align-items-top justify-content-center">
|
|
<a class="navbar-brand brand-logo" href="/panel/dashboard">
|
|
<img src="/static/assets/images/logo_long.svg" alt="logo" /> </a>
|
|
<a class="navbar-brand brand-logo-mini" href="/panel/dashboard">
|
|
<img src="/static/assets/images/logo_small.svg" alt="logo" /> </a>
|
|
</div>
|
|
<div class="navbar-menu-wrapper d-flex align-items-center">
|
|
<style>
|
|
body:not(.sidebar-icon-only) .navbar-toggler .mdi-chevron-double-right {
|
|
display: none;
|
|
}
|
|
|
|
body.sidebar-icon-only .navbar-toggler .mdi-chevron-double-left {
|
|
display: none;
|
|
}
|
|
</style>
|
|
<button class="navbar-toggler navbar-toggler align-self-center" type="button" data-toggle="minimize">
|
|
<span class="mdi mdi-chevron-double-left"></span>
|
|
<span class="mdi mdi-chevron-double-right"></span>
|
|
</button>
|
|
|
|
{% include notify.html %}
|
|
|
|
<button class="navbar-toggler navbar-toggler-right d-lg-none align-self-center" type="button"
|
|
data-toggle="offcanvas">
|
|
<span class="mdi mdi-menu"></span>
|
|
</button>
|
|
</div>
|
|
</nav>
|
|
|
|
{% include main_menu.html %}
|
|
|
|
<div class="main-panel">
|
|
|
|
<div class="warnings">
|
|
<noscript class="noscript-warning" style="padding: 20px; background-color: rgb(247, 151, 15);">
|
|
<div>{% raw translate('base', 'doesNotWorkWithoutJavascript', data['lang']) %}</div>
|
|
</noscript>
|
|
</div>
|
|
|
|
{% block content %}
|
|
{% end %}
|
|
|
|
{% include footer.html %}
|
|
|
|
</div>
|
|
<!-- main-panel ends -->
|
|
|
|
</div>
|
|
<!-- page-body-wrapper ends -->
|
|
</div>
|
|
|
|
|
|
<style>
|
|
.notifications {
|
|
position: fixed;
|
|
width: 200px;
|
|
top: 70px;
|
|
right: 0px;
|
|
}
|
|
|
|
.notification {
|
|
position: relative;
|
|
box-sizing: border-box;
|
|
padding: 0.5rem;
|
|
padding-left: 0.7rem;
|
|
width: 180px;
|
|
margin-left: 10px;
|
|
margin-right: 10px;
|
|
background: #282a40;
|
|
transition: right 0.75s, opacity 0.75s, top 0.75s;
|
|
right: -6rem;
|
|
opacity: 0.1;
|
|
margin-bottom: 1rem;
|
|
z-index: 999;
|
|
top: 0px;
|
|
}
|
|
|
|
.notification.active {
|
|
right: 0rem;
|
|
opacity: 1;
|
|
}
|
|
|
|
.notification.remove {
|
|
right: 0rem;
|
|
opacity: 0.1;
|
|
top: -2rem;
|
|
}
|
|
|
|
.notification p {
|
|
margin: 0px;
|
|
width: calc(160.8px - 16px);
|
|
z-index: inherit;
|
|
}
|
|
|
|
.notification span {
|
|
position: absolute;
|
|
right: 0.5rem;
|
|
top: 0.46rem;
|
|
cursor: pointer;
|
|
font-weight: bold;
|
|
line-height: 20px;
|
|
font-size: 22px;
|
|
user-select: none;
|
|
z-index: inherit;
|
|
}
|
|
</style>
|
|
<div class="notifications"></div>
|
|
|
|
|
|
<script src="/static/assets/vendors/js/vendor.bundle.base.js"></script>
|
|
<script src="/static/assets/js/shared/off-canvas.js"></script>
|
|
<script src="/static/assets/js/shared/hoverable-collapse.js"></script>
|
|
<script src="/static/assets/js/shared/misc.js"></script>
|
|
<script type="text/javascript"
|
|
src="https://cdn.datatables.net/v/bs4/dt-1.10.22/fh-3.1.7/r-2.2.6/sc-2.0.3/sp-1.2.2/datatables.min.js"></script>
|
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootbox.js/5.4.0/bootbox.min.js"></script>
|
|
<script type="text/javascript" src="/static/assets/js/motd.js"></script>
|
|
|
|
<script>
|
|
/**
|
|
* Returns a random number between min (inclusive) and max (exclusive)
|
|
*/
|
|
function getRandomArbitrary(min, max) {
|
|
return Math.random() * (max - min) + min;
|
|
}
|
|
|
|
/**
|
|
* Returns a random integer between min (inclusive) and max (inclusive).
|
|
* The value is no lower than min (or the next integer greater than min
|
|
* if min isn't an integer) and no greater than max (or the next integer
|
|
* lower than max if max isn't an integer).
|
|
* Using Math.round() will give you a non-uniform distribution!
|
|
*/
|
|
function getRandomInt(min, max) {
|
|
min = Math.ceil(min);
|
|
max = Math.floor(max);
|
|
return Math.floor(Math.random() * (max - min + 1)) + min;
|
|
}
|
|
|
|
|
|
$.extend($.fn.dataTable.defaults, {
|
|
// {{ '\nlanguage:' }} {% raw translate('datatables', 'i18n', data['lang']) %}
|
|
})
|
|
|
|
//used to get cookies from browser - this is part of tornados xsrf protection - it's for extra security
|
|
function getCookie(name) {
|
|
var r = document.cookie.match("\\b" + name + "=([^;]*)\\b");
|
|
return r ? r[1] : undefined;
|
|
}
|
|
|
|
// tool tips
|
|
$(function () {
|
|
$('[data-toggle="tooltip"]').tooltip()
|
|
})
|
|
|
|
// Notify
|
|
$(document).ready(function () {
|
|
$("#notificationDropdown").click(function () {
|
|
$.get("/ajax/announcements", function (data) {
|
|
console.log(data);
|
|
bootbox.alert({
|
|
title: "Notifications",
|
|
message: data,
|
|
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
let usingWebSockets = false;
|
|
let webSocket = null;
|
|
// {% if request.protocol == 'https' %}
|
|
usingWebSockets = true;
|
|
|
|
let listenEvents = [];
|
|
let wsOpen = false;
|
|
/**
|
|
* @type {number | null} reconnectorId An interval ID for the reconnector.
|
|
*/
|
|
let reconnectorId = null;
|
|
let failedConnectionCounter = 0; // https://stackoverflow.com/a/37038217/15388424
|
|
|
|
const wsPageQueryParams = 'page_query_params=' + encodeURIComponent(location.search)
|
|
const wsPage = 'page=' + encodeURIComponent(location.pathname)
|
|
|
|
const sendWssError = () => wsOpen || warn(
|
|
'WebSockets are required for Crafty to work. This websocket connection has been closed. Are you using a reverse proxy?',
|
|
'https://wiki.craftycontrol.com/en/4/docs/Reverse%20Proxy%20Examples',
|
|
'wssError'
|
|
)
|
|
|
|
function startWebSocket() {
|
|
console.log('%c[Crafty Controller] %cConnecting the WebSocket', 'font-weight: 900; color: #800080;', 'font-weight: 900; color: #eee;');
|
|
try {
|
|
var wsInternal = new WebSocket('wss://' + location.host + '/ws?' + wsPage + '&' + wsPageQueryParams);
|
|
wsInternal.onopen = function () {
|
|
console.log('opened WebSocket connection:', wsInternal)
|
|
wsOpen = true;
|
|
failedConnectionCounter = 0;
|
|
if (typeof reconnectorId === 'number') {
|
|
document.querySelectorAll('.wssError').forEach(el => el.remove())
|
|
clearInterval(reconnectorId);
|
|
reconnectorId = null;
|
|
}
|
|
};
|
|
wsInternal.onmessage = function (rawMessage) {
|
|
var message = JSON.parse(rawMessage.data);
|
|
|
|
console.log('got message: ', message)
|
|
|
|
listenEvents
|
|
.filter(listenedEvent => listenedEvent.event == message.event)
|
|
.forEach(listenedEvent => listenedEvent.callback(message.data))
|
|
};
|
|
wsInternal.onerror = function (errorEvent) {
|
|
console.error('WebSocket Error', errorEvent);
|
|
};
|
|
wsInternal.onclose = function (closeEvent) {
|
|
wsOpen = false;
|
|
console.log('Closed WebSocket', closeEvent);
|
|
|
|
if (typeof reconnectorId !== 'number') {
|
|
setTimeout(sendWssError, 7000);
|
|
}
|
|
console.info("Reconnecting with a timeout of", (getRandomArbitrary(0, 2 ** failedConnectionCounter - 1) + 5) * 1000, "milliseconds");
|
|
// Discard old websocket and create a new one in 5 seconds
|
|
wsInternal = null
|
|
reconnectorId = setTimeout(startWebSocket, (getRandomArbitrary(0, 2 ** failedConnectionCounter - 1) + 5) * 1000)
|
|
|
|
failedConnectionCounter++;
|
|
};
|
|
|
|
|
|
webSocket = {
|
|
on: function (event, callback) {
|
|
console.log('registered ' + event + ' event');
|
|
listenEvents.push({ event: event, callback: callback })
|
|
},
|
|
emit: function (event, data) {
|
|
var message = {
|
|
event: event,
|
|
data: data
|
|
}
|
|
|
|
wsInternal.send(JSON.stringify(message));
|
|
}
|
|
}
|
|
} catch (error) {
|
|
console.error('Error while making websocket helpers', error);
|
|
usingWebSockets = false;
|
|
}
|
|
}
|
|
|
|
startWebSocket();
|
|
// {% else %}
|
|
warn('WebSockets are not supported in Crafty if not using the https protocol')
|
|
// {% end%}
|
|
|
|
if (webSocket) {
|
|
webSocket.on('send_start_error', function (start_error) {
|
|
var x = document.querySelector('.bootbox');
|
|
if (x) {
|
|
x.remove()
|
|
}
|
|
var x = document.querySelector('.modal-backdrop');
|
|
if (x) {
|
|
x.remove()
|
|
}
|
|
bootbox.alert({
|
|
message: start_error.error,
|
|
callback: function () {
|
|
location.reload();
|
|
}
|
|
})
|
|
});
|
|
webSocket.on('support_status_update', function (logs) {
|
|
if (logs.percent >= 100) {
|
|
document.getElementById('logs_progress_bar').innerHTML = '100%';
|
|
document.getElementById('logs_progress_bar').style.width = '100%';
|
|
} else {
|
|
document.getElementById('logs_progress_bar').innerHTML = logs.percent + '%';
|
|
document.getElementById('logs_progress_bar').style.width = logs.percent + '%';
|
|
}
|
|
});
|
|
webSocket.on('send_logs_bootbox', function (server_id) {
|
|
var x = document.querySelector('.bootbox');
|
|
if (x) {
|
|
x.remove()
|
|
}
|
|
var x = document.querySelector('.modal-backdrop');
|
|
if (x) {
|
|
x.remove()
|
|
}
|
|
bootbox.alert({
|
|
title: "{{ translate('notify', 'downloadLogs', data['lang']) }}",
|
|
message: "{{ translate('notify', 'finishedPreparing', data['lang']) }}",
|
|
buttons: {
|
|
ok: {
|
|
label: 'Download',
|
|
className: 'btn-info'
|
|
}
|
|
},
|
|
callback: function () {
|
|
console.log("in callback")
|
|
location.href = "/panel/download_support_package";
|
|
}
|
|
});
|
|
});
|
|
webSocket.on('send_eula_bootbox', function (server_id) {
|
|
var x = document.querySelector('.bootbox');
|
|
if (x) {
|
|
x.remove()
|
|
}
|
|
var x = document.querySelector('.modal-backdrop');
|
|
if (x) {
|
|
x.remove()
|
|
}
|
|
bootbox.confirm({
|
|
title: "{% raw translate('error', 'eulaTitle', data['lang']) %}",
|
|
message: "{% raw translate('error', 'eulaMsg', data['lang']) %} <br><br><a href='https://account.mojang.com/documents/minecraft_eula' target='_blank'>EULA</a><br><br>{% raw translate('error', 'eulaAgree', data['lang']) %}",
|
|
buttons: {
|
|
confirm: {
|
|
label: 'Yes',
|
|
className: 'btn-info'
|
|
},
|
|
cancel: {
|
|
label: 'No',
|
|
className: 'btn-secondary'
|
|
}
|
|
},
|
|
callback: function (result) {
|
|
if (result == true) {
|
|
eulaAgree(server_id.id)
|
|
}
|
|
else {
|
|
location.reload()
|
|
}
|
|
}
|
|
})
|
|
});
|
|
}
|
|
|
|
function eulaAgree(server_id, command) {
|
|
//< !--this getCookie function is in base.html-- >
|
|
var token = getCookie("_xsrf");
|
|
|
|
$.ajax({
|
|
type: "POST",
|
|
headers: { 'X-XSRFToken': token },
|
|
url: '/ajax/eula?id=' + server_id,
|
|
success: function (data) {
|
|
console.log("got response:");
|
|
console.log(data);
|
|
location.reload();
|
|
}
|
|
});
|
|
}
|
|
|
|
|
|
function warn(message, link = null, className = null) {
|
|
var closeEl = document.createElement('span');
|
|
var strongEL = document.createElement('strong');
|
|
var msgEl = document.createElement('div');
|
|
|
|
closeEl.innerHTML = '×';
|
|
strongEL.textContent = 'Warning: ';
|
|
msgEl.append(strongEL, message);
|
|
|
|
|
|
closeEl.style.marginLeft = '15px';
|
|
closeEl.style.fontWeight = 'bold';
|
|
closeEl.style.float = 'right';
|
|
closeEl.style.fontSize = '22px';
|
|
closeEl.style.lineHeight = '20px';
|
|
closeEl.style.cursor = 'pointer';
|
|
|
|
closeEl.addEventListener('click', function () { this.parentElement.style.display = 'none'; });
|
|
|
|
var parentEl = document.createElement('div');
|
|
|
|
parentEl.style.padding = '20px';
|
|
parentEl.style.backgroundColor = '#f7970f';
|
|
|
|
parentEl.appendChild(closeEl);
|
|
parentEl.appendChild(msgEl);
|
|
if (link) {
|
|
let linkEl = document.createElement('a')
|
|
linkEl.href = link;
|
|
linkEl.innerHTML = "See our documentation for details.";
|
|
linkEl.style.color = 'white';
|
|
linkEl.style.textDecoration = 'underline';
|
|
linkEl.target = "_blank";
|
|
|
|
parentEl.appendChild(linkEl);
|
|
}
|
|
|
|
if (className) {
|
|
parentEl.classList.add(className);
|
|
}
|
|
|
|
document.querySelector('.warnings').appendChild(parentEl);
|
|
}
|
|
|
|
function closeNotification(element) {
|
|
element.parentElement.classList.add('remove');
|
|
setTimeout(function () {
|
|
element.parentElement.remove();
|
|
}, 500);
|
|
}
|
|
|
|
function notify(message) {
|
|
console.log(`notify(${message})`);
|
|
var paragraphEl = document.createElement('p');
|
|
var closeEl = document.createElement('span');
|
|
|
|
paragraphEl.textContent = message;
|
|
|
|
closeEl.innerHTML = '×';
|
|
closeEl.addEventListener('click', function () { closeNotification(this) });
|
|
|
|
var parentEl = document.createElement('div');
|
|
parentEl.appendChild(paragraphEl);
|
|
parentEl.appendChild(closeEl);
|
|
|
|
parentEl.classList.add('notification');
|
|
|
|
document.querySelector('.notifications').appendChild(parentEl);
|
|
|
|
|
|
setTimeout(function () {
|
|
parentEl.classList.add('active');
|
|
}, 200);
|
|
|
|
setTimeout(function (element) {
|
|
closeNotification(element);
|
|
}, 7500, closeEl);
|
|
|
|
`
|
|
<div class="notification">
|
|
<p>Hello, World! This text should overflow</p>
|
|
<span>×</span>
|
|
</div>
|
|
`
|
|
}
|
|
webSocket.on('notification', notify);
|
|
|
|
document.addEventListener('alpine:init', () => {
|
|
console.log('%c[Crafty Controller] %cAlpine.js pre-initialization!', 'font-weight: 900; color: #800080;', 'color: #eee;');
|
|
})
|
|
document.addEventListener('alpine:initialized', () => {
|
|
console.log('%c[Crafty Controller] %cAlpine.js initialized!', 'font-weight: 900; color: #800080;', 'color: #eee;');
|
|
})
|
|
|
|
$(document).ready(function () {
|
|
console.log('%c[Crafty Controller] %cReady for JS!', 'font-weight: 900; color: #800080;', 'font-weight: 900; color: #eee;');
|
|
$('#support_logs').click(function () {
|
|
var dialog = bootbox.dialog({
|
|
message: "<p class='text- center mb - 0'><i class='fa fa - spin fa - cog'></i>{{ translate('notify', 'preparingLogs', data['lang']) }}</p>",
|
|
closeButton: false
|
|
});
|
|
setTimeout(function () {
|
|
location.href = "/panel/support_logs";
|
|
}, 6000);
|
|
|
|
});
|
|
});
|
|
|
|
</script>
|
|
|
|
{% block js %}
|
|
<!-- Custom js for base.html page partial pages -->
|
|
<!-- End custom js for base.html page -->
|
|
{% end %}
|
|
|
|
</body>
|
|
|
|
</html>
|