prettyify code

This commit is contained in:
Lee
2023-12-30 23:03:54 +00:00
parent 6fd5fdb7fe
commit ea15b979d5
28 changed files with 2179 additions and 1688 deletions

View File

@ -1,17 +1,17 @@
@font-face {
font-family: 'icomoon';
src:
url('../fonts/icomoon.ttf?gn52nv') format('truetype'),
url('../fonts/icomoon.woff?gn52nv') format('woff'),
url('../fonts/icomoon.svg?gn52nv#icomoon') format('svg');
font-family: "icomoon";
src: url("../fonts/icomoon.ttf?gn52nv") format("truetype"),
url("../fonts/icomoon.woff?gn52nv") format("woff"),
url("../fonts/icomoon.svg?gn52nv#icomoon") format("svg");
font-weight: normal;
font-style: normal;
font-display: block;
}
[class^="icon-"], [class*=" icon-"] {
[class^="icon-"],
[class*=" icon-"] {
/* use !important to prevent issues with browser extensions that change fonts */
font-family: 'icomoon' !important;
font-family: "icomoon" !important;
speak: none;
font-style: normal;
font-weight: normal;

View File

@ -3,20 +3,20 @@
@import url(../css/icons.css);
* {
margin: 0;
padding: 0;
margin: 0;
padding: 0;
}
:root {
--color-dark-gray: #A3A3A3;
--color-gold: #FFD700;
--color-dark-gray: #a3a3a3;
--color-gold: #ffd700;
--color-dark-purple: #6c5ce7;
--color-light-purple: #a29bfe;
--color-dark-blue: #0984e3;
--color-light-blue: #74b9ff;
--theme-color-dark: #3B3738;
--theme-color-light: #EBEBEB;
--theme-color-dark: #3b3738;
--theme-color-light: #ebebeb;
--border-radius: 1px;
@ -27,12 +27,12 @@
--background-color: var(--theme-color-light);
--text-decoration-color: var(--theme-color-dark);
--text-color: #000;
--text-color-inverted: #FFF;
--text-color-inverted: #fff;
}
body {
background: #212021;
color: #FFF;
color: #fff;
}
@media (prefers-color-scheme: dark) {
@ -42,13 +42,13 @@ body {
--color-blue-inverted: var(--color-light-blue);
--background-color: var(--theme-color-dark);
--text-decoration-color: var(--theme-color-light);
--text-color: #FFF;
--text-color: #fff;
--text-color-inverted: #000;
}
body {
background: #1c1b1c;
color: #FFF;
color: #fff;
}
}
@ -60,8 +60,9 @@ body {
}
/* Page layout */
html, body {
height: 100%;
html,
body {
height: 100%;
}
a {
@ -81,7 +82,7 @@ a:hover {
}
strong {
font-weight: 700;
font-weight: 700;
}
/* Logo */
@ -316,12 +317,14 @@ footer a:hover {
padding-right: 65px;
}
#big-graph, #big-graph-controls, #big-graph-checkboxes {
width: 90%;
#big-graph,
#big-graph-controls,
#big-graph-checkboxes {
width: 90%;
}
#big-graph-checkboxes > table {
width: 100%;
width: 100%;
}
#big-graph {

View File

@ -1,73 +1,95 @@
<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" href="../css/main.css" />
<head>
<link
rel="stylesheet"
href="https://fonts.googleapis.com/css2?family=Open+Sans:wght@300;700&display=swap"
/>
<link rel="stylesheet" href="../css/main.css">
<link rel="icon" type="image/svg+xml" href="../images/logo.svg" />
<link rel="stylesheet" href="https://fonts.googleapis.com/css2?family=Open+Sans:wght@300;700&display=swap">
<meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="../images/logo.svg">
<script defer type="module" src="../js/main.js"></script>
<meta charset="UTF-8">
<title>Minetrack</title>
</head>
<script defer type="module" src="../js/main.js"></script>
<body>
<div id="tooltip"></div>
<title>Minetrack</title>
<div id="status-overlay">
<img class="logo-image" src="../images/logo.svg" />
<h1 class="logo-text">Minetrack</h1>
<div id="status-text">Connecting...</div>
</div>
</head>
<div id="push">
<div id="perc-bar"></div>
<body>
<header>
<div class="header-possible-row-break column-left">
<img class="logo-image" src="../images/logo.svg" />
<h1 class="logo-text">Minetrack</h1>
<p class="logo-status">
Counting
<span class="global-stat" id="stat_totalPlayers">0</span> players on
<span class="global-stat" id="stat_networks">0</span> Minecraft
servers.
</p>
</div>
<div id="tooltip"></div>
<div class="header-possible-row-break column-right">
<div id="sort-by" class="header-button header-button-single">
<span class="icon-sort-amount-desc"></span> Sort By<br /><strong
id="sort-by-text"
>...</strong
>
</div>
<div id="status-overlay">
<img class="logo-image" src="../images/logo.svg">
<h1 class="logo-text">Minetrack</h1>
<div id="status-text">Connecting...</div>
</div>
<div
id="settings-toggle"
class="header-button header-button-single"
style="margin-left: 20px"
>
<span class="icon-gears"></span> Graph Controls
</div>
</div>
</header>
<div id="push">
<div id="big-graph"></div>
<div id="perc-bar"></div>
<div id="big-graph-controls">
<div id="big-graph-controls-drawer">
<div id="big-graph-checkboxes"></div>
<header>
<div class="header-possible-row-break column-left">
<img class="logo-image" src="../images/logo.svg">
<h1 class="logo-text">Minetrack</h1>
<p class="logo-status">Counting <span class="global-stat" id="stat_totalPlayers">0</span> players on <span class="global-stat" id="stat_networks">0</span> Minecraft servers.</p>
</div>
<span class="graph-controls-setall">
<a minetrack-show-type="all" class="button graph-controls-show"
><span class="icon-eye"></span> Show All</a
>
<a minetrack-show-type="none" class="button graph-controls-show"
><span class="icon-eye-slash"></span> Hide All</a
>
<a
minetrack-show-type="favorites"
class="button graph-controls-show"
><span class="icon-star"></span> Only Favorites</a
>
</span>
</div>
</div>
<div class="header-possible-row-break column-right">
<div id="sort-by" class="header-button header-button-single"><span class="icon-sort-amount-desc"></span> Sort By<br><strong id="sort-by-text">...</strong></div>
<div id="settings-toggle" class="header-button header-button-single" style="margin-left: 20px;"><span class="icon-gears"></span> Graph Controls</div>
</div>
</header>
<div id="big-graph"></div>
<div id="big-graph-controls">
<div id="big-graph-controls-drawer">
<div id="big-graph-checkboxes"></div>
<span class="graph-controls-setall">
<a minetrack-show-type="all" class="button graph-controls-show"><span class="icon-eye"></span> Show All</a>
<a minetrack-show-type="none" class="button graph-controls-show"><span class="icon-eye-slash"></span> Hide All</a>
<a minetrack-show-type="favorites" class="button graph-controls-show"><span class="icon-star"></span> Only Favorites</a>
</span>
</div>
</div>
<div id="server-list"></div>
</div>
<footer id="footer">
<span class="icon-code"></span> Powered by open source software - <a href="https://github.com/Cryptkeeper/Minetrack">make it your own!</a>
</footer>
</body>
<div id="server-list"></div>
</div>
<footer id="footer">
<span class="icon-code"></span> Powered by open source software -
<a href="https://git.fascinated.cc/Fascinated/Minetrack"
>make it your own!</a
>
</footer>
</body>
</html>

View File

@ -1,100 +1,105 @@
import { ServerRegistry } from './servers'
import { SocketManager } from './socket'
import { SortController } from './sort'
import { GraphDisplayManager } from './graph'
import { PercentageBar } from './percbar'
import { FavoritesManager } from './favorites'
import { Tooltip, Caption, formatNumber } from './util'
import { FavoritesManager } from "./favorites";
import { GraphDisplayManager } from "./graph";
import { PercentageBar } from "./percbar";
import { ServerRegistry } from "./servers";
import { SocketManager } from "./socket";
import { SortController } from "./sort";
import { Caption, Tooltip, formatNumber } from "./util";
export class App {
publicConfig
publicConfig;
constructor () {
this.tooltip = new Tooltip()
this.caption = new Caption()
this.serverRegistry = new ServerRegistry(this)
this.socketManager = new SocketManager(this)
this.sortController = new SortController(this)
this.graphDisplayManager = new GraphDisplayManager(this)
this.percentageBar = new PercentageBar(this)
this.favoritesManager = new FavoritesManager(this)
constructor() {
this.tooltip = new Tooltip();
this.caption = new Caption();
this.serverRegistry = new ServerRegistry(this);
this.socketManager = new SocketManager(this);
this.sortController = new SortController(this);
this.graphDisplayManager = new GraphDisplayManager(this);
this.percentageBar = new PercentageBar(this);
this.favoritesManager = new FavoritesManager(this);
this._taskIds = []
this._taskIds = [];
}
// Called once the DOM is ready and the app can begin setup
init () {
this.socketManager.createWebSocket()
init() {
this.socketManager.createWebSocket();
}
setPageReady (isReady) {
document.getElementById('push').style.display = isReady ? 'block' : 'none'
document.getElementById('footer').style.display = isReady ? 'block' : 'none'
document.getElementById('status-overlay').style.display = isReady ? 'none' : 'block'
setPageReady(isReady) {
document.getElementById("push").style.display = isReady ? "block" : "none";
document.getElementById("footer").style.display = isReady
? "block"
: "none";
document.getElementById("status-overlay").style.display = isReady
? "none"
: "block";
}
setPublicConfig (publicConfig) {
this.publicConfig = publicConfig
setPublicConfig(publicConfig) {
this.publicConfig = publicConfig;
this.serverRegistry.assignServers(publicConfig.servers)
this.serverRegistry.assignServers(publicConfig.servers);
// Start repeating frontend tasks once it has received enough data to be considered active
// This simplifies management logic at the cost of each task needing to safely handle empty data
this.initTasks()
this.initTasks();
}
handleSyncComplete () {
this.caption.hide()
handleSyncComplete() {
this.caption.hide();
// Load favorites since all servers are registered
this.favoritesManager.loadLocalStorage()
this.favoritesManager.loadLocalStorage();
// Run a single bulk server sort instead of per-add event since there may be multiple
this.sortController.show()
this.percentageBar.redraw()
this.sortController.show();
this.percentageBar.redraw();
// The data may not be there to correctly compute values, but run an attempt
// Otherwise they will be updated by #initTasks
this.updateGlobalStats()
this.updateGlobalStats();
}
initTasks () {
this._taskIds.push(setInterval(this.sortController.sortServers, 5000))
initTasks() {
this._taskIds.push(setInterval(this.sortController.sortServers, 5000));
}
handleDisconnect () {
this.tooltip.hide()
handleDisconnect() {
this.tooltip.hide();
// Reset individual tracker elements to flush any held data
this.serverRegistry.reset()
this.socketManager.reset()
this.sortController.reset()
this.graphDisplayManager.reset()
this.percentageBar.reset()
this.serverRegistry.reset();
this.socketManager.reset();
this.sortController.reset();
this.graphDisplayManager.reset();
this.percentageBar.reset();
// Undefine publicConfig, resynced during the connection handshake
this.publicConfig = undefined
this.publicConfig = undefined;
// Clear all task ids, if any
this._taskIds.forEach(clearInterval)
this._taskIds.forEach(clearInterval);
this._taskIds = []
this._taskIds = [];
// Reset hidden values created by #updateGlobalStats
this._lastTotalPlayerCount = undefined
this._lastServerRegistrationCount = undefined
this._lastTotalPlayerCount = undefined;
this._lastServerRegistrationCount = undefined;
// Reset modified DOM structures
document.getElementById('stat_totalPlayers').innerText = 0
document.getElementById('stat_networks').innerText = 0
document.getElementById("stat_totalPlayers").innerText = 0;
document.getElementById("stat_networks").innerText = 0;
this.setPageReady(false)
this.setPageReady(false);
}
getTotalPlayerCount () {
return this.serverRegistry.getServerRegistrations()
.map(serverRegistration => serverRegistration.playerCount)
.reduce((sum, current) => sum + current, 0)
getTotalPlayerCount() {
return this.serverRegistry
.getServerRegistrations()
.map((serverRegistration) => serverRegistration.playerCount)
.reduce((sum, current) => sum + current, 0);
}
addServer = (serverId, payload, timestampPoints) => {
@ -102,51 +107,61 @@ export class App {
// result = undefined
// error = defined with "Waiting" description
// info = safely defined with configured data
const serverRegistration = this.serverRegistry.createServerRegistration(serverId)
const serverRegistration =
this.serverRegistry.createServerRegistration(serverId);
serverRegistration.initServerStatus(payload)
serverRegistration.initServerStatus(payload);
// playerCountHistory is only defined when the backend has previous ping data
// undefined playerCountHistory means this is a placeholder ping generated by the backend
if (typeof payload.playerCountHistory !== 'undefined') {
if (typeof payload.playerCountHistory !== "undefined") {
// Push the historical data into the graph
// This will trim and format the data so it is ready for the graph to render once init
serverRegistration.addGraphPoints(payload.playerCountHistory, timestampPoints)
serverRegistration.addGraphPoints(
payload.playerCountHistory,
timestampPoints
);
// Set initial playerCount to the payload's value
// This will always exist since it is explicitly generated by the backend
// This is used for any post-add rendering of things like the percentageBar
serverRegistration.playerCount = payload.playerCount
serverRegistration.playerCount = payload.playerCount;
}
// Create the plot instance internally with the restructured and cleaned data
serverRegistration.buildPlotInstance()
serverRegistration.buildPlotInstance();
// Handle the last known state (if any) as an incoming update
// This triggers the main update pipeline and enables centralized update handling
serverRegistration.updateServerStatus(payload, this.publicConfig.minecraftVersions)
serverRegistration.updateServerStatus(
payload,
this.publicConfig.minecraftVersions
);
// Allow the ServerRegistration to bind any DOM events with app instance context
serverRegistration.initEventListeners()
}
serverRegistration.initEventListeners();
};
updateGlobalStats = () => {
// Only redraw when needed
// These operations are relatively cheap, but the site already does too much rendering
const totalPlayerCount = this.getTotalPlayerCount()
const totalPlayerCount = this.getTotalPlayerCount();
if (totalPlayerCount !== this._lastTotalPlayerCount) {
this._lastTotalPlayerCount = totalPlayerCount
document.getElementById('stat_totalPlayers').innerText = formatNumber(totalPlayerCount)
this._lastTotalPlayerCount = totalPlayerCount;
document.getElementById("stat_totalPlayers").innerText =
formatNumber(totalPlayerCount);
}
// Only redraw when needed
// These operations are relatively cheap, but the site already does too much rendering
const serverRegistrationCount = this.serverRegistry.getServerRegistrations().length
const serverRegistrationCount =
this.serverRegistry.getServerRegistrations().length;
if (serverRegistrationCount !== this._lastServerRegistrationCount) {
this._lastServerRegistrationCount = serverRegistrationCount
document.getElementById('stat_networks').innerText = serverRegistrationCount
this._lastServerRegistrationCount = serverRegistrationCount;
document.getElementById("stat_networks").innerText =
serverRegistrationCount;
}
}
};
}

View File

@ -1,69 +1,83 @@
export const FAVORITE_SERVERS_STORAGE_KEY = 'minetrack_favorite_servers'
export const FAVORITE_SERVERS_STORAGE_KEY = "minetrack_favorite_servers";
export class FavoritesManager {
constructor (app) {
this._app = app
constructor(app) {
this._app = app;
}
loadLocalStorage () {
if (typeof localStorage !== 'undefined') {
let serverNames = localStorage.getItem(FAVORITE_SERVERS_STORAGE_KEY)
loadLocalStorage() {
if (typeof localStorage !== "undefined") {
let serverNames = localStorage.getItem(FAVORITE_SERVERS_STORAGE_KEY);
if (serverNames) {
serverNames = JSON.parse(serverNames)
serverNames = JSON.parse(serverNames);
for (let i = 0; i < serverNames.length; i++) {
const serverRegistration = this._app.serverRegistry.getServerRegistration(serverNames[i])
const serverRegistration =
this._app.serverRegistry.getServerRegistration(serverNames[i]);
// The serverName may not exist in the backend configuration anymore
// Ensure serverRegistration is defined before mutating data or considering valid
if (serverRegistration) {
serverRegistration.isFavorite = true
serverRegistration.isFavorite = true;
// Update icon since by default it is unfavorited
document.getElementById(`favorite-toggle_${serverRegistration.serverId}`).setAttribute('class', this.getIconClass(serverRegistration.isFavorite))
document
.getElementById(`favorite-toggle_${serverRegistration.serverId}`)
.setAttribute(
"class",
this.getIconClass(serverRegistration.isFavorite)
);
}
}
}
}
}
updateLocalStorage () {
if (typeof localStorage !== 'undefined') {
updateLocalStorage() {
if (typeof localStorage !== "undefined") {
// Mutate the serverIds array into server names for storage use
const serverNames = this._app.serverRegistry.getServerRegistrations()
.filter(serverRegistration => serverRegistration.isFavorite)
.map(serverRegistration => serverRegistration.data.name)
const serverNames = this._app.serverRegistry
.getServerRegistrations()
.filter((serverRegistration) => serverRegistration.isFavorite)
.map((serverRegistration) => serverRegistration.data.name);
if (serverNames.length > 0) {
// Only save if the array contains data, otherwise clear the item
localStorage.setItem(FAVORITE_SERVERS_STORAGE_KEY, JSON.stringify(serverNames))
localStorage.setItem(
FAVORITE_SERVERS_STORAGE_KEY,
JSON.stringify(serverNames)
);
} else {
localStorage.removeItem(FAVORITE_SERVERS_STORAGE_KEY)
localStorage.removeItem(FAVORITE_SERVERS_STORAGE_KEY);
}
}
}
handleFavoriteButtonClick = (serverRegistration) => {
serverRegistration.isFavorite = !serverRegistration.isFavorite
serverRegistration.isFavorite = !serverRegistration.isFavorite;
// Update the displayed favorite icon
document.getElementById(`favorite-toggle_${serverRegistration.serverId}`).setAttribute('class', this.getIconClass(serverRegistration.isFavorite))
document
.getElementById(`favorite-toggle_${serverRegistration.serverId}`)
.setAttribute("class", this.getIconClass(serverRegistration.isFavorite));
// Request the app controller instantly re-sort the server listing
// This handles the favorite sorting logic internally
this._app.sortController.sortServers()
this._app.sortController.sortServers();
this._app.graphDisplayManager.handleServerIsFavoriteUpdate(serverRegistration)
this._app.graphDisplayManager.handleServerIsFavoriteUpdate(
serverRegistration
);
// Write an updated settings payload
this.updateLocalStorage()
}
this.updateLocalStorage();
};
getIconClass (isFavorite) {
getIconClass(isFavorite) {
if (isFavorite) {
return 'icon-star server-is-favorite'
return "icon-star server-is-favorite";
} else {
return 'icon-star-o server-is-not-favorite'
return "icon-star-o server-is-not-favorite";
}
}
}

View File

@ -1,80 +1,87 @@
import uPlot from 'uplot'
import uPlot from "uplot";
import { RelativeScale } from './scale'
import { RelativeScale } from "./scale";
import { formatNumber, formatTimestampSeconds } from './util'
import { uPlotTooltipPlugin } from './plugins'
import { uPlotTooltipPlugin } from "./plugins";
import { formatNumber, formatTimestampSeconds } from "./util";
import { FAVORITE_SERVERS_STORAGE_KEY } from './favorites'
import { FAVORITE_SERVERS_STORAGE_KEY } from "./favorites";
const HIDDEN_SERVERS_STORAGE_KEY = 'minetrack_hidden_servers'
const SHOW_FAVORITES_STORAGE_KEY = 'minetrack_show_favorites'
const HIDDEN_SERVERS_STORAGE_KEY = "minetrack_hidden_servers";
const SHOW_FAVORITES_STORAGE_KEY = "minetrack_show_favorites";
export class GraphDisplayManager {
constructor (app) {
this._app = app
this._graphData = []
this._graphTimestamps = []
this._hasLoadedSettings = false
this._initEventListenersOnce = false
this._showOnlyFavorites = false
constructor(app) {
this._app = app;
this._graphData = [];
this._graphTimestamps = [];
this._hasLoadedSettings = false;
this._initEventListenersOnce = false;
this._showOnlyFavorites = false;
}
addGraphPoint (timestamp, playerCounts) {
addGraphPoint(timestamp, playerCounts) {
if (!this._hasLoadedSettings) {
// _hasLoadedSettings is controlled by #setGraphData
// It will only be true once the context has been loaded and initial payload received
// #addGraphPoint should not be called prior to that since it means the data is racing
// and the application has received updates prior to the initial state
return
return;
}
// Calculate isZoomed before mutating graphData otherwise the indexed values
// are out of date and will always fail when compared to plotScaleX.min/max
const plotScaleX = this._plotInstance.scales.x
const isZoomed = plotScaleX.min > this._graphTimestamps[0] || plotScaleX.max < this._graphTimestamps[this._graphTimestamps.length - 1]
const plotScaleX = this._plotInstance.scales.x;
const isZoomed =
plotScaleX.min > this._graphTimestamps[0] ||
plotScaleX.max < this._graphTimestamps[this._graphTimestamps.length - 1];
this._graphTimestamps.push(timestamp)
this._graphTimestamps.push(timestamp);
for (let i = 0; i < playerCounts.length; i++) {
this._graphData[i].push(playerCounts[i])
this._graphData[i].push(playerCounts[i]);
}
// Trim all data arrays to only the relevant portion
// This keeps it in sync with backend data structures
const graphMaxLength = this._app.publicConfig.graphMaxLength
const graphMaxLength = this._app.publicConfig.graphMaxLength;
if (this._graphTimestamps.length > graphMaxLength) {
this._graphTimestamps.splice(0, this._graphTimestamps.length - graphMaxLength)
this._graphTimestamps.splice(
0,
this._graphTimestamps.length - graphMaxLength
);
}
for (const series of this._graphData) {
if (series.length > graphMaxLength) {
series.splice(0, series.length - graphMaxLength)
series.splice(0, series.length - graphMaxLength);
}
}
// Avoid redrawing the plot when zoomed
this._plotInstance.setData(this.getGraphData(), !isZoomed)
this._plotInstance.setData(this.getGraphData(), !isZoomed);
}
loadLocalStorage () {
if (typeof localStorage !== 'undefined') {
const showOnlyFavorites = localStorage.getItem(SHOW_FAVORITES_STORAGE_KEY)
loadLocalStorage() {
if (typeof localStorage !== "undefined") {
const showOnlyFavorites = localStorage.getItem(
SHOW_FAVORITES_STORAGE_KEY
);
if (showOnlyFavorites) {
this._showOnlyFavorites = true
this._showOnlyFavorites = true;
}
// If only favorites mode is active, use the stored favorite servers data instead
let serverNames
let serverNames;
if (this._showOnlyFavorites) {
serverNames = localStorage.getItem(FAVORITE_SERVERS_STORAGE_KEY)
serverNames = localStorage.getItem(FAVORITE_SERVERS_STORAGE_KEY);
} else {
serverNames = localStorage.getItem(HIDDEN_SERVERS_STORAGE_KEY)
serverNames = localStorage.getItem(HIDDEN_SERVERS_STORAGE_KEY);
}
if (serverNames) {
serverNames = JSON.parse(serverNames)
serverNames = JSON.parse(serverNames);
// Iterate over all active serverRegistrations
// This merges saved state with current state to prevent desyncs
@ -83,384 +90,429 @@ export class GraphDisplayManager {
// OR, if it is NOT contains within HIDDEN_SERVERS_STORAGE_KEY
// Checks between FAVORITE/HIDDEN keys are mutually exclusive
if (this._showOnlyFavorites) {
serverRegistration.isVisible = serverNames.indexOf(serverRegistration.data.name) >= 0
serverRegistration.isVisible =
serverNames.indexOf(serverRegistration.data.name) >= 0;
} else {
serverRegistration.isVisible = serverNames.indexOf(serverRegistration.data.name) < 0
serverRegistration.isVisible =
serverNames.indexOf(serverRegistration.data.name) < 0;
}
}
}
}
}
updateLocalStorage () {
if (typeof localStorage !== 'undefined') {
updateLocalStorage() {
if (typeof localStorage !== "undefined") {
// Mutate the serverIds array into server names for storage use
const serverNames = this._app.serverRegistry.getServerRegistrations()
.filter(serverRegistration => !serverRegistration.isVisible)
.map(serverRegistration => serverRegistration.data.name)
const serverNames = this._app.serverRegistry
.getServerRegistrations()
.filter((serverRegistration) => !serverRegistration.isVisible)
.map((serverRegistration) => serverRegistration.data.name);
// Only store if the array contains data, otherwise clear the item
// If showOnlyFavorites is true, do NOT store serverNames since the state will be auto managed instead
if (serverNames.length > 0 && !this._showOnlyFavorites) {
localStorage.setItem(HIDDEN_SERVERS_STORAGE_KEY, JSON.stringify(serverNames))
localStorage.setItem(
HIDDEN_SERVERS_STORAGE_KEY,
JSON.stringify(serverNames)
);
} else {
localStorage.removeItem(HIDDEN_SERVERS_STORAGE_KEY)
localStorage.removeItem(HIDDEN_SERVERS_STORAGE_KEY);
}
// Only store SHOW_FAVORITES_STORAGE_KEY if true
if (this._showOnlyFavorites) {
localStorage.setItem(SHOW_FAVORITES_STORAGE_KEY, true)
localStorage.setItem(SHOW_FAVORITES_STORAGE_KEY, true);
} else {
localStorage.removeItem(SHOW_FAVORITES_STORAGE_KEY)
localStorage.removeItem(SHOW_FAVORITES_STORAGE_KEY);
}
}
}
getVisibleGraphData () {
return this._app.serverRegistry.getServerRegistrations()
.filter(serverRegistration => serverRegistration.isVisible)
.map(serverRegistration => this._graphData[serverRegistration.serverId])
getVisibleGraphData() {
return this._app.serverRegistry
.getServerRegistrations()
.filter((serverRegistration) => serverRegistration.isVisible)
.map(
(serverRegistration) => this._graphData[serverRegistration.serverId]
);
}
getPlotSize () {
getPlotSize() {
return {
width: Math.max(window.innerWidth, 800) * 0.9,
height: 400
height: 400,
};
}
getGraphData() {
return [this._graphTimestamps, ...this._graphData];
}
getGraphDataPoint(serverId, index) {
const graphData = this._graphData[serverId];
if (
graphData &&
index < graphData.length &&
typeof graphData[index] === "number"
) {
return graphData[index];
}
}
getGraphData () {
return [
this._graphTimestamps,
...this._graphData
]
}
getClosestPlotSeriesIndex(idx) {
let closestSeriesIndex = -1;
let closestSeriesDist = Number.MAX_VALUE;
getGraphDataPoint (serverId, index) {
const graphData = this._graphData[serverId]
if (graphData && index < graphData.length && typeof graphData[index] === 'number') {
return graphData[index]
}
}
getClosestPlotSeriesIndex (idx) {
let closestSeriesIndex = -1
let closestSeriesDist = Number.MAX_VALUE
const plotHeight = this._plotInstance.bbox.height / devicePixelRatio
const plotHeight = this._plotInstance.bbox.height / devicePixelRatio;
for (let i = 1; i < this._plotInstance.series.length; i++) {
const series = this._plotInstance.series[i]
const series = this._plotInstance.series[i];
if (!series.show) {
continue
continue;
}
const point = this._plotInstance.data[i][idx]
const point = this._plotInstance.data[i][idx];
if (typeof point === 'number') {
const scale = this._plotInstance.scales[series.scale]
const posY = (1 - ((point - scale.min) / (scale.max - scale.min))) * plotHeight
if (typeof point === "number") {
const scale = this._plotInstance.scales[series.scale];
const posY =
(1 - (point - scale.min) / (scale.max - scale.min)) * plotHeight;
const dist = Math.abs(posY - this._plotInstance.cursor.top)
const dist = Math.abs(posY - this._plotInstance.cursor.top);
if (dist < closestSeriesDist) {
closestSeriesIndex = i
closestSeriesDist = dist
closestSeriesIndex = i;
closestSeriesDist = dist;
}
}
}
return closestSeriesIndex
return closestSeriesIndex;
}
buildPlotInstance (timestamps, data) {
buildPlotInstance(timestamps, data) {
// Lazy load settings from localStorage, if any and if enabled
if (!this._hasLoadedSettings) {
this._hasLoadedSettings = true
this._hasLoadedSettings = true;
this.loadLocalStorage()
this.loadLocalStorage();
}
for (const playerCounts of data) {
// Each playerCounts value corresponds to a ServerRegistration
// Require each array is the length of timestamps, if not, pad at the start with null values to fit to length
// This ensures newer ServerRegistrations do not left align due to a lower length
const lengthDiff = timestamps.length - playerCounts.length
const lengthDiff = timestamps.length - playerCounts.length;
if (lengthDiff > 0) {
const padding = Array(lengthDiff).fill(null)
const padding = Array(lengthDiff).fill(null);
playerCounts.unshift(...padding)
playerCounts.unshift(...padding);
}
}
this._graphTimestamps = timestamps
this._graphData = data
this._graphTimestamps = timestamps;
this._graphData = data;
const series = this._app.serverRegistry.getServerRegistrations().map(serverRegistration => {
return {
stroke: serverRegistration.data.color,
width: 2,
value: (_, raw) => `${formatNumber(raw)} Players`,
show: serverRegistration.isVisible,
spanGaps: true,
points: {
show: false
}
}
})
const series = this._app.serverRegistry
.getServerRegistrations()
.map((serverRegistration) => {
return {
stroke: serverRegistration.data.color,
width: 2,
value: (_, raw) => `${formatNumber(raw)} Players`,
show: serverRegistration.isVisible,
spanGaps: true,
points: {
show: false,
},
};
});
const tickCount = 10
const maxFactor = 4
const tickCount = 10;
const maxFactor = 4;
// eslint-disable-next-line new-cap
this._plotInstance = new uPlot({
plugins: [
uPlotTooltipPlugin((pos, idx) => {
if (pos) {
const closestSeriesIndex = this.getClosestPlotSeriesIndex(idx)
this._plotInstance = new uPlot(
{
plugins: [
uPlotTooltipPlugin((pos, idx) => {
if (pos) {
const closestSeriesIndex = this.getClosestPlotSeriesIndex(idx);
const text = this._app.serverRegistry.getServerRegistrations()
.filter(serverRegistration => serverRegistration.isVisible)
.sort((a, b) => {
if (a.isFavorite !== b.isFavorite) {
return a.isFavorite ? -1 : 1
} else {
return a.data.name.localeCompare(b.data.name)
}
})
.map(serverRegistration => {
const point = this.getGraphDataPoint(serverRegistration.serverId, idx)
const text =
this._app.serverRegistry
.getServerRegistrations()
.filter((serverRegistration) => serverRegistration.isVisible)
.sort((a, b) => {
if (a.isFavorite !== b.isFavorite) {
return a.isFavorite ? -1 : 1;
} else {
return a.data.name.localeCompare(b.data.name);
}
})
.map((serverRegistration) => {
const point = this.getGraphDataPoint(
serverRegistration.serverId,
idx
);
let serverName = serverRegistration.data.name
if (closestSeriesIndex === serverRegistration.getGraphDataIndex()) {
serverName = `<strong>${serverName}</strong>`
}
if (serverRegistration.isFavorite) {
serverName = `<span class="${this._app.favoritesManager.getIconClass(true)}"></span> ${serverName}`
}
let serverName = serverRegistration.data.name;
if (
closestSeriesIndex ===
serverRegistration.getGraphDataIndex()
) {
serverName = `<strong>${serverName}</strong>`;
}
if (serverRegistration.isFavorite) {
serverName = `<span class="${this._app.favoritesManager.getIconClass(
true
)}"></span> ${serverName}`;
}
return `${serverName}: ${formatNumber(point)}`
}).join('<br>') + `<br><br><strong>${formatTimestampSeconds(this._graphTimestamps[idx])}</strong>`
return `${serverName}: ${formatNumber(point)}`;
})
.join("<br>") +
`<br><br><strong>${formatTimestampSeconds(
this._graphTimestamps[idx]
)}</strong>`;
this._app.tooltip.set(pos.left, pos.top, 10, 10, text)
} else {
this._app.tooltip.hide()
}
})
],
...this.getPlotSize(),
cursor: {
y: false
},
series: [
{
this._app.tooltip.set(pos.left, pos.top, 10, 10, text);
} else {
this._app.tooltip.hide();
}
}),
],
...this.getPlotSize(),
cursor: {
y: false,
},
...series
],
axes: [
{
font: '14px "Open Sans", sans-serif',
stroke: '#FFF',
grid: {
show: false
series: [{}, ...series],
axes: [
{
font: '14px "Open Sans", sans-serif',
stroke: "#FFF",
grid: {
show: false,
},
space: 60,
},
{
font: '14px "Open Sans", sans-serif',
stroke: "#FFF",
size: 65,
grid: {
stroke: "#333",
width: 1,
},
split: () => {
const visibleGraphData = this.getVisibleGraphData();
const { scaledMax, scale } = RelativeScale.scaleMatrix(
visibleGraphData,
tickCount,
maxFactor
);
const ticks = RelativeScale.generateTicks(0, scaledMax, scale);
return ticks;
},
},
],
scales: {
y: {
auto: false,
range: () => {
const visibleGraphData = this.getVisibleGraphData();
const { scaledMin, scaledMax } = RelativeScale.scaleMatrix(
visibleGraphData,
tickCount,
maxFactor
);
return [scaledMin, scaledMax];
},
},
space: 60
},
{
font: '14px "Open Sans", sans-serif',
stroke: '#FFF',
size: 65,
grid: {
stroke: '#333',
width: 1
},
split: () => {
const visibleGraphData = this.getVisibleGraphData()
const { scaledMax, scale } = RelativeScale.scaleMatrix(visibleGraphData, tickCount, maxFactor)
const ticks = RelativeScale.generateTicks(0, scaledMax, scale)
return ticks
}
}
],
scales: {
y: {
auto: false,
range: () => {
const visibleGraphData = this.getVisibleGraphData()
const { scaledMin, scaledMax } = RelativeScale.scaleMatrix(visibleGraphData, tickCount, maxFactor)
return [scaledMin, scaledMax]
}
}
legend: {
show: false,
},
},
legend: {
show: false
}
}, this.getGraphData(), document.getElementById('big-graph'))
this.getGraphData(),
document.getElementById("big-graph")
);
// Show the settings-toggle element
document.getElementById('settings-toggle').style.display = 'inline-block'
document.getElementById("settings-toggle").style.display = "inline-block";
}
redraw = () => {
// Use drawing as a hint to update settings
// This may cause unnessecary localStorage updates, but its a rare and harmless outcome
this.updateLocalStorage()
this.updateLocalStorage();
// Copy application state into the series data used by uPlot
for (const serverRegistration of this._app.serverRegistry.getServerRegistrations()) {
this._plotInstance.series[serverRegistration.getGraphDataIndex()].show = serverRegistration.isVisible
this._plotInstance.series[serverRegistration.getGraphDataIndex()].show =
serverRegistration.isVisible;
}
this._plotInstance.redraw()
}
this._plotInstance.redraw();
};
requestResize () {
requestResize() {
// Only resize when _plotInstance is defined
// Set a timeout to resize after resize events have not been fired for some duration of time
// This prevents burning CPU time for multiple, rapid resize events
if (this._plotInstance) {
if (this._resizeRequestTimeout) {
clearTimeout(this._resizeRequestTimeout)
clearTimeout(this._resizeRequestTimeout);
}
// Schedule new delayed resize call
// This can be cancelled by #requestResize, #resize and #reset
this._resizeRequestTimeout = setTimeout(this.resize, 200)
this._resizeRequestTimeout = setTimeout(this.resize, 200);
}
}
resize = () => {
this._plotInstance.setSize(this.getPlotSize())
this._plotInstance.setSize(this.getPlotSize());
// undefine value so #clearTimeout is not called
// This is safe even if #resize is manually called since it removes the pending work
if (this._resizeRequestTimeout) {
clearTimeout(this._resizeRequestTimeout)
clearTimeout(this._resizeRequestTimeout);
}
this._resizeRequestTimeout = undefined
}
this._resizeRequestTimeout = undefined;
};
initEventListeners () {
initEventListeners() {
if (!this._initEventListenersOnce) {
this._initEventListenersOnce = true
this._initEventListenersOnce = true;
// These listeners should only be init once since they attach to persistent elements
document.getElementById('settings-toggle').addEventListener('click', this.handleSettingsToggle, false)
document
.getElementById("settings-toggle")
.addEventListener("click", this.handleSettingsToggle, false);
document.querySelectorAll('.graph-controls-show').forEach((element) => {
element.addEventListener('click', this.handleShowButtonClick, false)
})
document.querySelectorAll(".graph-controls-show").forEach((element) => {
element.addEventListener("click", this.handleShowButtonClick, false);
});
}
// These listeners should be bound each #initEventListeners call since they are for newly created elements
document.querySelectorAll('.graph-control').forEach((element) => {
element.addEventListener('click', this.handleServerButtonClick, false)
})
document.querySelectorAll(".graph-control").forEach((element) => {
element.addEventListener("click", this.handleServerButtonClick, false);
});
}
handleServerButtonClick = (event) => {
const serverId = parseInt(event.target.getAttribute('minetrack-server-id'))
const serverRegistration = this._app.serverRegistry.getServerRegistration(serverId)
const serverId = parseInt(event.target.getAttribute("minetrack-server-id"));
const serverRegistration =
this._app.serverRegistry.getServerRegistration(serverId);
if (serverRegistration.isVisible !== event.target.checked) {
serverRegistration.isVisible = event.target.checked
serverRegistration.isVisible = event.target.checked;
// Any manual changes automatically disables "Only Favorites" mode
// Otherwise the auto management might overwrite their manual changes
this._showOnlyFavorites = false
this._showOnlyFavorites = false;
this.redraw()
this.redraw();
}
}
};
handleShowButtonClick = (event) => {
const showType = event.target.getAttribute('minetrack-show-type')
const showType = event.target.getAttribute("minetrack-show-type");
// If set to "Only Favorites", set internal state so that
// visible graphData is automatically updating when a ServerRegistration's #isVisible changes
// This is also saved and loaded by #loadLocalStorage & #updateLocalStorage
this._showOnlyFavorites = showType === 'favorites'
this._showOnlyFavorites = showType === "favorites";
let redraw = false
let redraw = false;
this._app.serverRegistry.getServerRegistrations().forEach(function (serverRegistration) {
let isVisible
if (showType === 'all') {
isVisible = true
} else if (showType === 'none') {
isVisible = false
} else if (showType === 'favorites') {
isVisible = serverRegistration.isFavorite
}
this._app.serverRegistry
.getServerRegistrations()
.forEach(function (serverRegistration) {
let isVisible;
if (showType === "all") {
isVisible = true;
} else if (showType === "none") {
isVisible = false;
} else if (showType === "favorites") {
isVisible = serverRegistration.isFavorite;
}
if (serverRegistration.isVisible !== isVisible) {
serverRegistration.isVisible = isVisible
redraw = true
}
})
if (serverRegistration.isVisible !== isVisible) {
serverRegistration.isVisible = isVisible;
redraw = true;
}
});
if (redraw) {
this.redraw()
this.updateCheckboxes()
this.redraw();
this.updateCheckboxes();
}
}
};
handleSettingsToggle = () => {
const element = document.getElementById('big-graph-controls-drawer')
const element = document.getElementById("big-graph-controls-drawer");
if (element.style.display !== 'block') {
element.style.display = 'block'
if (element.style.display !== "block") {
element.style.display = "block";
} else {
element.style.display = 'none'
element.style.display = "none";
}
}
};
handleServerIsFavoriteUpdate = (serverRegistration) => {
// When in "Only Favorites" mode, visibility is dependent on favorite status
// Redraw and update elements as needed
if (this._showOnlyFavorites && serverRegistration.isVisible !== serverRegistration.isFavorite) {
serverRegistration.isVisible = serverRegistration.isFavorite
if (
this._showOnlyFavorites &&
serverRegistration.isVisible !== serverRegistration.isFavorite
) {
serverRegistration.isVisible = serverRegistration.isFavorite;
this.redraw()
this.updateCheckboxes()
this.redraw();
this.updateCheckboxes();
}
};
updateCheckboxes() {
document.querySelectorAll(".graph-control").forEach((checkbox) => {
const serverId = parseInt(checkbox.getAttribute("minetrack-server-id"));
const serverRegistration =
this._app.serverRegistry.getServerRegistration(serverId);
checkbox.checked = serverRegistration.isVisible;
});
}
updateCheckboxes () {
document.querySelectorAll('.graph-control').forEach((checkbox) => {
const serverId = parseInt(checkbox.getAttribute('minetrack-server-id'))
const serverRegistration = this._app.serverRegistry.getServerRegistration(serverId)
checkbox.checked = serverRegistration.isVisible
})
}
reset () {
reset() {
// Destroy graphs and unload references
// uPlot#destroy handles listener de-registration, DOM reset, etc
if (this._plotInstance) {
this._plotInstance.destroy()
this._plotInstance = undefined
this._plotInstance.destroy();
this._plotInstance = undefined;
}
this._graphTimestamps = []
this._graphData = []
this._hasLoadedSettings = false
this._graphTimestamps = [];
this._graphData = [];
this._hasLoadedSettings = false;
// Fire #clearTimeout if the timeout is currently defined
if (this._resizeRequestTimeout) {
clearTimeout(this._resizeRequestTimeout)
clearTimeout(this._resizeRequestTimeout);
this._resizeRequestTimeout = undefined
this._resizeRequestTimeout = undefined;
}
// Reset modified DOM structures
document.getElementById('big-graph-checkboxes').innerHTML = ''
document.getElementById('big-graph-controls').style.display = 'none'
document.getElementById("big-graph-checkboxes").innerHTML = "";
document.getElementById("big-graph-controls").style.display = "none";
document.getElementById('settings-toggle').style.display = 'none'
document.getElementById("settings-toggle").style.display = "none";
}
}

View File

@ -1,14 +1,22 @@
import { App } from './app'
import { App } from "./app";
const app = new App()
const app = new App();
document.addEventListener('DOMContentLoaded', () => {
app.init()
document.addEventListener(
"DOMContentLoaded",
() => {
app.init();
window.addEventListener('resize', function () {
app.percentageBar.redraw()
window.addEventListener(
"resize",
function () {
app.percentageBar.redraw();
// Delegate to GraphDisplayManager which can check if the resize is necessary
app.graphDisplayManager.requestResize()
}, false)
}, false)
// Delegate to GraphDisplayManager which can check if the resize is necessary
app.graphDisplayManager.requestResize();
},
false
);
},
false
);

View File

@ -1,75 +1,99 @@
import { formatNumber, formatPercent } from './util'
import { formatNumber, formatPercent } from "./util";
export class PercentageBar {
constructor (app) {
this._app = app
this._parent = document.getElementById('perc-bar')
constructor(app) {
this._app = app;
this._parent = document.getElementById("perc-bar");
}
redraw = () => {
const serverRegistrations = this._app.serverRegistry.getServerRegistrations().sort(function (a, b) {
return a.playerCount - b.playerCount
})
const serverRegistrations = this._app.serverRegistry
.getServerRegistrations()
.sort(function (a, b) {
return a.playerCount - b.playerCount;
});
const totalPlayers = this._app.getTotalPlayerCount()
const totalPlayers = this._app.getTotalPlayerCount();
let leftPadding = 0
let leftPadding = 0;
for (const serverRegistration of serverRegistrations) {
const width = Math.round((serverRegistration.playerCount / totalPlayers) * this._parent.offsetWidth)
const width = Math.round(
(serverRegistration.playerCount / totalPlayers) *
this._parent.offsetWidth
);
// Update position/width
// leftPadding is a sum of previous iterations width value
const div = document.getElementById(`perc-bar-part_${serverRegistration.serverId}`) || this.createPart(serverRegistration)
const div =
document.getElementById(
`perc-bar-part_${serverRegistration.serverId}`
) || this.createPart(serverRegistration);
const widthPixels = `${width}px`
const leftPaddingPixels = `${leftPadding}px`
const widthPixels = `${width}px`;
const leftPaddingPixels = `${leftPadding}px`;
// Only redraw if needed
if (div.style.width !== widthPixels || div.style.left !== leftPaddingPixels) {
div.style.width = widthPixels
div.style.left = leftPaddingPixels
if (
div.style.width !== widthPixels ||
div.style.left !== leftPaddingPixels
) {
div.style.width = widthPixels;
div.style.left = leftPaddingPixels;
}
leftPadding += width
leftPadding += width;
}
}
};
createPart (serverRegistration) {
const div = document.createElement('div')
createPart(serverRegistration) {
const div = document.createElement("div");
div.id = `perc-bar-part_${serverRegistration.serverId}`
div.style.background = serverRegistration.data.color
div.id = `perc-bar-part_${serverRegistration.serverId}`;
div.style.background = serverRegistration.data.color;
div.setAttribute('class', 'perc-bar-part')
div.setAttribute('minetrack-server-id', serverRegistration.serverId)
div.setAttribute("class", "perc-bar-part");
div.setAttribute("minetrack-server-id", serverRegistration.serverId);
this._parent.appendChild(div)
this._parent.appendChild(div);
// Define events once during creation
div.addEventListener('mouseover', this.handleMouseOver, false)
div.addEventListener('mouseout', this.handleMouseOut, false)
div.addEventListener("mouseover", this.handleMouseOver, false);
div.addEventListener("mouseout", this.handleMouseOut, false);
return div
return div;
}
handleMouseOver = (event) => {
const serverId = parseInt(event.target.getAttribute('minetrack-server-id'))
const serverRegistration = this._app.serverRegistry.getServerRegistration(serverId)
const serverId = parseInt(event.target.getAttribute("minetrack-server-id"));
const serverRegistration =
this._app.serverRegistry.getServerRegistration(serverId);
this._app.tooltip.set(event.target.offsetLeft, event.target.offsetTop, 10, this._parent.offsetTop + this._parent.offsetHeight + 10,
`${typeof serverRegistration.rankIndex !== 'undefined' ? `#${serverRegistration.rankIndex + 1} ` : ''}
this._app.tooltip.set(
event.target.offsetLeft,
event.target.offsetTop,
10,
this._parent.offsetTop + this._parent.offsetHeight + 10,
`${
typeof serverRegistration.rankIndex !== "undefined"
? `#${serverRegistration.rankIndex + 1} `
: ""
}
${serverRegistration.data.name}<br>
${formatNumber(serverRegistration.playerCount)} Players<br>
<strong>${formatPercent(serverRegistration.playerCount, this._app.getTotalPlayerCount())}</strong>`)
}
<strong>${formatPercent(
serverRegistration.playerCount,
this._app.getTotalPlayerCount()
)}</strong>`
);
};
handleMouseOut = () => {
this._app.tooltip.hide()
}
this._app.tooltip.hide();
};
reset () {
reset() {
// Reset modified DOM elements
this._parent.innerHTML = ''
this._parent.innerHTML = "";
}
}

View File

@ -1,28 +1,31 @@
export function uPlotTooltipPlugin (onHover) {
let element
export function uPlotTooltipPlugin(onHover) {
let element;
return {
hooks: {
init: u => {
element = u.root.querySelector('.over')
init: (u) => {
element = u.root.querySelector(".over");
element.onmouseenter = () => onHover()
element.onmouseleave = () => onHover()
element.onmouseenter = () => onHover();
element.onmouseleave = () => onHover();
},
setCursor: u => {
const { left, top, idx } = u.cursor
setCursor: (u) => {
const { left, top, idx } = u.cursor;
if (idx === null) {
onHover()
onHover();
} else {
const bounds = element.getBoundingClientRect()
const bounds = element.getBoundingClientRect();
onHover({
left: bounds.left + left + window.pageXOffset,
top: bounds.top + top + window.pageYOffset
}, idx)
onHover(
{
left: bounds.left + left + window.pageXOffset,
top: bounds.top + top + window.pageYOffset,
},
idx
);
}
}
}
}
},
},
};
}

View File

@ -1,88 +1,91 @@
export class RelativeScale {
static scale (data, tickCount, maxFactor) {
const { min, max } = RelativeScale.calculateBounds(data)
static scale(data, tickCount, maxFactor) {
const { min, max } = RelativeScale.calculateBounds(data);
let factor = 1
let factor = 1;
while (true) {
const scale = Math.pow(10, factor)
const scale = Math.pow(10, factor);
const scaledMin = min - (min % scale)
let scaledMax = max + (max % scale === 0 ? 0 : scale - (max % scale))
const scaledMin = min - (min % scale);
let scaledMax = max + (max % scale === 0 ? 0 : scale - (max % scale));
// Prevent min/max from being equal (and generating 0 ticks)
// This happens when all data points are products of scale value
if (scaledMin === scaledMax) {
scaledMax += scale
scaledMax += scale;
}
const ticks = (scaledMax - scaledMin) / scale
const ticks = (scaledMax - scaledMin) / scale;
if (ticks <= tickCount || (typeof maxFactor === 'number' && factor === maxFactor)) {
if (
ticks <= tickCount ||
(typeof maxFactor === "number" && factor === maxFactor)
) {
return {
scaledMin,
scaledMax,
scale
}
scale,
};
} else {
// Too many steps between min/max, increase factor and try again
factor++
factor++;
}
}
}
static scaleMatrix (data, tickCount, maxFactor) {
const nonNullData = data.flat().filter((val) => val !== null)
static scaleMatrix(data, tickCount, maxFactor) {
const nonNullData = data.flat().filter((val) => val !== null);
// when used with the spread operator large nonNullData/data arrays can reach the max call stack size
// use reduce calls to safely determine min/max values for any size of array
// https://stackoverflow.com/questions/63705432/maximum-call-stack-size-exceeded-when-using-the-dots-operator/63706516#63706516
const max = nonNullData.reduce((a, b) => {
return Math.max(a, b)
}, Number.NEGATIVE_INFINITY)
return Math.max(a, b);
}, Number.NEGATIVE_INFINITY);
return RelativeScale.scale(
[0, RelativeScale.isFiniteOrZero(max)],
tickCount,
maxFactor
)
);
}
static generateTicks (min, max, step) {
const ticks = []
static generateTicks(min, max, step) {
const ticks = [];
for (let i = min; i <= max; i += step) {
ticks.push(i)
ticks.push(i);
}
return ticks
return ticks;
}
static calculateBounds (data) {
static calculateBounds(data) {
if (data.length === 0) {
return {
min: 0,
max: 0
}
max: 0,
};
} else {
const nonNullData = data.filter((val) => val !== null)
const nonNullData = data.filter((val) => val !== null);
// when used with the spread operator large nonNullData/data arrays can reach the max call stack size
// use reduce calls to safely determine min/max values for any size of array
// https://stackoverflow.com/questions/63705432/maximum-call-stack-size-exceeded-when-using-the-dots-operator/63706516#63706516
const min = nonNullData.reduce((a, b) => {
return Math.min(a, b)
}, Number.POSITIVE_INFINITY)
return Math.min(a, b);
}, Number.POSITIVE_INFINITY);
const max = nonNullData.reduce((a, b) => {
return Math.max(a, b)
}, Number.NEGATIVE_INFINITY)
return Math.max(a, b);
}, Number.NEGATIVE_INFINITY);
return {
min: RelativeScale.isFiniteOrZero(min),
max: RelativeScale.isFiniteOrZero(max)
}
max: RelativeScale.isFiniteOrZero(max),
};
}
}
static isFiniteOrZero (val) {
return Number.isFinite(val) ? val : 0
static isFiniteOrZero(val) {
return Number.isFinite(val) ? val : 0;
}
}

View File

@ -1,319 +1,396 @@
import uPlot from 'uplot'
import uPlot from "uplot";
import { RelativeScale } from './scale'
import { RelativeScale } from "./scale";
import { formatNumber, formatTimestampSeconds, formatDate, formatMinecraftServerAddress, formatMinecraftVersions } from './util'
import { uPlotTooltipPlugin } from './plugins'
import { uPlotTooltipPlugin } from "./plugins";
import {
formatDate,
formatMinecraftServerAddress,
formatMinecraftVersions,
formatNumber,
formatTimestampSeconds,
} from "./util";
import MISSING_FAVICON from 'url:../images/missing_favicon.svg'
import MISSING_FAVICON from "url:../images/missing_favicon.svg";
export class ServerRegistry {
constructor (app) {
this._app = app
this._serverIdsByName = []
this._serverDataById = []
this._registeredServers = []
constructor(app) {
this._app = app;
this._serverIdsByName = [];
this._serverDataById = [];
this._registeredServers = [];
}
assignServers (servers) {
assignServers(servers) {
for (let i = 0; i < servers.length; i++) {
const data = servers[i]
this._serverIdsByName[data.name] = i
this._serverDataById[i] = data
const data = servers[i];
this._serverIdsByName[data.name] = i;
this._serverDataById[i] = data;
}
}
createServerRegistration (serverId) {
const serverData = this._serverDataById[serverId]
const serverRegistration = new ServerRegistration(this._app, serverId, serverData)
this._registeredServers[serverId] = serverRegistration
return serverRegistration
createServerRegistration(serverId) {
const serverData = this._serverDataById[serverId];
const serverRegistration = new ServerRegistration(
this._app,
serverId,
serverData
);
this._registeredServers[serverId] = serverRegistration;
return serverRegistration;
}
getServerRegistration (serverKey) {
if (typeof serverKey === 'string') {
const serverId = this._serverIdsByName[serverKey]
return this._registeredServers[serverId]
} else if (typeof serverKey === 'number') {
return this._registeredServers[serverKey]
getServerRegistration(serverKey) {
if (typeof serverKey === "string") {
const serverId = this._serverIdsByName[serverKey];
return this._registeredServers[serverId];
} else if (typeof serverKey === "number") {
return this._registeredServers[serverKey];
}
}
getServerRegistrations = () => Object.values(this._registeredServers)
getServerRegistrations = () => Object.values(this._registeredServers);
reset () {
this._serverIdsByName = []
this._serverDataById = []
this._registeredServers = []
reset() {
this._serverIdsByName = [];
this._serverDataById = [];
this._registeredServers = [];
// Reset modified DOM structures
document.getElementById('server-list').innerHTML = ''
document.getElementById("server-list").innerHTML = "";
}
}
export class ServerRegistration {
playerCount = 0
isVisible = true
isFavorite = false
rankIndex
lastRecordData
lastPeakData
playerCount = 0;
isVisible = true;
isFavorite = false;
rankIndex;
lastRecordData;
lastPeakData;
constructor (app, serverId, data) {
this._app = app
this.serverId = serverId
this.data = data
this._graphData = [[], []]
this._failedSequentialPings = 0
constructor(app, serverId, data) {
this._app = app;
this.serverId = serverId;
this.data = data;
this._graphData = [[], []];
this._failedSequentialPings = 0;
}
getGraphDataIndex () {
return this.serverId + 1
getGraphDataIndex() {
return this.serverId + 1;
}
addGraphPoints (points, timestampPoints) {
this._graphData = [
timestampPoints.slice(),
points
]
addGraphPoints(points, timestampPoints) {
this._graphData = [timestampPoints.slice(), points];
}
buildPlotInstance () {
const tickCount = 4
buildPlotInstance() {
const tickCount = 4;
// eslint-disable-next-line new-cap
this._plotInstance = new uPlot({
plugins: [
uPlotTooltipPlugin((pos, id) => {
if (pos) {
const playerCount = this._graphData[1][id]
this._plotInstance = new uPlot(
{
plugins: [
uPlotTooltipPlugin((pos, id) => {
if (pos) {
const playerCount = this._graphData[1][id];
if (typeof playerCount !== 'number') {
this._app.tooltip.hide()
if (typeof playerCount !== "number") {
this._app.tooltip.hide();
} else {
this._app.tooltip.set(
pos.left,
pos.top,
10,
10,
`${formatNumber(
playerCount
)} Players<br>${formatTimestampSeconds(
this._graphData[0][id]
)}`
);
}
} else {
this._app.tooltip.set(pos.left, pos.top, 10, 10, `${formatNumber(playerCount)} Players<br>${formatTimestampSeconds(this._graphData[0][id])}`)
this._app.tooltip.hide();
}
} else {
this._app.tooltip.hide()
}
})
],
height: 100,
width: 400,
cursor: {
y: false,
drag: {
setScale: false,
x: false,
y: false
},
sync: {
key: 'minetrack-server',
setSeries: true
}
},
series: [
{},
{
stroke: '#E9E581',
width: 2,
value: (_, raw) => `${formatNumber(raw)} Players`,
spanGaps: true,
points: {
show: false
}
}
],
axes: [
{
show: false
},
{
ticks: {
show: false
}),
],
height: 100,
width: 400,
cursor: {
y: false,
drag: {
setScale: false,
x: false,
y: false,
},
font: '14px "Open Sans", sans-serif',
stroke: '#A3A3A3',
size: 55,
grid: {
stroke: '#333',
width: 1
sync: {
key: "minetrack-server",
setSeries: true,
},
split: () => {
const { scaledMin, scaledMax, scale } = RelativeScale.scale(this._graphData[1], tickCount)
const ticks = RelativeScale.generateTicks(scaledMin, scaledMax, scale)
return ticks
}
}
],
scales: {
y: {
auto: false,
range: () => {
const { scaledMin, scaledMax } = RelativeScale.scale(this._graphData[1], tickCount)
return [scaledMin, scaledMax]
}
}
},
series: [
{},
{
stroke: "#E9E581",
width: 2,
value: (_, raw) => `${formatNumber(raw)} Players`,
spanGaps: true,
points: {
show: false,
},
},
],
axes: [
{
show: false,
},
{
ticks: {
show: false,
},
font: '14px "Open Sans", sans-serif',
stroke: "#A3A3A3",
size: 55,
grid: {
stroke: "#333",
width: 1,
},
split: () => {
const { scaledMin, scaledMax, scale } = RelativeScale.scale(
this._graphData[1],
tickCount
);
const ticks = RelativeScale.generateTicks(
scaledMin,
scaledMax,
scale
);
return ticks;
},
},
],
scales: {
y: {
auto: false,
range: () => {
const { scaledMin, scaledMax } = RelativeScale.scale(
this._graphData[1],
tickCount
);
return [scaledMin, scaledMax];
},
},
},
legend: {
show: false,
},
},
legend: {
show: false
}
}, this._graphData, document.getElementById(`chart_${this.serverId}`))
this._graphData,
document.getElementById(`chart_${this.serverId}`)
);
}
handlePing (payload, timestamp) {
if (typeof payload.playerCount === 'number') {
this.playerCount = payload.playerCount
handlePing(payload, timestamp) {
if (typeof payload.playerCount === "number") {
this.playerCount = payload.playerCount;
// Reset failed ping counter to ensure the next connection error
// doesn't instantly retrigger a layout change
this._failedSequentialPings = 0
this._failedSequentialPings = 0;
} else {
// Attempt to retain a copy of the cached playerCount for up to N failed pings
// This prevents minor connection issues from constantly reshuffling the layout
if (++this._failedSequentialPings > 5) {
this.playerCount = 0
this.playerCount = 0;
}
}
// Use payload.playerCount so nulls WILL be pushed into the graphing data
this._graphData[0].push(timestamp)
this._graphData[1].push(payload.playerCount)
this._graphData[0].push(timestamp);
this._graphData[1].push(payload.playerCount);
// Trim graphData to within the max length by shifting out the leading elements
for (const series of this._graphData) {
if (series.length > this._app.publicConfig.serverGraphMaxLength) {
series.shift()
series.shift();
}
}
// Redraw the plot instance
this._plotInstance.setData(this._graphData)
this._plotInstance.setData(this._graphData);
}
updateServerRankIndex (rankIndex) {
this.rankIndex = rankIndex
updateServerRankIndex(rankIndex) {
this.rankIndex = rankIndex;
document.getElementById(`ranking_${this.serverId}`).innerText = `#${rankIndex + 1}`
document.getElementById(`ranking_${this.serverId}`).innerText = `#${
rankIndex + 1
}`;
}
_renderValue (prefix, handler) {
const labelElement = document.getElementById(`${prefix}_${this.serverId}`)
_renderValue(prefix, handler) {
const labelElement = document.getElementById(`${prefix}_${this.serverId}`);
labelElement.style.display = 'block'
labelElement.style.display = "block";
const valueElement = document.getElementById(`${prefix}-value_${this.serverId}`)
const targetElement = valueElement || labelElement
const valueElement = document.getElementById(
`${prefix}-value_${this.serverId}`
);
const targetElement = valueElement || labelElement;
if (targetElement) {
if (typeof handler === 'function') {
handler(targetElement)
if (typeof handler === "function") {
handler(targetElement);
} else {
targetElement.innerText = handler
targetElement.innerText = handler;
}
}
}
_hideValue (prefix) {
const element = document.getElementById(`${prefix}_${this.serverId}`)
_hideValue(prefix) {
const element = document.getElementById(`${prefix}_${this.serverId}`);
element.style.display = 'none'
element.style.display = "none";
}
updateServerStatus (ping, minecraftVersions) {
updateServerStatus(ping, minecraftVersions) {
if (ping.versions) {
this._renderValue('version', formatMinecraftVersions(ping.versions, minecraftVersions[this.data.type]) || '')
this._renderValue(
"version",
formatMinecraftVersions(
ping.versions,
minecraftVersions[this.data.type]
) || ""
);
}
if (ping.recordData) {
this._renderValue('record', (element) => {
this._renderValue("record", (element) => {
if (ping.recordData.timestamp > 0) {
element.innerText = `${formatNumber(ping.recordData.playerCount)} (${formatDate(ping.recordData.timestamp)})`
element.title = `At ${formatDate(ping.recordData.timestamp)} ${formatTimestampSeconds(ping.recordData.timestamp)}`
element.innerText = `${formatNumber(
ping.recordData.playerCount
)} (${formatDate(ping.recordData.timestamp)})`;
element.title = `At ${formatDate(
ping.recordData.timestamp
)} ${formatTimestampSeconds(ping.recordData.timestamp)}`;
} else {
element.innerText = formatNumber(ping.recordData.playerCount)
element.innerText = formatNumber(ping.recordData.playerCount);
}
})
});
this.lastRecordData = ping.recordData
this.lastRecordData = ping.recordData;
}
if (ping.graphPeakData) {
this._renderValue('peak', (element) => {
element.innerText = formatNumber(ping.graphPeakData.playerCount)
element.title = `At ${formatTimestampSeconds(ping.graphPeakData.timestamp)}`
})
this._renderValue("peak", (element) => {
element.innerText = formatNumber(ping.graphPeakData.playerCount);
element.title = `At ${formatTimestampSeconds(
ping.graphPeakData.timestamp
)}`;
});
this.lastPeakData = ping.graphPeakData
this.lastPeakData = ping.graphPeakData;
}
if (ping.error) {
this._hideValue('player-count')
this._renderValue('error', ping.error.message)
} else if (typeof ping.playerCount !== 'number') {
this._hideValue('player-count')
this._hideValue("player-count");
this._renderValue("error", ping.error.message);
} else if (typeof ping.playerCount !== "number") {
this._hideValue("player-count");
// If the frontend has freshly connection, and the server's last ping was in error, it may not contain an error object
// In this case playerCount will safely be null, so provide a generic error message instead
this._renderValue('error', 'Failed to ping')
} else if (typeof ping.playerCount === 'number') {
this._hideValue('error')
this._renderValue('player-count', formatNumber(ping.playerCount))
this._renderValue("error", "Failed to ping");
} else if (typeof ping.playerCount === "number") {
this._hideValue("error");
this._renderValue("player-count", formatNumber(ping.playerCount));
}
// An updated favicon has been sent, update the src
if (ping.favicon) {
const faviconElement = document.getElementById(`favicon_${this.serverId}`)
const faviconElement = document.getElementById(
`favicon_${this.serverId}`
);
// Since favicons may be URLs, only update the attribute when it has changed
// Otherwise the browser may send multiple requests to the same URL
if (faviconElement.getAttribute('src') !== ping.favicon) {
faviconElement.setAttribute('src', ping.favicon)
if (faviconElement.getAttribute("src") !== ping.favicon) {
faviconElement.setAttribute("src", ping.favicon);
}
}
}
initServerStatus (latestPing) {
const serverElement = document.createElement('div')
initServerStatus(latestPing) {
const serverElement = document.createElement("div");
serverElement.id = `container_${this.serverId}`
serverElement.id = `container_${this.serverId}`;
serverElement.innerHTML = `<div class="column column-favicon">
<img class="server-favicon" src="${latestPing.favicon || MISSING_FAVICON}" id="favicon_${this.serverId}" title="${this.data.name}\n${formatMinecraftServerAddress(this.data.ip, this.data.port)}">
<img class="server-favicon" src="${
latestPing.favicon || MISSING_FAVICON
}" id="favicon_${this.serverId}" title="${
this.data.name
}\n${formatMinecraftServerAddress(this.data.ip, this.data.port)}">
<span class="server-rank" id="ranking_${this.serverId}"></span>
</div>
<div class="column column-status">
<h3 class="server-name"><span class="${this._app.favoritesManager.getIconClass(this.isFavorite)}" id="favorite-toggle_${this.serverId}"></span> ${this.data.name}</h3>
<h3 class="server-name"><span class="${this._app.favoritesManager.getIconClass(
this.isFavorite
)}" id="favorite-toggle_${this.serverId}"></span> ${this.data.name}</h3>
<span class="server-error" id="error_${this.serverId}"></span>
<span class="server-label" id="player-count_${this.serverId}">Players: <span class="server-value" id="player-count-value_${this.serverId}"></span></span>
<span class="server-label" id="peak_${this.serverId}">${this._app.publicConfig.graphDurationLabel} Peak: <span class="server-value" id="peak-value_${this.serverId}">-</span></span>
<span class="server-label" id="record_${this.serverId}">Record: <span class="server-value" id="record-value_${this.serverId}">-</span></span>
<span class="server-label" id="player-count_${
this.serverId
}">Players: <span class="server-value" id="player-count-value_${
this.serverId
}"></span></span>
<span class="server-label" id="peak_${this.serverId}">${
this._app.publicConfig.graphDurationLabel
} Peak: <span class="server-value" id="peak-value_${
this.serverId
}">-</span></span>
<span class="server-label" id="record_${
this.serverId
}">Record: <span class="server-value" id="record-value_${
this.serverId
}">-</span></span>
<span class="server-label" id="version_${this.serverId}"></span>
</div>
<div class="column column-graph" id="chart_${this.serverId}"></div>`
<div class="column column-graph" id="chart_${this.serverId}"></div>`;
serverElement.setAttribute('class', 'server')
serverElement.setAttribute("class", "server");
document.getElementById('server-list').appendChild(serverElement)
document.getElementById("server-list").appendChild(serverElement);
}
updateHighlightedValue (selectedCategory) {
['player-count', 'peak', 'record'].forEach((category) => {
const labelElement = document.getElementById(`${category}_${this.serverId}`)
const valueElement = document.getElementById(`${category}-value_${this.serverId}`)
updateHighlightedValue(selectedCategory) {
["player-count", "peak", "record"].forEach((category) => {
const labelElement = document.getElementById(
`${category}_${this.serverId}`
);
const valueElement = document.getElementById(
`${category}-value_${this.serverId}`
);
if (selectedCategory && category === selectedCategory) {
labelElement.setAttribute('class', 'server-highlighted-label')
valueElement.setAttribute('class', 'server-highlighted-value')
labelElement.setAttribute("class", "server-highlighted-label");
valueElement.setAttribute("class", "server-highlighted-value");
} else {
labelElement.setAttribute('class', 'server-label')
valueElement.setAttribute('class', 'server-value')
labelElement.setAttribute("class", "server-label");
valueElement.setAttribute("class", "server-value");
}
})
});
}
initEventListeners () {
document.getElementById(`favorite-toggle_${this.serverId}`).addEventListener('click', () => {
this._app.favoritesManager.handleFavoriteButtonClick(this)
}, false)
initEventListeners() {
document
.getElementById(`favorite-toggle_${this.serverId}`)
.addEventListener(
"click",
() => {
this._app.favoritesManager.handleFavoriteButtonClick(this);
},
false
);
}
}

View File

@ -1,176 +1,205 @@
export class SocketManager {
constructor (app) {
this._app = app
this._hasRequestedHistoryGraph = false
this._reconnectDelayBase = 0
constructor(app) {
this._app = app;
this._hasRequestedHistoryGraph = false;
this._reconnectDelayBase = 0;
}
reset () {
this._hasRequestedHistoryGraph = false
reset() {
this._hasRequestedHistoryGraph = false;
}
createWebSocket () {
let webSocketProtocol = 'ws:'
if (location.protocol === 'https:') {
webSocketProtocol = 'wss:'
createWebSocket() {
let webSocketProtocol = "ws:";
if (location.protocol === "https:") {
webSocketProtocol = "wss:";
}
this._webSocket = new WebSocket(`${webSocketProtocol}//${location.host}`)
this._webSocket = new WebSocket(`${webSocketProtocol}//${location.host}`);
// The backend will automatically push data once connected
this._webSocket.onopen = () => {
this._app.caption.set('Loading...')
this._app.caption.set("Loading...");
// Reset reconnection scheduling since the WebSocket has been established
this._reconnectDelayBase = 0
}
this._reconnectDelayBase = 0;
};
this._webSocket.onclose = (event) => {
this._app.handleDisconnect()
this._app.handleDisconnect();
// Modify page state to display loading overlay
// Code 1006 denotes "Abnormal closure", most likely from the server or client losing connection
// See https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent
// Treat other codes as active errors (besides connectivity errors) when displaying the message
if (event.code === 1006) {
this._app.caption.set('Lost connection!')
this._app.caption.set("Lost connection!");
} else {
this._app.caption.set('Disconnected due to error.')
this._app.caption.set("Disconnected due to error.");
}
// Schedule socket reconnection attempt
this.scheduleReconnect()
}
this.scheduleReconnect();
};
this._webSocket.onmessage = (message) => {
const payload = JSON.parse(message.data)
const payload = JSON.parse(message.data);
switch (payload.message) {
case 'init':
this._app.setPublicConfig(payload.config)
case "init":
this._app.setPublicConfig(payload.config);
// Display the main page component
// Called here instead of syncComplete so the DOM can be drawn prior to the graphs being drawn
this._app.setPageReady(true)
this._app.setPageReady(true);
// Allow the graphDisplayManager to control whether or not the historical graph is loaded
// Defer to isGraphVisible from the publicConfig to understand if the frontend will ever receive a graph payload
if (this._app.publicConfig.isGraphVisible) {
this.sendHistoryGraphRequest()
this.sendHistoryGraphRequest();
}
payload.servers.forEach((serverPayload, serverId) => {
this._app.addServer(serverId, serverPayload, payload.timestampPoints)
})
this._app.addServer(
serverId,
serverPayload,
payload.timestampPoints
);
});
// Init payload contains all data needed to render the page
// Alert the app it is ready
this._app.handleSyncComplete()
this._app.handleSyncComplete();
break
break;
case 'updateServers': {
for (let serverId = 0; serverId < payload.updates.length; serverId++) {
case "updateServers": {
for (
let serverId = 0;
serverId < payload.updates.length;
serverId++
) {
// The backend may send "update" events prior to receiving all "add" events
// A server has only been added once it's ServerRegistration is defined
// Checking undefined protects from this race condition
const serverRegistration = this._app.serverRegistry.getServerRegistration(serverId)
const serverUpdate = payload.updates[serverId]
const serverRegistration =
this._app.serverRegistry.getServerRegistration(serverId);
const serverUpdate = payload.updates[serverId];
if (serverRegistration) {
serverRegistration.handlePing(serverUpdate, payload.timestamp)
serverRegistration.updateServerStatus(serverUpdate, this._app.publicConfig.minecraftVersions)
serverRegistration.handlePing(serverUpdate, payload.timestamp);
serverRegistration.updateServerStatus(
serverUpdate,
this._app.publicConfig.minecraftVersions
);
}
}
// Bulk add playerCounts into graph during #updateHistoryGraph
if (payload.updateHistoryGraph) {
this._app.graphDisplayManager.addGraphPoint(payload.timestamp, Object.values(payload.updates).map(update => update.playerCount))
this._app.graphDisplayManager.addGraphPoint(
payload.timestamp,
Object.values(payload.updates).map((update) => update.playerCount)
);
// Run redraw tasks after handling bulk updates
this._app.graphDisplayManager.redraw()
this._app.graphDisplayManager.redraw();
}
this._app.percentageBar.redraw()
this._app.updateGlobalStats()
this._app.percentageBar.redraw();
this._app.updateGlobalStats();
break
break;
}
case 'historyGraph': {
this._app.graphDisplayManager.buildPlotInstance(payload.timestamps, payload.graphData)
case "historyGraph": {
this._app.graphDisplayManager.buildPlotInstance(
payload.timestamps,
payload.graphData
);
// Build checkbox elements for graph controls
let lastRowCounter = 0
let controlsHTML = ''
let lastRowCounter = 0;
let controlsHTML = "";
this._app.serverRegistry.getServerRegistrations()
.map(serverRegistration => serverRegistration.data.name)
this._app.serverRegistry
.getServerRegistrations()
.map((serverRegistration) => serverRegistration.data.name)
.sort()
.forEach(serverName => {
const serverRegistration = this._app.serverRegistry.getServerRegistration(serverName)
.forEach((serverName) => {
const serverRegistration =
this._app.serverRegistry.getServerRegistration(serverName);
controlsHTML += `<td><label>
<input type="checkbox" class="graph-control" minetrack-server-id="${serverRegistration.serverId}" ${serverRegistration.isVisible ? 'checked' : ''}>
<input type="checkbox" class="graph-control" minetrack-server-id="${
serverRegistration.serverId
}" ${serverRegistration.isVisible ? "checked" : ""}>
${serverName}
</label></td>`
</label></td>`;
// Occasionally break table rows using a magic number
if (++lastRowCounter % 6 === 0) {
controlsHTML += '</tr><tr>'
controlsHTML += "</tr><tr>";
}
})
});
// Apply generated HTML and show controls
document.getElementById('big-graph-checkboxes').innerHTML = `<table><tr>${controlsHTML}</tr></table>`
document.getElementById('big-graph-controls').style.display = 'block'
document.getElementById(
"big-graph-checkboxes"
).innerHTML = `<table><tr>${controlsHTML}</tr></table>`;
document.getElementById("big-graph-controls").style.display = "block";
// Bind click event for updating graph data
this._app.graphDisplayManager.initEventListeners()
break
this._app.graphDisplayManager.initEventListeners();
break;
}
}
}
};
}
scheduleReconnect () {
scheduleReconnect() {
// Release any active WebSocket references
this._webSocket = undefined
this._webSocket = undefined;
this._reconnectDelayBase++
this._reconnectDelayBase++;
// Exponential backoff for reconnection attempts
// Clamp ceiling value to 30 seconds
this._reconnectDelaySeconds = Math.min((this._reconnectDelayBase * this._reconnectDelayBase), 30)
this._reconnectDelaySeconds = Math.min(
this._reconnectDelayBase * this._reconnectDelayBase,
30
);
const reconnectInterval = setInterval(() => {
this._reconnectDelaySeconds--
this._reconnectDelaySeconds--;
if (this._reconnectDelaySeconds === 0) {
// Explicitly clear interval, this avoids race conditions
// #clearInterval first to avoid potential errors causing pre-mature returns
clearInterval(reconnectInterval)
clearInterval(reconnectInterval);
// Update displayed text
this._app.caption.set('Reconnecting...')
this._app.caption.set("Reconnecting...");
// Attempt reconnection
// Only attempt when reconnectDelaySeconds === 0 and not <= 0, otherwise multiple attempts may be started
this.createWebSocket()
this.createWebSocket();
} else if (this._reconnectDelaySeconds > 0) {
// Update displayed text
this._app.caption.set(`Reconnecting in ${this._reconnectDelaySeconds}s...`)
this._app.caption.set(
`Reconnecting in ${this._reconnectDelaySeconds}s...`
);
}
}, 1000)
}, 1000);
}
sendHistoryGraphRequest () {
sendHistoryGraphRequest() {
if (!this._hasRequestedHistoryGraph) {
this._hasRequestedHistoryGraph = true
this._hasRequestedHistoryGraph = true;
// Send request as a plain text string to avoid the server needing to parse JSON
// This is mostly to simplify the backend server's need for error handling
this._webSocket.send('requestHistoryGraph')
this._webSocket.send("requestHistoryGraph");
}
}
}

View File

@ -1,199 +1,215 @@
const SORT_OPTIONS = [
{
getName: () => 'Players',
getName: () => "Players",
sortFunc: (a, b) => b.playerCount - a.playerCount,
highlightedValue: 'player-count'
highlightedValue: "player-count",
},
{
getName: (app) => {
return `${app.publicConfig.graphDurationLabel} Peak`
return `${app.publicConfig.graphDurationLabel} Peak`;
},
sortFunc: (a, b) => {
if (!a.lastPeakData && !b.lastPeakData) {
return 0
return 0;
} else if (a.lastPeakData && !b.lastPeakData) {
return -1
return -1;
} else if (b.lastPeakData && !a.lastPeakData) {
return 1
return 1;
}
return b.lastPeakData.playerCount - a.lastPeakData.playerCount
return b.lastPeakData.playerCount - a.lastPeakData.playerCount;
},
testFunc: (app) => {
// Require at least one ServerRegistration to have a lastPeakData value defined
for (const serverRegistration of app.serverRegistry.getServerRegistrations()) {
if (serverRegistration.lastPeakData) {
return true
return true;
}
}
return false
return false;
},
highlightedValue: 'peak'
highlightedValue: "peak",
},
{
getName: () => 'Record',
getName: () => "Record",
sortFunc: (a, b) => {
if (!a.lastRecordData && !b.lastRecordData) {
return 0
return 0;
} else if (a.lastRecordData && !b.lastRecordData) {
return -1
return -1;
} else if (b.lastRecordData && !a.lastRecordData) {
return 1
return 1;
}
return b.lastRecordData.playerCount - a.lastRecordData.playerCount
return b.lastRecordData.playerCount - a.lastRecordData.playerCount;
},
testFunc: (app) => {
// Require at least one ServerRegistration to have a lastRecordData value defined
for (const serverRegistration of app.serverRegistry.getServerRegistrations()) {
if (serverRegistration.lastRecordData) {
return true
return true;
}
}
return false
return false;
},
highlightedValue: 'record'
}
]
highlightedValue: "record",
},
];
const SORT_OPTION_INDEX_DEFAULT = 0
const SORT_OPTION_INDEX_STORAGE_KEY = 'minetrack_sort_option_index'
const SORT_OPTION_INDEX_DEFAULT = 0;
const SORT_OPTION_INDEX_STORAGE_KEY = "minetrack_sort_option_index";
export class SortController {
constructor (app) {
this._app = app
this._buttonElement = document.getElementById('sort-by')
this._textElement = document.getElementById('sort-by-text')
this._sortOptionIndex = SORT_OPTION_INDEX_DEFAULT
constructor(app) {
this._app = app;
this._buttonElement = document.getElementById("sort-by");
this._textElement = document.getElementById("sort-by-text");
this._sortOptionIndex = SORT_OPTION_INDEX_DEFAULT;
}
reset () {
this._lastSortedServers = undefined
reset() {
this._lastSortedServers = undefined;
// Reset modified DOM structures
this._buttonElement.style.display = 'none'
this._textElement.innerText = '...'
this._buttonElement.style.display = "none";
this._textElement.innerText = "...";
// Remove bound DOM event listeners
this._buttonElement.removeEventListener('click', this.handleSortByButtonClick)
this._buttonElement.removeEventListener(
"click",
this.handleSortByButtonClick
);
}
loadLocalStorage () {
if (typeof localStorage !== 'undefined') {
const sortOptionIndex = localStorage.getItem(SORT_OPTION_INDEX_STORAGE_KEY)
loadLocalStorage() {
if (typeof localStorage !== "undefined") {
const sortOptionIndex = localStorage.getItem(
SORT_OPTION_INDEX_STORAGE_KEY
);
if (sortOptionIndex) {
this._sortOptionIndex = parseInt(sortOptionIndex)
this._sortOptionIndex = parseInt(sortOptionIndex);
}
}
}
updateLocalStorage () {
if (typeof localStorage !== 'undefined') {
updateLocalStorage() {
if (typeof localStorage !== "undefined") {
if (this._sortOptionIndex !== SORT_OPTION_INDEX_DEFAULT) {
localStorage.setItem(SORT_OPTION_INDEX_STORAGE_KEY, this._sortOptionIndex)
localStorage.setItem(
SORT_OPTION_INDEX_STORAGE_KEY,
this._sortOptionIndex
);
} else {
localStorage.removeItem(SORT_OPTION_INDEX_STORAGE_KEY)
localStorage.removeItem(SORT_OPTION_INDEX_STORAGE_KEY);
}
}
}
show () {
show() {
// Load the saved option selection, if any
this.loadLocalStorage()
this.loadLocalStorage();
this.updateSortOption()
this.updateSortOption();
// Bind DOM event listeners
// This is removed by #reset to avoid multiple listeners
this._buttonElement.addEventListener('click', this.handleSortByButtonClick)
this._buttonElement.addEventListener("click", this.handleSortByButtonClick);
// Show #sort-by element
this._buttonElement.style.display = 'inline-block'
this._buttonElement.style.display = "inline-block";
}
handleSortByButtonClick = () => {
while (true) {
// Increment to the next sort option, wrap around if needed
this._sortOptionIndex = (this._sortOptionIndex + 1) % SORT_OPTIONS.length
this._sortOptionIndex = (this._sortOptionIndex + 1) % SORT_OPTIONS.length;
// Only break if the sortOption is supported
// This can technically cause an infinite loop, but never should assuming
// at least one sortOption does not implement the test OR always returns true
const sortOption = SORT_OPTIONS[this._sortOptionIndex]
const sortOption = SORT_OPTIONS[this._sortOptionIndex];
if (!sortOption.testFunc || sortOption.testFunc(this._app)) {
break
break;
}
}
// Redraw the button and sort the servers
this.updateSortOption()
this.updateSortOption();
// Save the updated option selection
this.updateLocalStorage()
}
this.updateLocalStorage();
};
updateSortOption = () => {
const sortOption = SORT_OPTIONS[this._sortOptionIndex]
const sortOption = SORT_OPTIONS[this._sortOptionIndex];
// Pass app instance so sortOption names can be dynamically generated
this._textElement.innerText = sortOption.getName(this._app)
this._textElement.innerText = sortOption.getName(this._app);
// Update all servers highlighted values
for (const serverRegistration of this._app.serverRegistry.getServerRegistrations()) {
serverRegistration.updateHighlightedValue(sortOption.highlightedValue)
serverRegistration.updateHighlightedValue(sortOption.highlightedValue);
}
this.sortServers()
}
this.sortServers();
};
sortServers = () => {
const sortOption = SORT_OPTIONS[this._sortOptionIndex]
const sortOption = SORT_OPTIONS[this._sortOptionIndex];
const sortedServers = this._app.serverRegistry.getServerRegistrations().sort((a, b) => {
if (a.isFavorite && !b.isFavorite) {
return -1
} else if (b.isFavorite && !a.isFavorite) {
return 1
}
const sortedServers = this._app.serverRegistry
.getServerRegistrations()
.sort((a, b) => {
if (a.isFavorite && !b.isFavorite) {
return -1;
} else if (b.isFavorite && !a.isFavorite) {
return 1;
}
return sortOption.sortFunc(a, b)
})
return sortOption.sortFunc(a, b);
});
// Test if sortedServers has changed from the previous listing
// This avoids DOM updates and graphs being redrawn
const sortedServerIds = sortedServers.map(server => server.serverId)
const sortedServerIds = sortedServers.map((server) => server.serverId);
if (this._lastSortedServers) {
let allMatch = true
let allMatch = true;
// Test if the arrays have actually changed
// No need to length check, they are the same source data each time
for (let i = 0; i < sortedServerIds.length; i++) {
if (sortedServerIds[i] !== this._lastSortedServers[i]) {
allMatch = false
break
allMatch = false;
break;
}
}
if (allMatch) {
return
return;
}
}
this._lastSortedServers = sortedServerIds
this._lastSortedServers = sortedServerIds;
// Sort a ServerRegistration list by the sortOption ONLY
// This is used to determine the ServerRegistration's rankIndex without #isFavorite skewing values
const rankIndexSort = this._app.serverRegistry.getServerRegistrations().sort(sortOption.sortFunc)
const rankIndexSort = this._app.serverRegistry
.getServerRegistrations()
.sort(sortOption.sortFunc);
// Update the DOM structure
sortedServers.forEach(function (serverRegistration) {
const parentElement = document.getElementById('server-list')
const serverElement = document.getElementById(`container_${serverRegistration.serverId}`)
const parentElement = document.getElementById("server-list");
const serverElement = document.getElementById(
`container_${serverRegistration.serverId}`
);
parentElement.appendChild(serverElement)
parentElement.appendChild(serverElement);
// Set the ServerRegistration's rankIndex to its indexOf the normal sort
serverRegistration.updateServerRankIndex(rankIndexSort.indexOf(serverRegistration))
})
}
serverRegistration.updateServerRankIndex(
rankIndexSort.indexOf(serverRegistration)
);
});
};
}

View File

@ -1,124 +1,131 @@
export class Tooltip {
constructor () {
this._div = document.getElementById('tooltip')
constructor() {
this._div = document.getElementById("tooltip");
}
set (x, y, offsetX, offsetY, html) {
this._div.innerHTML = html
set(x, y, offsetX, offsetY, html) {
this._div.innerHTML = html;
// Assign display: block so that the offsetWidth is valid
this._div.style.display = 'block'
this._div.style.display = "block";
// Prevent the div from overflowing the page width
const tooltipWidth = this._div.offsetWidth
const tooltipWidth = this._div.offsetWidth;
// 1.2 is a magic number used to pad the offset to ensure the tooltip
// never gets close or surpasses the page's X width
if (x + offsetX + (tooltipWidth * 1.2) > window.innerWidth) {
x -= tooltipWidth
offsetX *= -1
if (x + offsetX + tooltipWidth * 1.2 > window.innerWidth) {
x -= tooltipWidth;
offsetX *= -1;
}
this._div.style.top = `${y + offsetY}px`
this._div.style.left = `${x + offsetX}px`
this._div.style.top = `${y + offsetY}px`;
this._div.style.left = `${x + offsetX}px`;
}
hide = () => {
this._div.style.display = 'none'
}
this._div.style.display = "none";
};
}
export class Caption {
constructor () {
this._div = document.getElementById('status-text')
constructor() {
this._div = document.getElementById("status-text");
}
set (text) {
this._div.innerText = text
this._div.style.display = 'block'
set(text) {
this._div.innerText = text;
this._div.style.display = "block";
}
hide () {
this._div.style.display = 'none'
hide() {
this._div.style.display = "none";
}
}
// Minecraft Java Edition default server port: 25565
// Minecraft Bedrock Edition default server port: 19132
const MINECRAFT_DEFAULT_PORTS = [25565, 19132]
const MINECRAFT_DEFAULT_PORTS = [25565, 19132];
export function formatMinecraftServerAddress (ip, port) {
export function formatMinecraftServerAddress(ip, port) {
if (port && !MINECRAFT_DEFAULT_PORTS.includes(port)) {
return `${ip}:${port}`
return `${ip}:${port}`;
}
return ip
return ip;
}
// Detect gaps in versions by matching their indexes to knownVersions
export function formatMinecraftVersions (versions, knownVersions) {
if (!versions || !versions.length || !knownVersions || !knownVersions.length) {
return
export function formatMinecraftVersions(versions, knownVersions) {
if (
!versions ||
!versions.length ||
!knownVersions ||
!knownVersions.length
) {
return;
}
let currentVersionGroup = []
const versionGroups = []
let currentVersionGroup = [];
const versionGroups = [];
for (let i = 0; i < versions.length; i++) {
// Look for value mismatch between the previous index
// Require i > 0 since lastVersionIndex is undefined for i === 0
if (i > 0 && versions[i] - versions[i - 1] !== 1) {
versionGroups.push(currentVersionGroup)
currentVersionGroup = []
versionGroups.push(currentVersionGroup);
currentVersionGroup = [];
}
currentVersionGroup.push(versions[i])
currentVersionGroup.push(versions[i]);
}
// Ensure the last versionGroup is always pushed
if (currentVersionGroup.length > 0) {
versionGroups.push(currentVersionGroup)
versionGroups.push(currentVersionGroup);
}
if (versionGroups.length === 0) {
return
return;
}
// Remap individual versionGroups values into named versions
return versionGroups.map(versionGroup => {
const startVersion = knownVersions[versionGroup[0]]
return versionGroups
.map((versionGroup) => {
const startVersion = knownVersions[versionGroup[0]];
if (versionGroup.length === 1) {
// A versionGroup may contain a single version, only return its name
// This is a cosmetic catch to avoid version labels like 1.0-1.0
return startVersion
} else {
const endVersion = knownVersions[versionGroup[versionGroup.length - 1]]
return `${startVersion}-${endVersion}`
}
}).join(', ')
if (versionGroup.length === 1) {
// A versionGroup may contain a single version, only return its name
// This is a cosmetic catch to avoid version labels like 1.0-1.0
return startVersion;
} else {
const endVersion = knownVersions[versionGroup[versionGroup.length - 1]];
return `${startVersion}-${endVersion}`;
}
})
.join(", ");
}
export function formatTimestampSeconds (secs) {
const date = new Date(0)
date.setUTCSeconds(secs)
return date.toLocaleTimeString()
export function formatTimestampSeconds(secs) {
const date = new Date(0);
date.setUTCSeconds(secs);
return date.toLocaleTimeString();
}
export function formatDate (secs) {
const date = new Date(0)
date.setUTCSeconds(secs)
return date.toLocaleDateString()
export function formatDate(secs) {
const date = new Date(0);
date.setUTCSeconds(secs);
return date.toLocaleDateString();
}
export function formatPercent (x, over) {
const val = Math.round((x / over) * 100 * 10) / 10
return `${val}%`
export function formatPercent(x, over) {
const val = Math.round((x / over) * 100 * 10) / 10;
return `${val}%`;
}
export function formatNumber (x) {
if (typeof x !== 'number') {
return '-'
export function formatNumber(x) {
if (typeof x !== "number") {
return "-";
} else {
return x.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ',')
return x.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ",");
}
}