refacto: Removing the admin debugger
This commit is contained in:
@@ -62,17 +62,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Debug team switcher (only visible when admin unlocked) -->
|
|
||||||
<div class="teamCorner teamCorner--hidden" id="teamCorner">
|
|
||||||
<span class="teamCornerLabel">Équipe</span>
|
|
||||||
<div class="teamSegmented" role="group" aria-label="Active team">
|
|
||||||
<div class="teamSegmentedTrack" id="teamSegmentedTrack" data-active="blue">
|
|
||||||
<button type="button" class="teamSegmentedBtn" id="teamBlue" data-team="blue">Résistance</button>
|
|
||||||
<button type="button" class="teamSegmentedBtn" id="teamRed" data-team="red">Premier Ordre</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Main app layout: left info column + right galaxy square ─────────────── -->
|
<!-- Main app layout: left info column + right galaxy square ─────────────── -->
|
||||||
<div class="app">
|
<div class="app">
|
||||||
|
|
||||||
@@ -224,21 +213,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
<!-- Admin / options section -->
|
|
||||||
<div class="infoSection infoSection--options">
|
|
||||||
<details class="optionsDetails">
|
|
||||||
<summary class="optionsSummary">⚙ Options</summary>
|
|
||||||
<div class="optionsPanel">
|
|
||||||
<div class="authField">
|
|
||||||
<label>Mot de passe admin</label>
|
|
||||||
<input type="password" id="adminPasswordInput" placeholder="Entrez le mot de passe d'administration" autocomplete="off" />
|
|
||||||
</div>
|
|
||||||
<button type="button" id="adminUnlockBtn" class="adminUnlockBtn">Débloquer</button>
|
|
||||||
<div id="adminStatus" class="adminStatus hidden"></div>
|
|
||||||
</div>
|
|
||||||
</details>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</aside>
|
</aside>
|
||||||
|
|
||||||
<!-- ── Galaxy (square, 1000×1000, fixed ratio) ────────────────────────── -->
|
<!-- ── Galaxy (square, 1000×1000, fixed ratio) ────────────────────────── -->
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { apiLogin, apiRegister, apiGetMe } from "./api.js";
|
import { apiLogin, apiRegister, apiGetMe } from "./api.js";
|
||||||
import { setCurrentTeam, updateTeamSegmented, refreshFromServer } from "./game.js";
|
import { setCurrentTeam, refreshFromServer } from "./game.js";
|
||||||
|
|
||||||
// ── DOM refs ──────────────────────────────────────────────────────────────────
|
// ── DOM refs ──────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
@@ -40,7 +40,6 @@ export function applyUser(user, token) {
|
|||||||
setCurrentTeam(user.team);
|
setCurrentTeam(user.team);
|
||||||
userDisplayEl.textContent = `${user.username} [${user.team}]`;
|
userDisplayEl.textContent = `${user.username} [${user.team}]`;
|
||||||
logoutBtn.classList.remove("hidden");
|
logoutBtn.classList.remove("hidden");
|
||||||
updateTeamSegmented();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function logout() {
|
function logout() {
|
||||||
|
|||||||
@@ -164,11 +164,6 @@ const econScoreRedEl = document.getElementById("econScoreRed");
|
|||||||
const econDeltaBlueEl = document.getElementById("econDeltaBlue");
|
const econDeltaBlueEl = document.getElementById("econDeltaBlue");
|
||||||
const econDeltaRedEl = document.getElementById("econDeltaRed");
|
const econDeltaRedEl = document.getElementById("econDeltaRed");
|
||||||
const elemBonusTableEl = document.getElementById("elementBonusTableBody");
|
const elemBonusTableEl = document.getElementById("elementBonusTableBody");
|
||||||
const teamCorner = document.getElementById("teamCorner");
|
|
||||||
const teamTrack = document.getElementById("teamSegmentedTrack");
|
|
||||||
const teamBlueBtn = document.getElementById("teamBlue");
|
|
||||||
const teamRedBtn = document.getElementById("teamRed");
|
|
||||||
|
|
||||||
// ── Cell helpers ──────────────────────────────────────────────────────────────
|
// ── Cell helpers ──────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
export function cellKey(x, y) { return `${x},${y}`; }
|
export function cellKey(x, y) { return `${x},${y}`; }
|
||||||
@@ -211,8 +206,6 @@ export function applyConfigPayload(data) {
|
|||||||
|
|
||||||
updateResetCountdown();
|
updateResetCountdown();
|
||||||
|
|
||||||
// Team switcher visibility is managed exclusively via unlockTeamSwitcher() /
|
|
||||||
// lockTeamSwitcher() — NOT by debugModeForTeams config.
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export function updateResetCountdown() {
|
export function updateResetCountdown() {
|
||||||
@@ -226,17 +219,6 @@ export function updateResetCountdown() {
|
|||||||
String(s % 60).padStart(2, "0");
|
String(s % 60).padStart(2, "0");
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── Team switcher (admin-only) ────────────────────────────────────────────────
|
|
||||||
|
|
||||||
/** Show the team switcher widget. Called only after successful admin unlock. */
|
|
||||||
export function unlockTeamSwitcher() {
|
|
||||||
teamCorner.classList.remove("teamCorner--hidden");
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Hide the team switcher widget. */
|
|
||||||
export function lockTeamSwitcher() {
|
|
||||||
teamCorner.classList.add("teamCorner--hidden");
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── Scores ────────────────────────────────────────────────────────────────────
|
// ── Scores ────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
@@ -668,11 +650,6 @@ async function onCanvasClick(ev) {
|
|||||||
|
|
||||||
// ── Team segmented control ────────────────────────────────────────────────────
|
// ── Team segmented control ────────────────────────────────────────────────────
|
||||||
|
|
||||||
export function updateTeamSegmented() {
|
|
||||||
teamTrack.dataset.active = currentTeam;
|
|
||||||
teamBlueBtn.setAttribute("aria-pressed", currentTeam === "blue" ? "true" : "false");
|
|
||||||
teamRedBtn.setAttribute("aria-pressed", currentTeam === "red" ? "true" : "false");
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── Lightweight grid refresh (called every second) ────────────────────────────
|
// ── Lightweight grid refresh (called every second) ────────────────────────────
|
||||||
|
|
||||||
@@ -742,7 +719,3 @@ elemBonusTableEl?.addEventListener("click", (ev) => {
|
|||||||
canvas.addEventListener("mousemove", (ev) => { lastPointerEvent = ev; refreshCursor(ev); });
|
canvas.addEventListener("mousemove", (ev) => { lastPointerEvent = ev; refreshCursor(ev); });
|
||||||
canvas.addEventListener("mouseleave", () => { lastPointerEvent = null; canvas.style.cursor = "default"; });
|
canvas.addEventListener("mouseleave", () => { lastPointerEvent = null; canvas.style.cursor = "default"; });
|
||||||
canvas.addEventListener("click", onCanvasClick);
|
canvas.addEventListener("click", onCanvasClick);
|
||||||
|
|
||||||
// Team switcher buttons (kept in codebase, only functional when admin-unlocked)
|
|
||||||
teamBlueBtn.addEventListener("click", () => { setCurrentTeam("blue"); updateTeamSegmented(); updateEconomyDisplay(); draw(); refreshCursorFromLast(); });
|
|
||||||
teamRedBtn.addEventListener( "click", () => { setCurrentTeam("red"); updateTeamSegmented(); updateEconomyDisplay(); draw(); refreshCursorFromLast(); });
|
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import {
|
import {
|
||||||
GAME_CONFIG,
|
GAME_CONFIG,
|
||||||
seedStr,
|
seedStr,
|
||||||
updateTeamSegmented,
|
|
||||||
updateResetCountdown,
|
updateResetCountdown,
|
||||||
fetchConfig,
|
fetchConfig,
|
||||||
fetchGridForSeed,
|
fetchGridForSeed,
|
||||||
@@ -15,7 +14,6 @@ import {
|
|||||||
refreshGridDisplay,
|
refreshGridDisplay,
|
||||||
loadPlayfieldMask,
|
loadPlayfieldMask,
|
||||||
draw,
|
draw,
|
||||||
unlockTeamSwitcher,
|
|
||||||
} from "./game.js";
|
} from "./game.js";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
@@ -31,9 +29,6 @@ const cooldownEl = document.getElementById("cooldownConfig");
|
|||||||
const burgerBtn = document.getElementById("burgerBtn");
|
const burgerBtn = document.getElementById("burgerBtn");
|
||||||
const closeMenuBtn = document.getElementById("closeMenuBtn");
|
const closeMenuBtn = document.getElementById("closeMenuBtn");
|
||||||
const infoColumn = document.getElementById("infoColumn");
|
const infoColumn = document.getElementById("infoColumn");
|
||||||
const adminPasswordIn = document.getElementById("adminPasswordInput");
|
|
||||||
const adminUnlockBtn = document.getElementById("adminUnlockBtn");
|
|
||||||
const adminStatus = document.getElementById("adminStatus");
|
|
||||||
|
|
||||||
// ── Polling ───────────────────────────────────────────────────────────────────
|
// ── Polling ───────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
@@ -89,49 +84,9 @@ document.addEventListener("click", (ev) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Admin password unlock ─────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
function showAdminStatus(message, isOk) {
|
|
||||||
adminStatus.textContent = message;
|
|
||||||
adminStatus.className = "adminStatus " + (isOk ? "adminStatus--ok" : "adminStatus--err");
|
|
||||||
}
|
|
||||||
|
|
||||||
adminUnlockBtn.addEventListener("click", async () => {
|
|
||||||
const password = adminPasswordIn.value.trim();
|
|
||||||
if (!password) {
|
|
||||||
showAdminStatus("Enter a password.", false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const res = await fetch("/api/admin/verify", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({ password }),
|
|
||||||
});
|
|
||||||
const data = await res.json();
|
|
||||||
if (res.ok && data.ok) {
|
|
||||||
showAdminStatus("Admin unlocked — team switcher enabled.", true);
|
|
||||||
unlockTeamSwitcher();
|
|
||||||
adminPasswordIn.value = "";
|
|
||||||
} else {
|
|
||||||
showAdminStatus("Invalid password.", false);
|
|
||||||
}
|
|
||||||
} catch {
|
|
||||||
showAdminStatus("Could not verify — server unreachable.", false);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Allow Enter key in password field
|
|
||||||
adminPasswordIn.addEventListener("keydown", (ev) => {
|
|
||||||
if (ev.key === "Enter") adminUnlockBtn.click();
|
|
||||||
});
|
|
||||||
|
|
||||||
// ── Boot ──────────────────────────────────────────────────────────────────────
|
// ── Boot ──────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
async function boot() {
|
async function boot() {
|
||||||
updateTeamSegmented();
|
|
||||||
|
|
||||||
// Load the SVG playfield mask before any drawing or data fetch
|
// Load the SVG playfield mask before any drawing or data fetch
|
||||||
await loadPlayfieldMask();
|
await loadPlayfieldMask();
|
||||||
|
|
||||||
|
|||||||
169
public/style.css
169
public/style.css
@@ -192,85 +192,7 @@ body {
|
|||||||
background: rgba(113, 199, 255, 0.28);
|
background: rgba(113, 199, 255, 0.28);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ── Team corner (admin only) ─────────────────────────────────────────────── */
|
|
||||||
|
|
||||||
.teamCorner {
|
|
||||||
position: fixed;
|
|
||||||
top: 12px;
|
|
||||||
left: 12px;
|
|
||||||
z-index: 10;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
padding: 8px 10px;
|
|
||||||
border-radius: 12px;
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
|
||||||
background: rgba(7, 10, 20, 0.85);
|
|
||||||
backdrop-filter: blur(8px);
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamCornerLabel {
|
|
||||||
font-size: 11px;
|
|
||||||
font-weight: 600;
|
|
||||||
text-transform: uppercase;
|
|
||||||
letter-spacing: 0.06em;
|
|
||||||
opacity: 0.75;
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamCorner--hidden {
|
|
||||||
display: none !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamSegmented {
|
|
||||||
min-width: 148px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamSegmentedTrack {
|
|
||||||
position: relative;
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
border-radius: 12px;
|
|
||||||
overflow: hidden;
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.18);
|
|
||||||
background: rgba(0, 0, 0, 0.25);
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamSegmentedTrack::before {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
top: 2px;
|
|
||||||
bottom: 2px;
|
|
||||||
left: 2px;
|
|
||||||
width: calc(50% - 4px);
|
|
||||||
border-radius: 9px;
|
|
||||||
background: linear-gradient(180deg, rgba(90, 200, 255, 0.42), rgba(35, 95, 150, 0.55));
|
|
||||||
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.12);
|
|
||||||
transition: left 0.22s ease, background 0.22s ease;
|
|
||||||
z-index: 0;
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamSegmentedTrack[data-active="red"]::before {
|
|
||||||
left: calc(50% + 2px);
|
|
||||||
background: linear-gradient(180deg, rgba(255, 130, 130, 0.42), rgba(150, 45, 45, 0.55));
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamSegmentedBtn {
|
|
||||||
position: relative;
|
|
||||||
z-index: 1;
|
|
||||||
margin: 0;
|
|
||||||
padding: 9px 12px;
|
|
||||||
border: none;
|
|
||||||
background: transparent;
|
|
||||||
color: #e9eef6;
|
|
||||||
font-weight: 700;
|
|
||||||
font-size: 13px;
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
|
|
||||||
.teamSegmentedBtn:hover {
|
|
||||||
color: #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ── Score board ──────────────────────────────────────────────────────────── */
|
/* ── Score board ──────────────────────────────────────────────────────────── */
|
||||||
|
|
||||||
@@ -972,97 +894,6 @@ button:hover {
|
|||||||
color: rgba(255, 220, 100, 0.9);
|
color: rgba(255, 220, 100, 0.9);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ── Options / admin section ──────────────────────────────────────────────── */
|
|
||||||
|
|
||||||
.infoSection--options {
|
|
||||||
padding-top: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsDetails {
|
|
||||||
border-radius: 10px;
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.07);
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsSummary {
|
|
||||||
padding: 8px 12px;
|
|
||||||
font-size: 11px;
|
|
||||||
font-weight: 600;
|
|
||||||
opacity: 0.6;
|
|
||||||
cursor: pointer;
|
|
||||||
list-style: none;
|
|
||||||
user-select: none;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsSummary::after {
|
|
||||||
content: "›";
|
|
||||||
margin-left: auto;
|
|
||||||
font-size: 16px;
|
|
||||||
font-weight: 400;
|
|
||||||
opacity: 0.4;
|
|
||||||
display: inline-block;
|
|
||||||
transform: rotate(0deg);
|
|
||||||
transition: transform 0.2s ease, opacity 0.15s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsDetails[open] .optionsSummary::after {
|
|
||||||
transform: rotate(90deg);
|
|
||||||
opacity: 0.7;
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsSummary::-webkit-details-marker {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsSummary:hover {
|
|
||||||
opacity: 0.9;
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsPanel {
|
|
||||||
padding: 12px;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 10px;
|
|
||||||
border-top: 1px solid rgba(255, 255, 255, 0.07);
|
|
||||||
}
|
|
||||||
|
|
||||||
.optionsPanel .authField input[type="password"] {
|
|
||||||
padding: 8px 10px;
|
|
||||||
font-size: 12px;
|
|
||||||
font-family: "Courier New", Courier, monospace;
|
|
||||||
}
|
|
||||||
|
|
||||||
.adminUnlockBtn {
|
|
||||||
padding: 7px 12px;
|
|
||||||
font-size: 12px;
|
|
||||||
align-self: flex-start;
|
|
||||||
}
|
|
||||||
|
|
||||||
.adminStatus {
|
|
||||||
font-size: 11px;
|
|
||||||
padding: 6px 10px;
|
|
||||||
border-radius: 7px;
|
|
||||||
font-family: "Courier New", Courier, monospace;
|
|
||||||
}
|
|
||||||
|
|
||||||
.adminStatus.hidden {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.adminStatus--ok {
|
|
||||||
color: rgba(90, 200, 130, 0.95);
|
|
||||||
background: rgba(30, 120, 60, 0.15);
|
|
||||||
border: 1px solid rgba(30, 150, 70, 0.25);
|
|
||||||
}
|
|
||||||
|
|
||||||
.adminStatus--err {
|
|
||||||
color: rgba(255, 130, 100, 0.95);
|
|
||||||
background: rgba(200, 50, 30, 0.12);
|
|
||||||
border: 1px solid rgba(200, 50, 30, 0.25);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ── Galaxy: square, fills viewport height, 1:1 ratio ────────────────────── */
|
/* ── Galaxy: square, fills viewport height, 1:1 ratio ────────────────────── */
|
||||||
|
|
||||||
.galaxyMain {
|
.galaxyMain {
|
||||||
|
|||||||
@@ -164,19 +164,6 @@ router.post("/cell/reveal", authMiddleware, async (req, res) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// POST /api/admin/verify
|
|
||||||
router.post("/admin/verify", (req, res) => {
|
|
||||||
const password = String(req.body?.password ?? "");
|
|
||||||
const adminPwd = process.env.ADMIN_PASSWORD;
|
|
||||||
if (!adminPwd) {
|
|
||||||
return res.status(503).json({ ok: false, error: "not_configured" });
|
|
||||||
}
|
|
||||||
if (password && password === adminPwd) {
|
|
||||||
return res.json({ ok: true });
|
|
||||||
}
|
|
||||||
return res.status(401).json({ ok: false, error: "invalid_password" });
|
|
||||||
});
|
|
||||||
|
|
||||||
// GET /api/econ-scores
|
// GET /api/econ-scores
|
||||||
router.get("/econ-scores", async (_req, res) => {
|
router.get("/econ-scores", async (_req, res) => {
|
||||||
try {
|
try {
|
||||||
|
|||||||
Reference in New Issue
Block a user