mirror of
https://github.com/minetest/contentdb.git
synced 2025-01-03 19:57:29 +01:00
260 lines
5.7 KiB
JavaScript
260 lines
5.7 KiB
JavaScript
"use strict";
|
|
|
|
|
|
const labelColor = "#bbb";
|
|
const annotationColor = "#bbb";
|
|
const annotationLabelBgColor = "#444";
|
|
const gridColor = "#333";
|
|
|
|
|
|
const chartColors = [
|
|
"#7eb26d",
|
|
"#eab839",
|
|
"#6ed0e0",
|
|
"#e24d42",
|
|
"#1f78c1",
|
|
"#ba43a9",
|
|
];
|
|
|
|
|
|
const annotationNov5 = {
|
|
type: "line",
|
|
borderColor: annotationColor,
|
|
borderWidth: 1,
|
|
click: function({chart, element}) {
|
|
document.location = "https://fosstodon.org/@rubenwardy/109303281233703275";
|
|
},
|
|
label: {
|
|
backgroundColor: annotationLabelBgColor,
|
|
content: "YouTube Video",
|
|
display: true,
|
|
position: "end",
|
|
color: "#00bc8c",
|
|
rotation: "auto",
|
|
backgroundShadowColor: "rgba(0, 0, 0, 0.4)",
|
|
shadowBlur: 3,
|
|
},
|
|
scaleID: "x",
|
|
value: "2022-11-05",
|
|
};
|
|
|
|
|
|
function hexToRgb(hex) {
|
|
var bigint = parseInt(hex, 16);
|
|
var r = (bigint >> 16) & 255;
|
|
var g = (bigint >> 8) & 255;
|
|
var b = bigint & 255;
|
|
|
|
return r + "," + g + "," + b;
|
|
}
|
|
|
|
|
|
function sum(list) {
|
|
return list.reduce((acc, x) => acc + x, 0);
|
|
}
|
|
|
|
|
|
const chartColorsBg = chartColors.map(color => `rgba(${hexToRgb(color.slice(1))}, 0.2)`);
|
|
|
|
const SECONDS_IN_A_DAY = 1000 * 3600 * 24;
|
|
|
|
async function load_data() {
|
|
const root = document.getElementById("stats-root");
|
|
const source = root.getAttribute("data-source");
|
|
const response = await fetch(source);
|
|
const json = await response.json();
|
|
|
|
document.getElementById("loading").style.display = "none";
|
|
|
|
if (json == null) {
|
|
document.getElementById("empty-view").style.display = "block";
|
|
return;
|
|
}
|
|
|
|
const startDate = new Date(json.start);
|
|
const endDate = new Date(json.end);
|
|
const numberOfDays = Math.round((endDate.valueOf() - startDate.valueOf()) / SECONDS_IN_A_DAY) + 1;
|
|
const dates = [...Array(numberOfDays)].map((_, i) => {
|
|
const date = new Date(startDate.valueOf() + i*SECONDS_IN_A_DAY);
|
|
return date.toISOString().split("T")[0];
|
|
});
|
|
|
|
const total7 = sum(json.platform_minetest.slice(-7)) + sum(json.platform_other.slice(-7));
|
|
document.getElementById("downloads_total7d").textContent = total7;
|
|
document.getElementById("downloads_avg7d").textContent = (total7 / 7).toFixed(0);
|
|
|
|
if (json.platform_minetest.length >= 30) {
|
|
const total30 = sum(json.platform_minetest.slice(-30)) + sum(json.platform_other.slice(-30));
|
|
document.getElementById("downloads_total30d").textContent = total30;
|
|
document.getElementById("downloads_avg30d").textContent = (total30 / 30).toFixed(0);
|
|
} else {
|
|
document.getElementById("downloads30").style.display = "none";
|
|
}
|
|
|
|
const jsonOther = json.platform_minetest.map((value, i) =>
|
|
value + json.platform_other[i]
|
|
- json.reason_new[i] - json.reason_dependency[i]
|
|
- json.reason_update[i]);
|
|
|
|
root.style.display = "block";
|
|
|
|
function getData(list) {
|
|
return list.map((value, i) => ({ x: dates[i], y: value }));
|
|
}
|
|
|
|
if (json.package_downloads) {
|
|
const packageRecentDownloads = Object.fromEntries(Object.entries(json.package_downloads)
|
|
.map(([label, values]) => [label, sum(values.slice(-30))]));
|
|
|
|
document.getElementById("downloads-by-package").classList.remove("d-none");
|
|
const ctx = document.getElementById("chart-packages").getContext("2d");
|
|
|
|
const data = {
|
|
datasets: Object.entries(json.package_downloads)
|
|
.sort((a, b) => packageRecentDownloads[a[0]] - packageRecentDownloads[b[0]])
|
|
.map(([label, values]) => ({ label, data: getData(values) })),
|
|
};
|
|
setup_chart(ctx, data);
|
|
}
|
|
|
|
{
|
|
const ctx = document.getElementById("chart-platform").getContext("2d");
|
|
const data = {
|
|
datasets: [
|
|
{ label: "Web / other", data: getData(json.platform_other) },
|
|
{ label: "Minetest", data: getData(json.platform_minetest) },
|
|
],
|
|
};
|
|
setup_chart(ctx, data);
|
|
}
|
|
|
|
{
|
|
const ctx = document.getElementById("chart-reason").getContext("2d");
|
|
const data = {
|
|
datasets: [
|
|
{ label: "Other / Unknown", data: getData(jsonOther) },
|
|
{ label: "Update", data: getData(json.reason_update) },
|
|
{ label: "Dependency", data: getData(json.reason_dependency) },
|
|
{ label: "New Install", data: getData(json.reason_new) },
|
|
],
|
|
};
|
|
setup_chart(ctx, data);
|
|
}
|
|
|
|
{
|
|
const ctx = document.getElementById("chart-reason-pie").getContext("2d");
|
|
const data = {
|
|
labels: [
|
|
"New Install",
|
|
"Dependency",
|
|
"Update",
|
|
"Other / Unknown",
|
|
],
|
|
datasets: [{
|
|
label: "My First Dataset",
|
|
data: [
|
|
sum(json.reason_new),
|
|
sum(json.reason_dependency),
|
|
sum(json.reason_update),
|
|
sum(jsonOther),
|
|
],
|
|
backgroundColor: chartColors,
|
|
hoverOffset: 4,
|
|
borderWidth: 0,
|
|
}]
|
|
};
|
|
const config = {
|
|
type: "doughnut",
|
|
data: data,
|
|
options: {
|
|
responsive: true,
|
|
plugins: {
|
|
legend: {
|
|
labels: {
|
|
color: labelColor,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
};
|
|
new Chart(ctx, config);
|
|
}
|
|
}
|
|
|
|
|
|
function setup_chart(ctx, data) {
|
|
data.datasets = data.datasets.map((set, i) => {
|
|
const colorIdx = (data.datasets.length - i - 1) % chartColors.length;
|
|
return {
|
|
fill: true,
|
|
backgroundColor: chartColorsBg[colorIdx],
|
|
borderColor: chartColors[colorIdx],
|
|
pointBackgroundColor: chartColors[colorIdx],
|
|
...set,
|
|
};
|
|
});
|
|
|
|
const config = {
|
|
type: "line",
|
|
data: data,
|
|
options: {
|
|
responsive: true,
|
|
plugins: {
|
|
tooltip: {
|
|
mode: "index"
|
|
},
|
|
|
|
legend: {
|
|
reverse: true,
|
|
labels: {
|
|
color: labelColor,
|
|
}
|
|
},
|
|
|
|
annotation: {
|
|
annotations: {
|
|
annotationNov5,
|
|
},
|
|
},
|
|
},
|
|
interaction: {
|
|
mode: "nearest",
|
|
axis: "x",
|
|
intersect: false
|
|
},
|
|
scales: {
|
|
x: {
|
|
type: "time",
|
|
time: {
|
|
// min: start,
|
|
// max: end,
|
|
unit: "day",
|
|
},
|
|
ticks: {
|
|
color: labelColor,
|
|
},
|
|
grid: {
|
|
color: gridColor,
|
|
}
|
|
},
|
|
y: {
|
|
stacked: true,
|
|
min: 0,
|
|
precision: 0,
|
|
ticks: {
|
|
color: labelColor,
|
|
},
|
|
grid: {
|
|
color: gridColor,
|
|
}
|
|
},
|
|
}
|
|
}
|
|
};
|
|
|
|
new Chart(ctx, config);
|
|
}
|
|
|
|
|
|
$(load_data);
|