// Progressive enhancement for local frontend-backend checks.
const DEFAULT_SERVER_POLL_INTERVAL_MS = 300 * 1000;
const TRUSTED_SERVER_ACTIONS = Object.freeze({
"comunidad-hispana-01": Object.freeze({
publicScoreboardUrl: "https://scoreboard.comunidadhll.es",
historicalUrl: "./historico.html?server=comunidad-hispana-01",
currentMatchUrl: "./partida-actual.html?server=comunidad-hispana-01",
}),
"comunidad-hispana-02": Object.freeze({
publicScoreboardUrl: "https://scoreboard.comunidadhll.es:5443",
historicalUrl: "./historico.html?server=comunidad-hispana-02",
currentMatchUrl: "./partida-actual.html?server=comunidad-hispana-02",
}),
});
const COMMUNITY_CLANS = Object.freeze([
{
name: "LCM",
badge: "Clan CH",
description:
"Clan activo de la comunidad, con acceso directo a su discord.",
logoSrc: "./assets/img/clans/lcm.png",
logoAlt: "Logo de LCM",
logoClassName: "",
discordUrl: "https://discord.gg/9F9S353QZv",
discordLabel: "Abrir Discord",
},
{
name: "La 129",
badge: "Clan CH",
description:
"Clan activo de la comunidad.",
logoSrc: "./assets/img/clans/la129.png",
logoAlt: "Logo de La 129",
logoClassName: "clan-card__logo--wide",
discordUrl: "",
discordLabel: "Proximamente",
},
{
name: "250 Hispania",
badge: "Clan CH",
description:
"Clan activo de la comunidad, con acceso directo a su discord.",
logoSrc: "./assets/img/clans/250hispania-shield.png",
logoAlt: "Escudo de 250 Hispania",
logoClassName: "clan-card__logo--shield",
discordUrl: "https://discord.gg/3E62Yb6Aw3",
discordLabel: "Abrir Discord",
},
{
name: "H9H",
badge: "Clan CH",
description:
"Clan activo de la comunidad, con acceso directo a su discord.",
logoSrc: "./assets/img/clans/h9h.png",
logoAlt: "",
logoClassName: "",
discordUrl: "https://discord.gg/tYnXK7MQjB",
discordLabel: "Abrir Discord",
placeholderLabel: "H9H",
},
{
name: "BxB",
badge: "Clan CH",
description:
"Clan activo de la comunidad.",
logoSrc: "./assets/img/clans/bxb.png",
logoAlt: "Logo de BxB",
logoClassName: "",
discordUrl: "",
discordLabel: "Proximamente",
},
{
name: "7dv",
badge: "Clan CH",
description:
"Clan activo de la comunidad, con acceso directo a su discord.",
logoSrc: "./assets/img/clans/7dv.png",
logoAlt: "Logo de 7dv",
logoClassName: "",
discordUrl: "https://discord.gg/3sxNQZwrg6",
discordLabel: "Abrir Discord",
},
]);
document.addEventListener("DOMContentLoaded", () => {
console.info("HLL Vietnam frontend ready");
const backendBaseUrl =
document.body.dataset.backendBaseUrl || "http://127.0.0.1:8000";
const serverPollIntervalMs = getServerPollIntervalMs(
document.body.dataset.serverRefreshMs,
);
const statusNode = document.getElementById("backend-status");
const trailerFrame = document.getElementById("trailer-frame");
const trailerTitle = document.getElementById("trailer-title");
const serversTitle = document.getElementById("servers-title");
const serversList = document.getElementById("servers-list");
const serversBadge = document.getElementById("servers-badge");
const communityClansList = document.getElementById("community-clans-list");
updateBackendStatus(statusNode, "Backend comprobando", "status-chip--idle");
setServersDataState(serversBadge, { timestampLabel: "" });
renderServersLoadingState(serversList);
hydrateCommunityClans(communityClansList);
let serverRefreshInFlight = false;
const refreshServers = async () => {
if (serverRefreshInFlight) {
return;
}
serverRefreshInFlight = true;
try {
await hydrateServers(
backendBaseUrl,
serversTitle,
serversList,
serversBadge,
);
} finally {
serverRefreshInFlight = false;
}
};
Promise.allSettled([
fetchHealth(backendBaseUrl, statusNode),
hydrateTrailer(backendBaseUrl, trailerFrame, trailerTitle),
refreshServers(),
]).catch((error) => {
console.warn("Progressive enhancement failed", error);
});
if (serverPollIntervalMs > 0) {
window.setInterval(() => {
void refreshServers();
}, serverPollIntervalMs);
}
});
async function fetchHealth(backendBaseUrl, statusNode) {
try {
const response = await fetch(`${backendBaseUrl}/health`);
if (!response.ok) {
throw new Error(`Health request failed with ${response.status}`);
}
const payload = await response.json();
if (payload.status === "ok") {
updateBackendStatus(statusNode, "Backend operativo", "status-chip--ok");
return;
}
throw new Error("Unexpected health payload");
} catch (error) {
console.warn("Backend health check unavailable", error);
updateBackendStatus(
statusNode,
"Modo estatico activo",
"status-chip--fallback",
);
}
}
async function hydrateTrailer(backendBaseUrl, trailerFrame, trailerTitle) {
if (!trailerFrame || !trailerTitle) {
return;
}
try {
const response = await fetch(`${backendBaseUrl}/api/trailer`);
if (!response.ok) {
throw new Error(`Trailer request failed with ${response.status}`);
}
const payload = await response.json();
const trailer = payload.data;
if (!trailer || !trailer.video_url || !trailer.title) {
throw new Error("Trailer payload incomplete");
}
trailerFrame.src = trailer.video_url;
trailerFrame.title = trailer.title;
trailerTitle.textContent = trailer.title;
} catch (error) {
console.warn("Trailer placeholder remains static", error);
}
}
async function hydrateServers(
backendBaseUrl,
serversTitle,
serversList,
serversBadge,
) {
if (!serversTitle || !serversList || !serversBadge) {
return;
}
try {
const payload = await fetchJson(`${backendBaseUrl}/api/servers`);
const serversData = payload.data;
if (!serversData || !Array.isArray(serversData.items)) {
throw new Error("Servers payload incomplete");
}
serversTitle.textContent =
serversData.title || "Estado actual de servidores";
setServersDataState(serversBadge, deriveSnapshotState(serversData));
if (serversData.items.length === 0) {
serversList.innerHTML =
'
Informacion de servidores disponible mas adelante.
';
return;
}
const visibleItems = selectPrimaryServerItems(serversData.items);
serversList.innerHTML = renderServerSections(visibleItems);
} catch (error) {
console.warn("Servers panel failed to hydrate with live data", error);
serversList.innerHTML =
'No se pudo cargar el estado real de servidores en este momento.
';
setServersDataState(serversBadge, {
label: "Actualizacion no disponible",
isFresh: false,
});
}
}
function renderServersLoadingState(serversList) {
if (!serversList) {
return;
}
serversList.innerHTML = `
Cargando estado real de servidores...
`;
}
function updateBackendStatus(statusNode, label, stateClass) {
if (!statusNode) {
return;
}
statusNode.textContent = label;
statusNode.classList.remove("status-chip--ok", "status-chip--fallback");
if (stateClass) {
statusNode.classList.add(stateClass);
}
}
function setServersDataState(badgeNode, state) {
if (!badgeNode) {
return;
}
const hasLabel = typeof state.label === "string" && state.label;
badgeNode.textContent = hasLabel
? state.label
: "Actualizado no disponible";
badgeNode.classList.toggle("status-chip--ok", Boolean(hasLabel && state.isFresh));
badgeNode.classList.toggle(
"status-chip--fallback",
!hasLabel || !state.isFresh,
);
}
function renderServerStatsCard(server) {
const serverName = server.server_name || "Servidor sin nombre";
const statusLabel = formatServerStatus(server.status);
const stateClass =
server.status === "online" ? "server-state--online" : "server-state--offline";
const isRealSnapshot = isRealLiveSnapshot(server);
const currentMap = server.current_map || "Sin mapa disponible";
const region = normalizeServerRegion(server.region);
const players = Number.isFinite(server.players) ? server.players : 0;
const maxPlayers = Number.isFinite(server.max_players) ? server.max_players : 0;
const actionMarkup = renderServerAction(server);
const cardVariantClass = isRealSnapshot ? "server-card--real" : "server-card--reference";
const eyebrowLabel = isRealSnapshot ? "Servidor de comunidad" : "Referencia actual";
const quickFactItems = [
{ label: "Mapa", value: currentMap, valueClassName: "server-card__quickfact-value--map" },
];
if (region) {
quickFactItems.push({ label: "Region", value: region });
}
const quickFacts = renderQuickFacts(quickFactItems);
return `
${escapeHtml(eyebrowLabel)}
${escapeHtml(serverName)}
${escapeHtml(statusLabel)}
${escapeHtml(`${players} / ${maxPlayers}`)}
${quickFacts}
${actionMarkup}
`;
}
function renderServerSections(latestItems) {
return latestItems.map((server) => renderServerStatsCard(server)).join("");
}
function normalizeServerRegion(value) {
if (typeof value !== "string") {
return "";
}
const trimmedValue = value.trim();
if (!trimmedValue) {
return "";
}
const normalizedValue = trimmedValue
.normalize("NFD")
.replace(/[\u0300-\u036f]/g, "")
.toLowerCase();
const placeholderValues = new Set([
"region pendiente",
"region pending",
"pending",
"unknown",
"desconocida",
"no disponible",
"por confirmar",
"n/a",
]);
return placeholderValues.has(normalizedValue) ? "" : trimmedValue;
}
function renderServerAction(server) {
const actions = getTrustedServerActions(server);
if (!actions) {
return "";
}
return `
`;
}
function hydrateCommunityClans(listNode) {
if (!listNode) {
return;
}
listNode.innerHTML = shuffleItems(COMMUNITY_CLANS)
.map((clan) => renderCommunityClanCard(clan))
.join("");
}
function renderCommunityClanCard(clan) {
const logoMarkup = renderClanLogo(clan);
const discordMarkup = renderClanDiscordLink(clan);
return `
${logoMarkup}
${escapeHtml(clan.badge)}
${escapeHtml(clan.name)}
${escapeHtml(clan.description)}
${discordMarkup}
`;
}
function renderClanLogo(clan) {
const logoClassNames = ["clan-card__logo"];
if (clan.logoClassName) {
logoClassNames.push(clan.logoClassName);
}
if (clan.logoSrc) {
return `
`;
}
return `
${escapeHtml(clan.placeholderLabel || clan.name)}
`;
}
function renderClanDiscordLink(clan) {
if (!clan.discordUrl) {
return `
${escapeHtml(clan.discordLabel)}
`;
}
return `
${escapeHtml(clan.discordLabel)}
`;
}
function renderQuickFacts(items) {
return `
${items
.map(
(item) => `
${escapeHtml(item.label)}
${escapeHtml(item.value)}
`,
)
.join("")}
`;
}
function getTrustedServerActions(server) {
const trustedActionKey = resolveTrustedServerActionKey(server);
return TRUSTED_SERVER_ACTIONS[trustedActionKey] || null;
}
function resolveTrustedServerActionKey(server) {
if (!server) {
return "";
}
const externalServerId = getTrimmedServerValue(server.external_server_id);
if (TRUSTED_SERVER_ACTIONS[externalServerId]) {
return externalServerId;
}
const trustedSlugFields = [
server.server_slug,
server.target_key,
server.slug,
server.community_slug,
];
const trustedSlug = trustedSlugFields
.map(getTrimmedServerValue)
.find((value) => TRUSTED_SERVER_ACTIONS[value]);
if (trustedSlug) {
return trustedSlug;
}
const serverNames = [server.server_name, server.name].map(getTrimmedServerValue);
if (
serverNames.some(
(name) => name.startsWith("#01") || name.includes("Comunidad Hispana #01"),
)
) {
return "comunidad-hispana-01";
}
if (
serverNames.some(
(name) => name.startsWith("#02") || name.includes("Comunidad Hispana #02"),
)
) {
return "comunidad-hispana-02";
}
const serverReference = [
getTrimmedServerValue(server.source_ref),
externalServerId,
].join(" ");
if (serverReference.includes("152.114.195.174") || serverReference.includes(":7779")) {
return "comunidad-hispana-01";
}
if (serverReference.includes("152.114.195.150") || serverReference.includes(":7879")) {
return "comunidad-hispana-02";
}
return "";
}
function getTrimmedServerValue(value) {
return typeof value === "string" ? value.trim() : "";
}
function selectPrimaryServerItems(items) {
if (!Array.isArray(items)) {
return [];
}
const realItems = items.filter(isRealLiveSnapshot);
return realItems.length > 0 ? realItems : items;
}
function isRealLiveSnapshot(item) {
return item?.snapshot_origin === "real-a2s" || item?.snapshot_origin === "real-rcon";
}
function deriveSnapshotState(serversData) {
const timestampLabel = serversData?.last_snapshot_at
? formatTimestamp(serversData.last_snapshot_at)
: "";
if (!timestampLabel) {
return {
label: "",
isFresh: false,
};
}
const isFresh = serversData?.is_stale !== true;
return {
label: isFresh
? `Actualizado ${timestampLabel}`
: `Ultimo snapshot ${timestampLabel}`,
isFresh,
};
}
function formatServerStatus(status) {
if (status === "online") {
return "Online";
}
if (status === "offline") {
return "Offline";
}
return "Estado pendiente";
}
function formatTimestamp(timestamp) {
const value = new Date(timestamp);
if (Number.isNaN(value.getTime())) {
return "Fecha no disponible";
}
return new Intl.DateTimeFormat("es-ES", {
dateStyle: "short",
timeStyle: "short",
}).format(value);
}
function getServerPollIntervalMs(rawValue) {
const parsedValue = Number(rawValue);
if (!Number.isFinite(parsedValue) || parsedValue <= 0) {
return DEFAULT_SERVER_POLL_INTERVAL_MS;
}
return parsedValue;
}
async function fetchJson(url) {
const response = await fetch(url);
if (!response.ok) {
throw new Error(`Request failed with ${response.status}`);
}
return response.json();
}
function shuffleItems(items) {
const shuffledItems = [...items];
for (let currentIndex = shuffledItems.length - 1; currentIndex > 0; currentIndex -= 1) {
const randomIndex = Math.floor(Math.random() * (currentIndex + 1));
[shuffledItems[currentIndex], shuffledItems[randomIndex]] = [
shuffledItems[randomIndex],
shuffledItems[currentIndex],
];
}
return shuffledItems;
}
function escapeHtml(value) {
return String(value)
.replaceAll("&", "&")
.replaceAll("<", "<")
.replaceAll(">", ">")
.replaceAll('"', """)
.replaceAll("'", "'");
}