Merge pull request 'niklas' (#2) from niklas into master

Reviewed-on: #2
This commit is contained in:
Luca Jakob 2026-05-31 18:24:01 +02:00
commit ba05fb6298
22 changed files with 1889 additions and 2078 deletions

View File

@ -1,92 +0,0 @@
[
{
"name": "Switzerland",
"hint": "Alpine country in Central Europe",
"cities": [
{ "name": "Bern", "x": 48, "y": 58 },
{ "name": "Zürich", "x": 58, "y": 38 },
{ "name": "Geneva", "x": 22, "y": 72 }
]
},
{
"name": "Norway",
"hint": "Scandinavian country with long coastline",
"cities": [
{ "name": "Oslo", "x": 55, "y": 72 },
{ "name": "Bergen", "x": 32, "y": 60 },
{ "name": "Tromsø", "x": 62, "y": 18 }
]
},
{
"name": "Italy",
"hint": "Boot-shaped peninsula in Southern Europe",
"cities": [
{ "name": "Rome", "x": 52, "y": 58 },
{ "name": "Milan", "x": 42, "y": 22 },
{ "name": "Naples", "x": 58, "y": 72 }
]
},
{
"name": "Japan",
"hint": "Island nation in East Asia",
"cities": [
{ "name": "Tokyo", "x": 72, "y": 48 },
{ "name": "Osaka", "x": 58, "y": 58 },
{ "name": "Sapporo", "x": 70, "y": 22 }
]
},
{
"name": "Brazil",
"hint": "Largest country in South America",
"cities": [
{ "name": "Brasília", "x": 58, "y": 52 },
{ "name": "São Paulo", "x": 60, "y": 68 },
{ "name": "Manaus", "x": 38, "y": 38 }
]
},
{
"name": "Australia",
"hint": "Continent and country in the Southern Hemisphere",
"cities": [
{ "name": "Canberra", "x": 72, "y": 72 },
{ "name": "Sydney", "x": 78, "y": 68 },
{ "name": "Perth", "x": 22, "y": 65 }
]
},
{
"name": "France",
"hint": "Country in Western Europe, hexagonal shape",
"cities": [
{ "name": "Paris", "x": 50, "y": 32 },
{ "name": "Lyon", "x": 58, "y": 55 },
{ "name": "Marseille", "x": 58, "y": 72 }
]
},
{
"name": "India",
"hint": "Large peninsula in South Asia",
"cities": [
{ "name": "New Delhi", "x": 46, "y": 28 },
{ "name": "Mumbai", "x": 32, "y": 55 },
{ "name": "Chennai", "x": 52, "y": 72 }
]
},
{
"name": "Canada",
"hint": "Second largest country in the world",
"cities": [
{ "name": "Ottawa", "x": 62, "y": 52 },
{ "name": "Vancouver", "x": 22, "y": 55 },
{ "name": "Toronto", "x": 60, "y": 58 }
]
},
{
"name": "Germany",
"hint": "Central European country",
"cities": [
{ "name": "Berlin", "x": 58, "y": 28 },
{ "name": "Munich", "x": 48, "y": 68 },
{ "name": "Hamburg", "x": 42, "y": 18 }
]
}
]

View File

@ -8,13 +8,16 @@
<link rel="stylesheet" href="./styles/game.css" />
</head>
<body>
<noscript>
<p class="noscript-msg">GeoDraw requires JavaScript. Please enable it in your browser settings.</p>
</noscript>
<div class="wrap">
<header class="header">
<div class="container header__inner">
<a href="./index.html" class="logo">
<div class="logo__mark">
<svg viewBox="0 0 24 24" fill="none">
<svg viewBox="0 0 24 24" fill="none" aria-hidden="true">
<circle cx="12" cy="12" r="9" stroke="white" stroke-width="1.6"/>
<path d="M3 12 Q8 8 12 12 Q16 16 21 12" stroke="white" stroke-width="1.4" fill="none"/>
<path d="M12 3 Q14 8 12 12 Q10 16 12 21" stroke="white" stroke-width="1.4" fill="none"/>
@ -22,7 +25,7 @@
</div>
<span class="logo__text">GeoDraw</span>
</a>
<nav class="nav">
<nav class="nav" aria-label="Main navigation">
<a href="lobby.html" class="nav__link">← Back to lobby</a>
</nav>
</div>
@ -63,7 +66,7 @@
<div class="canvas-area">
<div class="canvas-wrap" id="canvas-wrap">
<canvas id="draw-canvas"></canvas>
<canvas id="draw-canvas" role="application" aria-label="Drawing canvas sketch the country border here"></canvas>
<div id="score-feedback"></div>
</div>
@ -105,7 +108,7 @@
<!-- ── Controls ── -->
<div class="game-controls">
<button id="btn-clear" class="btn btn--ghost btn--sm">
<svg width="14" height="14" viewBox="0 0 14 14" fill="none">
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" aria-hidden="true">
<path d="M2 2l10 10M12 2L2 12" stroke="currentColor" stroke-width="1.8" stroke-linecap="round"/>
</svg>
Clear

View File

@ -4,18 +4,19 @@
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>GeoDraw</title>
<link rel="preconnect" href="https://fonts.googleapis.com">
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
<link href="https://fonts.googleapis.com/css2?family=Syne:wght@400;500;700;800&family=DM+Sans:wght@300;400;500&display=swap" rel="stylesheet">
<link rel="stylesheet" href="styles/main.css" />
<link rel="stylesheet" href="styles/index.css" />
</head>
<body>
<noscript>
<p class="noscript-msg">GeoDraw requires JavaScript. Please enable it in your browser settings.</p>
</noscript>
<div class="wrap">
<header class="header">
<div class="container header__inner">
<a href="#hero" class="logo">
<div class="logo__mark">
<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true">
<circle cx="12" cy="12" r="9" stroke="white" stroke-width="1.6"/>
<path d="M3 12 Q8 8 12 12 Q16 16 21 12" stroke="white" stroke-width="1.4" fill="none"/>
<path d="M12 3 Q14 8 12 12 Q10 16 12 21" stroke="white" stroke-width="1.4" fill="none"/>
@ -24,7 +25,7 @@
<span class="logo__text">GeoDraw</span>
</a>
<nav class="nav">
<nav class="nav" aria-label="Main navigation">
<a href="#hero" class="nav__link">Play</a>
<a href="#about" class="nav__link">How to play</a>
<a href="#register" class="nav__cta">Login / Register</a>
@ -53,7 +54,7 @@
<div class="hero__actions reveal reveal-delay-3">
<a href="#register" class="btn btn--primary">
Start Playing
<svg width="16" height="16" viewBox="0 0 16 16" fill="none"><path d="M3 8h10M9 4l4 4-4 4" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/></svg>
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" aria-hidden="true"><path d="M3 8h10M9 4l4 4-4 4" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/></svg>
</a>
<a href="#about" class="btn btn--ghost">How it works</a>
</div>
@ -171,7 +172,7 @@
<button type="button" class="btn btn--primary btn--full" id="reg-btn">
Create game
<svg width="16" height="16" viewBox="0 0 16 16" fill="none"><path d="M3 8h10M9 4l4 4-4 4" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/></svg>
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" aria-hidden="true"><path d="M3 8h10M9 4l4 4-4 4" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/></svg>
</button>
<p class="form-footer"><a href="#" style="color:var(--sea);font-weight:600;">Join a game instead</a></p>

View File

@ -6,16 +6,19 @@
<title>GeoDraw — Leaderboard</title>
<link rel="stylesheet" href="./styles/main.css" />
<link rel="stylesheet" href="./styles/leader.css" />
</head>
<body>
<noscript>
<p class="noscript-msg">GeoDraw requires JavaScript. Please enable it in your browser settings.</p>
</noscript>
<div class="wrap">
<header class="header">
<div class="container header__inner">
<a href="index.html" class="logo">
<div class="logo__mark">
<svg viewBox="0 0 24 24" fill="none">
<svg viewBox="0 0 24 24" fill="none" aria-hidden="true">
<circle cx="12" cy="12" r="9" stroke="white" stroke-width="1.6"/>
<path d="M3 12 Q8 8 12 12 Q16 16 21 12" stroke="white" stroke-width="1.4" fill="none"/>
<path d="M12 3 Q14 8 12 12 Q10 16 12 21" stroke="white" stroke-width="1.4" fill="none"/>
@ -23,7 +26,7 @@
</div>
<span class="logo__text">GeoDraw</span>
</a>
<nav class="nav">
<nav class="nav" aria-label="Main navigation">
<a href="index.html" class="nav__link">Home</a>
<a href="lobby.html" class="nav__cta">Play again →</a>
</nav>
@ -87,68 +90,6 @@
</div>
<script src="scripts/storage.js"></script>
<script>
document.addEventListener("DOMContentLoaded", () => {
const currentPlayer = Storage.getPlayerName();
const body = document.getElementById("lb-body");
function render() {
const board = Storage.getLeaderboard();
body.innerHTML = "";
if (!board.length) {
body.innerHTML = `
<div class="lb-empty">
<div class="lb-empty__icon">🏁</div>
<p class="lb-empty__text">No games played yet. Be the first on the board!</p>
<a href="lobby.html" class="btn btn--primary btn--sm">Play now →</a>
</div>`;
return;
}
const medals = ["🥇","🥈","🥉"];
const scoreClasses = ["gold","silver","bronze"];
board.forEach((entry, i) => {
const rank = i + 1;
const isYou = entry.name === currentPlayer;
const date = entry.date
? new Date(entry.date).toLocaleDateString("en-CH", { day:"2-digit", month:"short" })
: "—";
const rounds = (entry.scores || []).join(" · ") || "—";
const row = document.createElement("div");
row.className = `lb-row${rank <= 3 ? ` rank-${rank}` : ""}`;
row.style.animationDelay = `${i * 0.05}s`;
row.innerHTML = `
<span class="lb-rank">${rank <= 3 ? `<span class="lb-medal">${medals[rank-1]}</span>` : rank}</span>
<span class="lb-name${isYou ? " is-you" : ""}">${escHtml(entry.name)}</span>
<span class="lb-rounds hide-sm">${escHtml(rounds)}</span>
<span class="lb-date hide-sm">${date}</span>
<span class="lb-score${rank <= 3 ? " " + scoreClasses[rank-1] : ""}">${entry.totalScore}</span>
`;
body.appendChild(row);
});
// Your latest score bar
const latest = board.find(e => e.name === currentPlayer);
if (latest) {
document.getElementById("your-bar").style.display = "flex";
document.getElementById("your-bar-name").textContent = latest.name;
document.getElementById("your-bar-score").textContent = latest.totalScore;
}
}
function escHtml(str) {
return String(str)
.replace(/&/g,"&amp;")
.replace(/</g,"&lt;")
.replace(/>/g,"&gt;");
}
render();
});
</script>
<script src="scripts/leaderboard.js"></script>
</body>
</html>

View File

@ -8,13 +8,16 @@
<link rel="stylesheet" href="./styles/lobby.css" />
</head>
<body>
<noscript>
<p class="noscript-msg">GeoDraw requires JavaScript. Please enable it in your browser settings.</p>
</noscript>
<div class="wrap">
<header class="header">
<div class="container header__inner">
<a href="index.html" class="logo">
<div class="logo__mark">
<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true">
<circle cx="12" cy="12" r="9" stroke="white" stroke-width="1.6"/>
<path d="M3 12 Q8 8 12 12 Q16 16 21 12" stroke="white" stroke-width="1.4" fill="none"/>
<path d="M12 3 Q14 8 12 12 Q10 16 12 21" stroke="white" stroke-width="1.4" fill="none"/>
@ -22,7 +25,7 @@
</div>
<span class="logo__text">GeoDraw</span>
</a>
<nav class="nav">
<nav class="nav" aria-label="Main navigation">
<a href="index.html" class="nav__link">Home</a>
<a href="leaderboard.html" class="nav__link">Leaderboard</a>
</nav>
@ -80,7 +83,7 @@
<button id="btn-start" class="btn btn--primary btn--full">
Start Game
<svg width="16" height="16" viewBox="0 0 16 16" fill="none">
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" aria-hidden="true">
<path d="M3 8h10M9 4l4 4-4 4" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/>
</svg>
</button>
@ -99,7 +102,7 @@
</div>
<div class="footer__center">
<a href="html" class="footer__brand">GeoDraw</a>
<a href="index.html" class="footer__brand">GeoDraw</a>
<p class="footer__sub">Browser geography game</p>
</div>

View File

@ -5,209 +5,19 @@
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>GeoDraw — Results</title>
<link rel="stylesheet" href="styles/main.css" />
<style>
.results-page {
min-height: calc(100vh - var(--hh));
display: grid;
place-items: center;
padding: 60px 0;
}
.results-inner {
width: 100%;
max-width: 680px;
display: grid;
gap: 24px;
}
.results-header {
text-align: center;
}
.results-emoji {
font-size: 3.5rem;
margin-bottom: 16px;
display: block;
animation: popIn .5s cubic-bezier(.34,1.56,.64,1) both;
}
@keyframes popIn {
from { transform: scale(0); opacity: 0; }
to { transform: scale(1); opacity: 1; }
}
.results-title {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: clamp(2rem, 5vw, 3.4rem);
letter-spacing: -.03em;
line-height: 1.05;
margin-bottom: 8px;
}
.results-player {
font-size: 1rem;
color: var(--ink-soft);
}
.results-player strong { color: var(--ink); }
/* Total score */
.score-total-card {
background: linear-gradient(135deg, var(--sea) 0%, #159fd4 50%, var(--leaf) 100%);
border-radius: var(--r-xl);
padding: 36px 40px;
text-align: center;
color: var(--white);
box-shadow: 0 16px 40px rgba(26,127,196,.3);
animation: slideUp .5s .1s cubic-bezier(.34,1.56,.64,1) both;
}
@keyframes slideUp {
from { transform: translateY(30px); opacity: 0; }
to { transform: translateY(0); opacity: 1; }
}
.score-total-label {
font-size: .78rem;
font-weight: 700;
letter-spacing: .1em;
text-transform: uppercase;
opacity: .78;
margin-bottom: 8px;
}
.score-total-num {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: clamp(4rem, 10vw, 6rem);
line-height: 1;
letter-spacing: -.04em;
}
.score-total-max {
font-size: 1.4rem;
opacity: .6;
font-weight: 400;
}
.score-grade {
display: inline-block;
margin-top: 12px;
padding: 4px 18px;
border-radius: 999px;
background: rgba(255,255,255,.22);
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: 1.1rem;
letter-spacing: .04em;
}
/* Round breakdown */
.rounds-card {
background: var(--white);
border: 1px solid var(--line);
border-radius: var(--r-xl);
padding: 28px 32px;
box-shadow: var(--shadow-sm);
animation: slideUp .5s .2s cubic-bezier(.34,1.56,.64,1) both;
}
.rounds-card-title {
font-family: 'Syne', sans-serif;
font-weight: 700;
font-size: 1rem;
letter-spacing: -.01em;
margin-bottom: 20px;
color: var(--ink-soft);
}
.round-rows { display: grid; gap: 14px; }
.round-row {
display: grid;
grid-template-columns: auto 1fr auto;
align-items: center;
gap: 16px;
}
.round-row__label {
font-size: .82rem;
font-weight: 600;
color: var(--ink-soft);
white-space: nowrap;
}
.round-row__bar-wrap {
height: 8px;
background: var(--cream);
border-radius: 999px;
overflow: hidden;
}
.round-row__bar {
height: 100%;
border-radius: 999px;
background: linear-gradient(90deg, var(--sea), var(--leaf));
transition: width .8s cubic-bezier(.34,1.56,.64,1);
}
.round-row__score {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: .9rem;
color: var(--sea);
min-width: 36px;
text-align: right;
}
/* Country list */
.countries-row {
display: flex;
flex-wrap: wrap;
gap: 8px;
margin-top: 8px;
}
.country-tag {
padding: 5px 12px;
border-radius: 999px;
background: var(--sea-dim);
color: var(--sea);
font-size: .8rem;
font-weight: 600;
}
/* Actions */
.results-actions {
display: flex;
gap: 12px;
flex-wrap: wrap;
animation: slideUp .5s .3s cubic-bezier(.34,1.56,.64,1) both;
}
.results-actions .btn { flex: 1 1 180px; }
@media (max-width: 600px) {
.results-page { padding: 40px 0; }
.score-total-card { padding: 28px 24px; }
.rounds-card { padding: 22px 20px; }
.results-actions .btn { flex-basis: 100%; }
}
@media (max-width: 360px) {
.score-total-num { font-size: 3.4rem; }
}
</style>
<link rel="stylesheet" href="styles/results.css" />
</head>
<body>
<noscript>
<p class="noscript-msg">GeoDraw requires JavaScript. Please enable it in your browser settings.</p>
</noscript>
<div class="wrap">
<header class="header">
<div class="container header__inner">
<a href="index.html" class="logo">
<div class="logo__mark">
<svg viewBox="0 0 24 24" fill="none">
<svg viewBox="0 0 24 24" fill="none" aria-hidden="true">
<circle cx="12" cy="12" r="9" stroke="white" stroke-width="1.6"/>
<path d="M3 12 Q8 8 12 12 Q16 16 21 12" stroke="white" stroke-width="1.4" fill="none"/>
<path d="M12 3 Q14 8 12 12 Q10 16 12 21" stroke="white" stroke-width="1.4" fill="none"/>
@ -215,7 +25,7 @@
</div>
<span class="logo__text">GeoDraw</span>
</a>
<nav class="nav">
<nav class="nav" aria-label="Main navigation">
<a href="index.html" class="nav__link">Home</a>
<a href="leaderboard.html" class="nav__link">Leaderboard</a>
<a href="lobby.html" class="nav__cta">Play again →</a>
@ -288,69 +98,6 @@
<script src="scripts/storage.js"></script>
<script src="scripts/scoring.js"></script>
<script>
document.addEventListener("DOMContentLoaded", () => {
const state = Storage.getGameState();
const name = Storage.getPlayerName();
document.getElementById("results-player").textContent = name;
if (!state) {
document.getElementById("results-title").textContent = "No game found.";
document.getElementById("total-score").textContent = "—";
return;
}
const { scores, totalScore, countries } = state;
// Emoji + title based on score
const avg = totalScore / 3;
let emoji = "🌍", title = "Not bad!";
if (avg >= 85) { emoji = "🔥"; title = "Incredible!"; }
else if (avg >= 70) { emoji = "🎉"; title = "Well done!"; }
else if (avg >= 50) { emoji = "👏"; title = "Good effort!"; }
else { emoji = "😅"; title = "Keep practising!"; }
document.getElementById("results-emoji").textContent = emoji;
document.getElementById("results-title").textContent = title;
document.getElementById("total-score").textContent = totalScore;
// Grade
const grade = Scoring.getGrade(avg);
const gradeEl = document.getElementById("total-grade");
gradeEl.textContent = `Grade ${grade.label}`;
// Round rows
const rowsContainer = document.getElementById("round-rows");
(scores || []).forEach((s, i) => {
const row = document.createElement("div");
row.className = "round-row";
row.innerHTML = `
<span class="round-row__label">Round ${i + 1}</span>
<div class="round-row__bar-wrap">
<div class="round-row__bar" style="width:0%" data-target="${s}"></div>
</div>
<span class="round-row__score">${s}%</span>
`;
rowsContainer.appendChild(row);
});
// Animate bars
requestAnimationFrame(() => {
document.querySelectorAll(".round-row__bar").forEach(bar => {
bar.style.width = bar.dataset.target + "%";
});
});
// Country tags
const tagsContainer = document.getElementById("countries-row");
(countries || []).forEach(c => {
const tag = document.createElement("span");
tag.className = "country-tag";
tag.textContent = c;
tagsContainer.appendChild(tag);
});
});
</script>
<script src="scripts/results.js"></script>
</body>
</html>

View File

@ -1,116 +1,136 @@
// countries.js — country data helpers (inline, no fetch needed)
// countries.js — country data and helpers
/**
* @typedef {{ name: string, x: number, y: number }} City
* @typedef {{ name: string, hint: string, cities: City[] }} Country
*/
const Countries = (() => {
const COUNTRIES_DATA = [
{
"name": "Switzerland",
"hint": "Alpine country in Central Europe",
"cities": [
{ "name": "Bern", "x": 48, "y": 58 },
{ "name": "Zürich", "x": 58, "y": 38 },
{ "name": "Geneva", "x": 22, "y": 72 }
]
},
{
"name": "Norway",
"hint": "Scandinavian country with long coastline",
"cities": [
{ "name": "Oslo", "x": 55, "y": 72 },
{ "name": "Bergen", "x": 32, "y": 60 },
{ "name": "Tromsø", "x": 62, "y": 18 }
]
},
{
"name": "Italy",
"hint": "Boot-shaped peninsula in Southern Europe",
"cities": [
{ "name": "Rome", "x": 52, "y": 58 },
{ "name": "Milan", "x": 42, "y": 22 },
{ "name": "Naples", "x": 58, "y": 72 }
]
},
{
"name": "Japan",
"hint": "Island nation in East Asia",
"cities": [
{ "name": "Tokyo", "x": 72, "y": 48 },
{ "name": "Osaka", "x": 58, "y": 58 },
{ "name": "Sapporo", "x": 70, "y": 22 }
]
},
{
"name": "Brazil",
"hint": "Largest country in South America",
"cities": [
{ "name": "Brasília", "x": 58, "y": 52 },
{ "name": "São Paulo", "x": 60, "y": 68 },
{ "name": "Manaus", "x": 38, "y": 38 }
]
},
{
"name": "Australia",
"hint": "Continent and country in the Southern Hemisphere",
"cities": [
{ "name": "Canberra", "x": 72, "y": 72 },
{ "name": "Sydney", "x": 78, "y": 68 },
{ "name": "Perth", "x": 22, "y": 65 }
]
},
{
"name": "France",
"hint": "Western Europe, roughly hexagonal shape",
"cities": [
{ "name": "Paris", "x": 50, "y": 32 },
{ "name": "Lyon", "x": 58, "y": 55 },
{ "name": "Marseille", "x": 58, "y": 72 }
]
},
{
"name": "India",
"hint": "Large peninsula in South Asia",
"cities": [
{ "name": "New Delhi", "x": 46, "y": 28 },
{ "name": "Mumbai", "x": 32, "y": 55 },
{ "name": "Chennai", "x": 52, "y": 72 }
]
},
{
"name": "Canada",
"hint": "Second largest country in the world",
"cities": [
{ "name": "Ottawa", "x": 62, "y": 52 },
{ "name": "Vancouver", "x": 22, "y": 55 },
{ "name": "Toronto", "x": 60, "y": 58 }
]
},
{
"name": "Germany",
"hint": "Central European country",
"cities": [
{ "name": "Berlin", "x": 58, "y": 28 },
{ "name": "Munich", "x": 48, "y": 68 },
{ "name": "Hamburg", "x": 42, "y": 18 }
]
}
];
/** @type {Country[]} */
const COUNTRIES_DATA = [
{
name: "Switzerland",
hint: "Alpine country in Central Europe",
cities: [
{ name: "Bern", x: 48, y: 58 },
{ name: "Zürich", x: 58, y: 38 },
{ name: "Geneva", x: 22, y: 72 },
],
},
{
name: "Norway",
hint: "Scandinavian country with long coastline",
cities: [
{ name: "Oslo", x: 55, y: 72 },
{ name: "Bergen", x: 32, y: 60 },
{ name: "Tromsø", x: 62, y: 18 },
],
},
{
name: "Italy",
hint: "Boot-shaped peninsula in Southern Europe",
cities: [
{ name: "Rome", x: 52, y: 58 },
{ name: "Milan", x: 42, y: 22 },
{ name: "Naples", x: 58, y: 72 },
],
},
{
name: "Japan",
hint: "Island nation in East Asia",
cities: [
{ name: "Tokyo", x: 72, y: 48 },
{ name: "Osaka", x: 58, y: 58 },
{ name: "Sapporo", x: 70, y: 22 },
],
},
{
name: "Brazil",
hint: "Largest country in South America",
cities: [
{ name: "Brasília", x: 58, y: 52 },
{ name: "São Paulo", x: 60, y: 68 },
{ name: "Manaus", x: 38, y: 38 },
],
},
{
name: "Australia",
hint: "Continent and country in the Southern Hemisphere",
cities: [
{ name: "Canberra", x: 72, y: 72 },
{ name: "Sydney", x: 78, y: 68 },
{ name: "Perth", x: 22, y: 65 },
],
},
{
name: "France",
hint: "Western Europe, roughly hexagonal shape",
cities: [
{ name: "Paris", x: 50, y: 32 },
{ name: "Lyon", x: 58, y: 55 },
{ name: "Marseille", x: 58, y: 72 },
],
},
{
name: "India",
hint: "Large peninsula in South Asia",
cities: [
{ name: "New Delhi", x: 46, y: 28 },
{ name: "Mumbai", x: 32, y: 55 },
{ name: "Chennai", x: 52, y: 72 },
],
},
{
name: "Canada",
hint: "Second largest country in the world",
cities: [
{ name: "Ottawa", x: 62, y: 52 },
{ name: "Vancouver", x: 22, y: 55 },
{ name: "Toronto", x: 60, y: 58 },
],
},
{
name: "Germany",
hint: "Central European country",
cities: [
{ name: "Berlin", x: 58, y: 28 },
{ name: "Munich", x: 48, y: 68 },
{ name: "Hamburg", x: 42, y: 18 },
],
},
];
let _data = [];
/** @type {Country[]} */
let _data = [];
async function loadCountries() {
if (_data.length) return _data;
_data = COUNTRIES_DATA;
return _data;
}
/**
* Load country data into memory. Safe to call multiple times.
* Returns a Promise for future compatibility with a real API fetch.
* @returns {Promise<Country[]>}
*/
function loadCountries() {
if (!_data.length) _data = COUNTRIES_DATA;
return Promise.resolve(_data);
}
function getRandomCountries(count = 3) {
const shuffled = [..._data].sort(() => Math.random() - 0.5);
return shuffled.slice(0, count);
}
/**
* Return a random subset of countries.
* @param {number} count
* @returns {Country[]}
*/
function getRandomCountries(count = 3) {
return [..._data].sort(() => Math.random() - 0.5).slice(0, count);
}
function getCities(countryName) {
const c = _data.find(c => c.name === countryName);
return c ? c.cities : [];
}
/**
* Get city list for a specific country by name.
* @param {string} countryName
* @returns {City[]}
*/
function getCities(countryName) {
const country = _data.find((c) => c.name === countryName);
return country ? country.cities : [];
}
return { loadCountries, getRandomCountries, getCities };
return { loadCountries, getRandomCountries, getCities };
})();

View File

@ -1,148 +1,172 @@
// drawing.js — canvas drawing module
const Drawing = (() => {
let canvas, ctx;
let isDrawing = false;
let points = []; // [{x, y}, ...]
let cities = []; // [{name, x, y}, ...] (% coords)
/** @type {HTMLCanvasElement} */
let canvas;
/** @type {CanvasRenderingContext2D} */
let ctx;
const STROKE_COLOR = "#1a7fc4";
const STROKE_WIDTH = 2.5;
let isDrawing = false;
/** @type {{ x: number, y: number }[]} */
let points = [];
/** @type {{ name: string, x: number, y: number }[]} */
let cities = [];
function init(canvasEl) {
canvas = canvasEl;
ctx = canvas.getContext("2d");
const STROKE_COLOR = "#1a7fc4";
const STROKE_WIDTH = 2.5;
// Pointer events (mouse + touch)
canvas.addEventListener("pointerdown", onDown);
canvas.addEventListener("pointermove", onMove);
canvas.addEventListener("pointerup", onUp);
canvas.addEventListener("pointerleave", onUp);
canvas.style.touchAction = "none";
/**
* Initialise the drawing module on a canvas element.
* @param {HTMLCanvasElement} canvasEl
*/
function init(canvasEl) {
canvas = canvasEl;
ctx = canvas.getContext("2d");
_resize();
window.addEventListener("resize", _resize);
}
canvas.addEventListener("pointerdown", onDown);
canvas.addEventListener("pointermove", onMove);
canvas.addEventListener("pointerup", onUp);
canvas.addEventListener("pointerleave", onUp);
canvas.style.touchAction = "none";
function _resize() {
if (!canvas) return;
const { width, height } = canvas.getBoundingClientRect();
canvas.width = width * window.devicePixelRatio;
canvas.height = height * window.devicePixelRatio;
ctx.scale(window.devicePixelRatio, window.devicePixelRatio);
_redraw();
}
_resize();
window.addEventListener("resize", _resize);
}
function _pos(e) {
const r = canvas.getBoundingClientRect();
return {
x: (e.clientX - r.left),
y: (e.clientY - r.top),
};
}
/** Resize canvas to match its CSS size, accounting for device pixel ratio. */
function _resize() {
if (!canvas) return;
const { width, height } = canvas.getBoundingClientRect();
canvas.width = width * window.devicePixelRatio;
canvas.height = height * window.devicePixelRatio;
ctx.scale(window.devicePixelRatio, window.devicePixelRatio);
_redraw();
}
function onDown(e) {
e.preventDefault();
isDrawing = true;
const p = _pos(e);
points.push(p);
ctx.beginPath();
ctx.moveTo(p.x, p.y);
}
/**
* Convert a pointer event to canvas-local coordinates.
* @param {PointerEvent} e
* @returns {{ x: number, y: number }}
*/
function _pos(e) {
const rect = canvas.getBoundingClientRect();
return { x: e.clientX - rect.left, y: e.clientY - rect.top };
}
function onMove(e) {
if (!isDrawing) return;
e.preventDefault();
const p = _pos(e);
points.push(p);
ctx.lineTo(p.x, p.y);
ctx.strokeStyle = STROKE_COLOR;
ctx.lineWidth = STROKE_WIDTH;
ctx.lineJoin = "round";
ctx.lineCap = "round";
ctx.stroke();
}
/** @param {PointerEvent} e */
function onDown(e) {
e.preventDefault();
isDrawing = true;
const p = _pos(e);
points.push(p);
ctx.beginPath();
ctx.moveTo(p.x, p.y);
}
function onUp(e) {
if (!isDrawing) return;
isDrawing = false;
e.preventDefault();
}
/** @param {PointerEvent} e */
function onMove(e) {
if (!isDrawing) return;
e.preventDefault();
const p = _pos(e);
points.push(p);
ctx.lineTo(p.x, p.y);
ctx.strokeStyle = STROKE_COLOR;
ctx.lineWidth = STROKE_WIDTH;
ctx.lineJoin = "round";
ctx.lineCap = "round";
ctx.stroke();
}
function clear() {
points = [];
if (!ctx) return;
const w = canvas.getBoundingClientRect().width;
const h = canvas.getBoundingClientRect().height;
ctx.clearRect(0, 0, w, h);
_drawCities();
}
/** @param {PointerEvent} e */
function onUp(e) {
if (!isDrawing) return;
isDrawing = false;
e.preventDefault();
}
function setCities(cityList) {
cities = cityList || [];
_drawCities();
}
/** Clear the canvas and redraw city markers. */
function clear() {
points = [];
if (!ctx) return;
const { width, height } = canvas.getBoundingClientRect();
ctx.clearRect(0, 0, width, height);
_drawCities();
}
function _drawCities() {
if (!ctx || !cities.length) return;
const w = canvas.getBoundingClientRect().width;
const h = canvas.getBoundingClientRect().height;
/**
* Set city markers to display on the canvas.
* @param {{ name: string, x: number, y: number }[]} cityList - Coords in percent (0100).
*/
function setCities(cityList) {
cities = cityList || [];
_drawCities();
}
cities.forEach(city => {
const cx = (city.x / 100) * w;
const cy = (city.y / 100) * h;
/** Render all city markers with labels. */
function _drawCities() {
if (!ctx || !cities.length) return;
const { width, height } = canvas.getBoundingClientRect();
// Dot
ctx.beginPath();
ctx.arc(cx, cy, 5, 0, Math.PI * 2);
ctx.fillStyle = "rgba(240,180,40,0.9)";
ctx.fill();
ctx.strokeStyle = "rgba(255,255,255,0.9)";
ctx.lineWidth = 1.5;
ctx.stroke();
cities.forEach((city) => {
const cx = (city.x / 100) * width;
const cy = (city.y / 100) * height;
// Label
ctx.font = "bold 11px 'DM Sans', sans-serif";
ctx.fillStyle = "#0b1f2a";
ctx.textAlign = "center";
// Dot
ctx.beginPath();
ctx.arc(cx, cy, 5, 0, Math.PI * 2);
ctx.fillStyle = "rgba(240,180,40,0.9)";
ctx.fill();
ctx.strokeStyle = "rgba(255,255,255,0.9)";
ctx.lineWidth = 1.5;
ctx.stroke();
// White pill background
const textW = ctx.measureText(city.name).width + 10;
const textH = 16;
const tx = cx;
const ty = cy - 14;
// White pill label background
ctx.font = "bold 11px 'DM Sans', sans-serif";
const textW = ctx.measureText(city.name).width + 10;
const textH = 16;
const tx = cx;
const ty = cy - 14;
ctx.save();
ctx.fillStyle = "rgba(255,255,255,0.88)";
ctx.beginPath();
ctx.roundRect(tx - textW / 2, ty - textH / 2 - 1, textW, textH, 4);
ctx.fill();
ctx.restore();
ctx.save();
ctx.fillStyle = "rgba(255,255,255,0.88)";
ctx.beginPath();
ctx.roundRect(tx - textW / 2, ty - textH / 2 - 1, textW, textH, 4);
ctx.fill();
ctx.restore();
ctx.fillStyle = "#0b1f2a";
ctx.fillText(city.name, tx, ty + 4);
});
}
ctx.fillStyle = "#0b1f2a";
ctx.textAlign = "center";
ctx.fillText(city.name, tx, ty + 4);
});
}
function _redraw() {
_drawCities();
if (!points.length) return;
ctx.beginPath();
ctx.strokeStyle = STROKE_COLOR;
ctx.lineWidth = STROKE_WIDTH;
ctx.lineJoin = "round";
ctx.lineCap = "round";
points.forEach((p, i) => i === 0 ? ctx.moveTo(p.x, p.y) : ctx.lineTo(p.x, p.y));
ctx.stroke();
}
/** Redraw the full stroke from stored points. */
function _redraw() {
_drawCities();
if (!points.length) return;
ctx.beginPath();
ctx.strokeStyle = STROKE_COLOR;
ctx.lineWidth = STROKE_WIDTH;
ctx.lineJoin = "round";
ctx.lineCap = "round";
points.forEach((p, i) =>
i === 0 ? ctx.moveTo(p.x, p.y) : ctx.lineTo(p.x, p.y),
);
ctx.stroke();
}
function getPoints() {
return [...points];
}
/**
* Return a copy of the current drawn points.
* @returns {{ x: number, y: number }[]}
*/
function getPoints() {
return [...points];
}
function destroy() {
window.removeEventListener("resize", _resize);
}
/** Remove event listeners and clean up. */
function destroy() {
window.removeEventListener("resize", _resize);
}
return { init, clear, setCities, getPoints, destroy };
return { init, clear, setCities, getPoints, destroy };
})();

View File

@ -1,153 +1,158 @@
// game.js — round management, timer, submit
const TOTAL_ROUNDS = 3;
const TOTAL_ROUNDS = 3;
const ROUND_DURATION = 60; // seconds
/** @type {import('./countries.js').Country[]} */
let roundCountries = [];
let currentRound = 0;
let scores = [];
let timerInterval = null;
let timeLeft = ROUND_DURATION;
let currentRound = 0;
/** @type {number[]} */
let scores = [];
let timerInterval = null;
let timeLeft = ROUND_DURATION;
// ── DOM refs
const elCountryName = document.getElementById("country-name");
const elCountryHint = document.getElementById("country-hint");
const elRoundNum = document.getElementById("round-num");
const elTimerNum = document.getElementById("timer-num");
const elTimerBar = document.getElementById("timer-bar");
const elBtnClear = document.getElementById("btn-clear");
const elBtnSubmit = document.getElementById("btn-submit");
const elCanvas = document.getElementById("draw-canvas");
const elRoundNum = document.getElementById("round-num");
const elTimerNum = document.getElementById("timer-num");
const elTimerBar = document.getElementById("timer-bar");
const elTimerWrap = document.querySelector(".game-timer");
const elBtnClear = document.getElementById("btn-clear");
const elBtnSubmit = document.getElementById("btn-submit");
// ── Init
/** Load countries and start the first round. */
async function initGame() {
await Countries.loadCountries();
roundCountries = Countries.getRandomCountries(TOTAL_ROUNDS);
currentRound = 0;
scores = [];
startRound();
await Countries.loadCountries();
roundCountries = Countries.getRandomCountries(TOTAL_ROUNDS);
currentRound = 0;
scores = [];
startRound();
}
// ── Round
/** Set up UI and timer for the current round. */
function startRound() {
const country = roundCountries[currentRound];
const country = roundCountries[currentRound];
// Update UI
elRoundNum.textContent = currentRound + 1;
elCountryName.textContent = country.name;
elCountryHint.textContent = country.hint || "";
elRoundNum.textContent = currentRound + 1;
elCountryName.textContent = country.name;
elCountryHint.textContent = country.hint || "";
// Round pips
if (typeof window.updateRoundPips === "function") {
window.updateRoundPips(currentRound + 1);
}
if (typeof window.updateRoundPips === "function") {
window.updateRoundPips(currentRound + 1);
}
// Reset canvas placeholder
document.getElementById("canvas-wrap")?.classList.remove("has-drawing");
document.getElementById("canvas-wrap")?.classList.remove("has-drawing");
// Cities on canvas
Drawing.clear();
Drawing.setCities(country.cities || []);
Drawing.clear();
Drawing.setCities(country.cities || []);
// Timer
timeLeft = ROUND_DURATION;
updateTimerUI();
clearInterval(timerInterval);
timerInterval = setInterval(tickTimer, 1000);
timeLeft = ROUND_DURATION;
updateTimerUI();
clearInterval(timerInterval);
timerInterval = setInterval(tickTimer, 1000);
// Button state
elBtnSubmit.disabled = false;
elBtnSubmit.textContent = currentRound < TOTAL_ROUNDS - 1
? "Submit & Next Round →"
: "Submit & See Results →";
elBtnSubmit.disabled = false;
elBtnSubmit.textContent =
currentRound < TOTAL_ROUNDS - 1
? "Submit & Next Round →"
: "Submit & See Results →";
}
/** Decrement timer by one second and auto-submit when time runs out. */
function tickTimer() {
timeLeft--;
updateTimerUI();
if (timeLeft <= 0) {
clearInterval(timerInterval);
submitRound(true); // auto-submit
}
timeLeft--;
updateTimerUI();
if (timeLeft <= 0) {
clearInterval(timerInterval);
submitRound(true);
}
}
/**
* Sync timer bar width and apply urgency CSS classes.
* Uses `.timer--warning` and `.timer--danger` instead of inline styles.
*/
function updateTimerUI() {
elTimerNum.textContent = timeLeft;
const pct = (timeLeft / ROUND_DURATION) * 100;
elTimerBar.style.width = pct + "%";
elTimerNum.textContent = timeLeft;
elTimerBar.style.width = `${(timeLeft / ROUND_DURATION) * 100}%`;
// Colour shift
if (timeLeft <= 10) {
elTimerBar.style.background = "#e05c5c";
elTimerNum.style.color = "#e05c5c";
} else if (timeLeft <= 20) {
elTimerBar.style.background = "#f0b429";
elTimerNum.style.color = "#f0b429";
} else {
elTimerBar.style.background = "";
elTimerNum.style.color = "";
}
elTimerWrap.classList.toggle("timer--danger", timeLeft <= 10);
elTimerWrap.classList.toggle(
"timer--warning",
timeLeft > 10 && timeLeft <= 20,
);
}
/**
* Submit the current round, record the score, and advance or finish.
* @param {boolean} [auto=false] - True when triggered by timer expiry.
*/
function submitRound(auto = false) {
clearInterval(timerInterval);
elBtnSubmit.disabled = true;
clearInterval(timerInterval);
elBtnSubmit.disabled = true;
const points = Drawing.getPoints();
const score = Scoring.calculateScore(points);
scores.push(score);
const points = Drawing.getPoints();
const score = Scoring.calculateScore(points);
scores.push(score);
// Update sidebar score row
if (typeof window.updateScoreDisplay === "function") {
window.updateScoreDisplay(currentRound, score);
}
if (typeof window.updateScoreDisplay === "function") {
window.updateScoreDisplay(currentRound, score);
}
// Flash score feedback
showScoreFeedback(score);
showScoreFeedback(score);
const delay = auto ? 400 : 1200;
setTimeout(() => {
if (currentRound < TOTAL_ROUNDS - 1) {
currentRound++;
startRound();
} else {
finishGame();
}
}, delay);
setTimeout(
() => {
if (currentRound < TOTAL_ROUNDS - 1) {
currentRound++;
startRound();
} else {
finishGame();
}
},
auto ? 400 : 1200,
);
}
/**
* Briefly display the score grade overlay on the canvas.
* @param {number} score
*/
function showScoreFeedback(score) {
const grade = Scoring.getGrade(score);
const el = document.getElementById("score-feedback");
el.textContent = `${score}% ${grade.label}`;
el.style.color = grade.color;
el.style.opacity = "1";
el.style.transform = "translateY(0)";
setTimeout(() => {
el.style.opacity = "0";
el.style.transform = "translateY(-10px)";
}, 900);
const grade = Scoring.getGrade(score);
const el = document.getElementById("score-feedback");
el.textContent = `${score}% ${grade.label}`;
el.style.color = grade.color;
el.style.opacity = "1";
el.style.transform = "translateY(0)";
setTimeout(() => {
el.style.opacity = "0";
el.style.transform = "translateY(-10px)";
}, 900);
}
/** Persist game state, update leaderboard, and navigate to results. */
function finishGame() {
const totalScore = scores.reduce((a, b) => a + b, 0);
const state = {
currentRound: TOTAL_ROUNDS,
scores,
totalScore,
countries: roundCountries.map(c => c.name),
};
Storage.saveGameState(state);
// Save to leaderboard
Storage.saveLeaderboard({
name: Storage.getPlayerName(),
totalScore,
scores,
date: new Date().toISOString(),
});
location.href = "results.html";
const totalScore = scores.reduce((sum, s) => sum + s, 0);
const state = {
currentRound: TOTAL_ROUNDS,
scores,
totalScore,
countries: roundCountries.map((c) => c.name),
};
Storage.saveGameState(state);
Storage.saveLeaderboard({
name: Storage.getPlayerName(),
totalScore,
scores,
date: new Date().toISOString(),
});
location.href = "results.html";
}
// ── Events

View File

@ -1,6 +1,15 @@
// index.js — landing page logic
/**
* Handle the "Create game" button click.
* Validates the lobby name, persists it, and navigates to the lobby page.
*/
document.getElementById("reg-btn")?.addEventListener("click", () => {
const lobbyInput = document.getElementById("username");
const lobbyInput = /** @type {HTMLInputElement|null} */ (
document.getElementById("username")
);
const lobbyName = lobbyInput ? lobbyInput.value.trim() : "";
if (lobbyName) {
Storage.saveLobbyName(lobbyName);
window.location.href = "lobby.html";
@ -9,19 +18,18 @@ document.getElementById("reg-btn")?.addEventListener("click", () => {
}
});
// Scroll reveal — animate elements into view as they enter the viewport
const reveals = document.querySelectorAll(".reveal");
const io = new IntersectionObserver(
const observer = new IntersectionObserver(
(entries) => {
entries.forEach((entry) => {
if (entry.isIntersecting) {
entry.target.classList.add("visible");
io.unobserve(entry.target);
observer.unobserve(entry.target);
}
});
},
{ threshold: 0.12 },
);
reveals.forEach((el) => {
io.observe(el);
});
reveals.forEach((el) => observer.observe(el));

View File

@ -0,0 +1,72 @@
// leaderboard.js — leaderboard page logic
document.addEventListener("DOMContentLoaded", () => {
const currentPlayer = Storage.getPlayerName();
const body = document.getElementById("lb-body");
/**
* Escape HTML special characters to prevent XSS.
* @param {string} str - Raw user-provided string.
* @returns {string} HTML-safe string.
*/
function escHtml(str) {
return String(str)
.replace(/&/g, "&amp;")
.replace(/</g, "&lt;")
.replace(/>/g, "&gt;");
}
/** Render the leaderboard from localStorage data. */
function render() {
const board = Storage.getLeaderboard();
body.innerHTML = "";
if (!board.length) {
body.innerHTML = `
<div class="lb-empty">
<div class="lb-empty__icon">🏁</div>
<p class="lb-empty__text">No games played yet. Be the first on the board!</p>
<a href="lobby.html" class="btn btn--primary btn--sm">Play now </a>
</div>`;
return;
}
const medals = ["🥇", "🥈", "🥉"];
const scoreClasses = ["gold", "silver", "bronze"];
board.forEach((entry, index) => {
const rank = index + 1;
const isYou = entry.name === currentPlayer;
const date = entry.date
? new Date(entry.date).toLocaleDateString("en-CH", {
day: "2-digit",
month: "short",
})
: "—";
const rounds = (entry.scores || []).join(" · ") || "—";
const row = document.createElement("div");
row.className = `lb-row${rank <= 3 ? ` rank-${rank}` : ""}`;
row.style.animationDelay = `${index * 0.05}s`;
row.innerHTML = `
<span class="lb-rank">${rank <= 3 ? `<span class="lb-medal">${medals[rank - 1]}</span>` : rank}</span>
<span class="lb-name${isYou ? " is-you" : ""}">${escHtml(entry.name)}</span>
<span class="lb-rounds hide-sm">${escHtml(rounds)}</span>
<span class="lb-date hide-sm">${date}</span>
<span class="lb-score${rank <= 3 ? ` ${scoreClasses[rank - 1]}` : ""}">${entry.totalScore}</span>
`;
body.appendChild(row);
});
// Show the current player's latest score bar
const latest = board.find((entry) => entry.name === currentPlayer);
if (latest) {
document.getElementById("your-bar").style.display = "flex";
document.getElementById("your-bar-name").textContent = latest.name;
document.getElementById("your-bar-score").textContent = latest.totalScore;
}
}
render();
});

View File

@ -1,47 +1,55 @@
// lobby.js
// lobby.js — lobby page logic
document.addEventListener("DOMContentLoaded", () => {
const input = document.getElementById("username");
const btn = document.getElementById("btn-start");
const errMsg = document.getElementById("name-error");
const input = document.getElementById("username");
const btn = document.getElementById("btn-start");
const errMsg = document.getElementById("name-error");
// Show lobby name
const lobbyNameEl = document.getElementById("lobby-name-display");
if (lobbyNameEl) {
lobbyNameEl.textContent = Storage.getLobbyName();
}
// Display the current lobby name
const lobbyNameEl = document.getElementById("lobby-name-display");
if (lobbyNameEl) {
lobbyNameEl.textContent = Storage.getLobbyName();
}
btn.addEventListener("click", () => {
const name = input.value.trim();
if (!name) {
errMsg.textContent = "Please enter a username to continue.";
input.classList.add("input--error");
input.focus();
return;
}
if (name.length < 2) {
errMsg.textContent = "Username must be at least 2 characters.";
input.classList.add("input--error");
input.focus();
return;
}
Storage.savePlayerName(name);
Storage.clearGameState();
location.href = "game.html";
});
/**
* Validate the username input and navigate to the game.
* Shows an inline error message on invalid input.
*/
btn.addEventListener("click", () => {
const name = input.value.trim();
input.addEventListener("input", () => {
errMsg.textContent = "";
input.classList.remove("input--error");
});
if (!name) {
errMsg.textContent = "Please enter a username to continue.";
input.classList.add("input--error");
input.focus();
return;
}
if (name.length < 2) {
errMsg.textContent = "Username must be at least 2 characters.";
input.classList.add("input--error");
input.focus();
return;
}
input.addEventListener("keydown", (e) => {
if (e.key === "Enter") btn.click();
});
Storage.savePlayerName(name);
Storage.clearGameState();
location.href = "game.html";
});
// Pre-fill if returning player
const existing = Storage.getPlayerName();
if (existing && existing !== "Anonymous") {
input.value = existing;
}
// Clear validation state on every keystroke
input.addEventListener("input", () => {
errMsg.textContent = "";
input.classList.remove("input--error");
});
// Allow form submission via Enter key
input.addEventListener("keydown", (e) => {
if (e.key === "Enter") btn.click();
});
// Pre-fill username if the player has played before
const existing = Storage.getPlayerName();
if (existing && existing !== "Anonymous") {
input.value = existing;
}
});

View File

@ -1,36 +0,0 @@
// main.js — index.html
document.addEventListener("DOMContentLoaded", () => {
// Smooth scroll for nav links
document.querySelectorAll('a[href^="#"]').forEach(link => {
link.addEventListener("click", e => {
const id = link.getAttribute("href").slice(1);
const target = document.getElementById(id);
if (!target) return;
e.preventDefault();
const headerH = document.querySelector(".header")?.offsetHeight || 0;
const top = target.getBoundingClientRect().top + window.scrollY - headerH - 16;
window.scrollTo({ top, behavior: "smooth" });
});
});
// Register button stub
const regBtn = document.getElementById("reg-btn");
if (regBtn) {
regBtn.addEventListener("click", () => {
alert("Frontend skeleton only. Backend will be connected later.");
});
}
// Scroll reveal
const reveals = document.querySelectorAll(".reveal");
const io = new IntersectionObserver((entries) => {
entries.forEach(entry => {
if (entry.isIntersecting) {
entry.target.classList.add("visible");
io.unobserve(entry.target);
}
});
}, { threshold: 0.12 });
reveals.forEach(el => io.observe(el));
});

View File

@ -0,0 +1,74 @@
// results.js — results page logic
document.addEventListener("DOMContentLoaded", () => {
const state = Storage.getGameState();
const name = Storage.getPlayerName();
document.getElementById("results-player").textContent = name;
if (!state) {
document.getElementById("results-title").textContent = "No game found.";
document.getElementById("total-score").textContent = "—";
return;
}
const { scores, totalScore, countries } = state;
// Determine emoji and title based on average score
const avg = totalScore / 3;
let emoji = "🌍";
let title = "Not bad!";
if (avg >= 85) {
emoji = "🔥";
title = "Incredible!";
} else if (avg >= 70) {
emoji = "🎉";
title = "Well done!";
} else if (avg >= 50) {
emoji = "👏";
title = "Good effort!";
} else {
emoji = "😅";
title = "Keep practising!";
}
document.getElementById("results-emoji").textContent = emoji;
document.getElementById("results-title").textContent = title;
document.getElementById("total-score").textContent = totalScore;
// Grade badge
const grade = Scoring.getGrade(avg);
const gradeEl = document.getElementById("total-grade");
gradeEl.textContent = `Grade ${grade.label}`;
// Round breakdown rows
const rowsContainer = document.getElementById("round-rows");
(scores || []).forEach((score, index) => {
const row = document.createElement("div");
row.className = "round-row";
row.innerHTML = `
<span class="round-row__label">Round ${index + 1}</span>
<div class="round-row__bar-wrap">
<div class="round-row__bar" style="width:0%" data-target="${score}"></div>
</div>
<span class="round-row__score">${score}%</span>
`;
rowsContainer.appendChild(row);
});
// Animate score bars on next frame so CSS transition fires
requestAnimationFrame(() => {
document.querySelectorAll(".round-row__bar").forEach((bar) => {
bar.style.width = `${bar.dataset.target}%`;
});
});
// Country tags
const tagsContainer = document.getElementById("countries-row");
(countries || []).forEach((country) => {
const tag = document.createElement("span");
tag.className = "country-tag";
tag.textContent = country;
tagsContainer.appendChild(tag);
});
});

View File

@ -1,41 +1,46 @@
// scoring.js — accuracy calculation
const Scoring = (() => {
/**
* Calculate a score from the drawn path points.
* NOTE: Currently uses an effort-based approximation.
* TODO: Replace with real polygon comparison (IoU or Hausdorff distance).
* @param {{ x: number, y: number }[]} drawnPoints
* @returns {number} Score between 0 and 100.
*/
function calculateScore(drawnPoints) {
if (!drawnPoints || drawnPoints.length < 10) return 0;
/**
* Calculate score from drawn path points.
* MVP: fake score based on number of points drawn (effort-based).
* TODO: replace with real compareShapes() using polygon overlap.
*/
function calculateScore(drawnPoints) {
if (!drawnPoints || drawnPoints.length < 10) return 0;
const effort = Math.min(drawnPoints.length / 300, 1); // 01
const base = 40 + Math.round(effort * 45); // 4085
const jitter = Math.round((Math.random() - 0.5) * 14); // ±7
return Math.max(0, Math.min(100, base + jitter));
}
// Fake scoring: reward effort + randomness so it feels real
const effort = Math.min(drawnPoints.length / 300, 1); // 01
const base = 40 + Math.round(effort * 45); // 4085
const jitter = Math.round((Math.random() - 0.5) * 14); // ±7
const score = Math.max(0, Math.min(100, base + jitter));
return score;
}
/**
* Compare a drawn polygon against a reference polygon.
* Stub reserved for future IoU / Hausdorff implementation.
* @param {{ x: number, y: number }[]} _drawnPoints
* @param {{ x: number, y: number }[]} _referencePolygon - Normalised 01 coords.
* @returns {number} Score between 0 and 100.
*/
function compareShapes(_drawnPoints, _referencePolygon) {
// TODO: implement real shape comparison
return 0;
}
/**
* Stub for real shape comparison (future).
* drawnPoints: [{x,y}, ...]
* referencePolygon: [{x,y}, ...] (normalised 01 coords)
*/
function compareShapes(_drawnPoints, _referencePolygon) {
// TODO: implement IoU (Intersection over Union) or
// Hausdorff distance for polygon comparison.
return 0;
}
/**
* Map a numeric score to a letter grade with colour.
* @param {number} score
* @returns {{ label: string, color: string }}
*/
function getGrade(score) {
if (score >= 90) return { label: "S", color: "#f0b429" };
if (score >= 75) return { label: "A", color: "#41b869" };
if (score >= 60) return { label: "B", color: "#1a7fc4" };
if (score >= 40) return { label: "C", color: "#7a9aaa" };
return { label: "D", color: "#e05c5c" };
}
function getGrade(score) {
if (score >= 90) return { label: "S", color: "#f0b429" };
if (score >= 75) return { label: "A", color: "#41b869" };
if (score >= 60) return { label: "B", color: "#1a7fc4" };
if (score >= 40) return { label: "C", color: "#7a9aaa" };
return { label: "D", color: "#e05c5c" };
}
return { calculateScore, compareShapes, getGrade };
return { calculateScore, compareShapes, getGrade };
})();

View File

@ -1,75 +1,105 @@
// storage.js — localStorage helpers
const Storage = (() => {
const KEYS = {
PLAYER_NAME: "gd_playerName",
LOBBY_NAME: "gd_lobbyName",
GAME_STATE: "gd_gameState",
LEADERBOARD: "gd_leaderboard",
};
const KEYS = {
PLAYER_NAME: "gd_playerName",
LOBBY_NAME: "gd_lobbyName",
GAME_STATE: "gd_gameState",
LEADERBOARD: "gd_leaderboard",
};
function savePlayerName(name) {
localStorage.setItem(KEYS.PLAYER_NAME, name.trim());
}
/**
* Safely write a value to localStorage.
* Silently fails if storage quota is exceeded or unavailable.
* @param {string} key
* @param {string} value
*/
function _set(key, value) {
try {
localStorage.setItem(key, value);
} catch {
console.warn(`Storage: could not write key "${key}"`);
}
}
function getPlayerName() {
return localStorage.getItem(KEYS.PLAYER_NAME) || "Anonymous";
}
/** @param {string} name */
function savePlayerName(name) {
_set(KEYS.PLAYER_NAME, name.trim());
}
function saveLobbyName(name) {
localStorage.setItem(KEYS.LOBBY_NAME, name.trim());
}
/** @returns {string} */
function getPlayerName() {
return localStorage.getItem(KEYS.PLAYER_NAME) || "Anonymous";
}
function getLobbyName() {
return localStorage.getItem(KEYS.LOBBY_NAME) || "My Lobby";
}
/** @param {string} name */
function saveLobbyName(name) {
_set(KEYS.LOBBY_NAME, name.trim());
}
function saveGameState(state) {
localStorage.setItem(KEYS.GAME_STATE, JSON.stringify(state));
}
/** @returns {string} */
function getLobbyName() {
return localStorage.getItem(KEYS.LOBBY_NAME) || "My Lobby";
}
function getGameState() {
try {
return JSON.parse(localStorage.getItem(KEYS.GAME_STATE)) || null;
} catch {
return null;
}
}
/**
* @param {{ scores: number[], totalScore: number, countries: string[] }} state
*/
function saveGameState(state) {
_set(KEYS.GAME_STATE, JSON.stringify(state));
}
function clearGameState() {
localStorage.removeItem(KEYS.GAME_STATE);
}
/**
* @returns {{ scores: number[], totalScore: number, countries: string[] } | null}
*/
function getGameState() {
try {
return JSON.parse(localStorage.getItem(KEYS.GAME_STATE)) || null;
} catch {
return null;
}
}
function saveLeaderboard(entry) {
const board = getLeaderboard();
board.push(entry);
board.sort((a, b) => b.totalScore - a.totalScore);
const top20 = board.slice(0, 20);
localStorage.setItem(KEYS.LEADERBOARD, JSON.stringify(top20));
}
function clearGameState() {
localStorage.removeItem(KEYS.GAME_STATE);
}
function getLeaderboard() {
try {
return JSON.parse(localStorage.getItem(KEYS.LEADERBOARD)) || [];
} catch {
return [];
}
}
/**
* Add an entry to the leaderboard and keep the top 20.
* @param {{ name: string, totalScore: number, scores: number[], date: string }} entry
*/
function saveLeaderboard(entry) {
const board = getLeaderboard();
board.push(entry);
board.sort((a, b) => b.totalScore - a.totalScore);
_set(KEYS.LEADERBOARD, JSON.stringify(board.slice(0, 20)));
}
function clearLeaderboard() {
localStorage.removeItem(KEYS.LEADERBOARD);
}
/**
* @returns {{ name: string, totalScore: number, scores: number[], date: string }[]}
*/
function getLeaderboard() {
try {
return JSON.parse(localStorage.getItem(KEYS.LEADERBOARD)) || [];
} catch {
return [];
}
}
return {
savePlayerName,
getPlayerName,
saveLobbyName,
getLobbyName,
saveGameState,
getGameState,
clearGameState,
saveLeaderboard,
getLeaderboard,
clearLeaderboard,
};
function clearLeaderboard() {
localStorage.removeItem(KEYS.LEADERBOARD);
}
return {
savePlayerName,
getPlayerName,
saveLobbyName,
getLobbyName,
saveGameState,
getGameState,
clearGameState,
saveLeaderboard,
getLeaderboard,
clearLeaderboard,
};
})();

View File

@ -1,282 +1,339 @@
/* game.css */
.game-layout {
min-height: calc(100vh - var(--hh));
display: grid;
grid-template-rows: auto 1fr auto;
padding: 20px 0 28px;
gap: 16px;
min-height: calc(100vh - var(--hh));
display: grid;
grid-template-rows: auto 1fr auto;
padding: 20px 0 28px;
gap: 16px;
}
/* ── Top bar */
.game-topbar {
display: grid;
grid-template-columns: 1fr auto 1fr;
align-items: center;
gap: 16px;
display: grid;
grid-template-columns: 1fr auto 1fr;
align-items: center;
gap: 16px;
}
.game-round {
display: flex;
align-items: center;
gap: 10px;
display: flex;
align-items: center;
gap: 10px;
}
.round-pip {
width: 10px;
height: 10px;
border-radius: 50%;
background: var(--line);
transition: background .3s;
width: 10px;
height: 10px;
border-radius: 50%;
background: var(--line);
transition: background 0.3s;
}
.round-pip.active { background: var(--sea); }
.round-pip.done { background: var(--leaf); }
.round-pip.active {
background: var(--sea);
}
.round-pip.done {
background: var(--leaf);
}
.round-label {
font-family: 'Syne', sans-serif;
font-size: .82rem;
font-weight: 700;
color: var(--ink-muted);
letter-spacing: .06em;
text-transform: uppercase;
font-family: "Syne", sans-serif;
font-size: 0.82rem;
font-weight: 700;
color: var(--ink-muted);
letter-spacing: 0.06em;
text-transform: uppercase;
}
.round-label span {
color: var(--ink);
color: var(--ink);
}
/* Country name */
.game-country {
text-align: center;
text-align: center;
}
.country-name {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: clamp(1.5rem, 3vw, 2.4rem);
letter-spacing: -.03em;
line-height: 1.1;
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: clamp(1.5rem, 3vw, 2.4rem);
letter-spacing: -0.03em;
line-height: 1.1;
}
.country-hint {
font-size: .82rem;
color: var(--ink-muted);
margin-top: 4px;
font-size: 0.82rem;
color: var(--ink-muted);
margin-top: 4px;
}
/* Timer */
.game-timer {
display: flex;
flex-direction: column;
align-items: flex-end;
gap: 6px;
display: flex;
flex-direction: column;
align-items: flex-end;
gap: 6px;
}
.timer-display {
display: flex;
align-items: baseline;
gap: 4px;
display: flex;
align-items: baseline;
gap: 4px;
}
.timer-num {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: 2rem;
line-height: 1;
color: var(--ink);
transition: color .3s;
min-width: 2ch;
text-align: right;
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 2rem;
line-height: 1;
color: var(--ink);
transition: color 0.3s;
min-width: 2ch;
text-align: right;
}
.timer-unit {
font-size: .78rem;
font-weight: 600;
color: var(--ink-muted);
font-size: 0.78rem;
font-weight: 600;
color: var(--ink-muted);
}
.timer-track {
width: 120px;
height: 4px;
background: var(--line);
border-radius: 999px;
overflow: hidden;
width: 120px;
height: 4px;
background: var(--line);
border-radius: 999px;
overflow: hidden;
}
.timer-bar {
height: 100%;
width: 100%;
background: linear-gradient(90deg, var(--sea), var(--leaf));
border-radius: 999px;
transition: width 1s linear, background .4s ease;
height: 100%;
width: 100%;
background: linear-gradient(90deg, var(--sea), var(--leaf));
border-radius: 999px;
transition:
width 1s linear,
background 0.4s ease;
}
/* Timer urgency states — applied via JS classList */
.timer--warning .timer-bar,
.timer--warning .timer-num {
color: var(--gold);
}
.timer--warning .timer-bar {
background: var(--gold);
}
.timer--danger .timer-bar,
.timer--danger .timer-num {
color: var(--danger);
}
.timer--danger .timer-bar {
background: var(--danger);
}
/* ── Canvas area */
.canvas-area {
display: grid;
grid-template-columns: 1fr 220px;
gap: 16px;
min-height: 0;
display: grid;
grid-template-columns: 1fr 220px;
gap: 16px;
min-height: 0;
}
.canvas-wrap {
position: relative;
background: var(--white);
border: 1.5px solid var(--line);
border-radius: var(--r-xl);
box-shadow: var(--shadow);
overflow: hidden;
cursor: crosshair;
position: relative;
background: var(--white);
border: 1.5px solid var(--line);
border-radius: var(--r-xl);
box-shadow: var(--shadow);
overflow: hidden;
cursor: crosshair;
}
.canvas-wrap::before {
content: 'Draw the border here';
position: absolute;
inset: 0;
display: flex;
align-items: center;
justify-content: center;
font-size: .9rem;
font-weight: 500;
color: var(--ink-muted);
pointer-events: none;
opacity: 1;
transition: opacity .3s;
content: "Draw the border here";
position: absolute;
inset: 0;
display: flex;
align-items: center;
justify-content: center;
font-size: 0.9rem;
font-weight: 500;
color: var(--ink-muted);
pointer-events: none;
opacity: 1;
transition: opacity 0.3s;
}
.canvas-wrap.has-drawing::before { opacity: 0; }
.canvas-wrap.has-drawing::before {
opacity: 0;
}
#draw-canvas {
display: block;
width: 100%;
height: 100%;
touch-action: none;
display: block;
width: 100%;
height: 100%;
touch-action: none;
}
/* Score feedback overlay */
#score-feedback {
position: absolute;
top: 20px;
left: 50%;
transform: translateX(-50%) translateY(-10px);
font-family: 'Syne', sans-serif;
font-size: 2rem;
font-weight: 800;
opacity: 0;
transition: opacity .4s ease, transform .4s ease;
pointer-events: none;
text-shadow: 0 2px 8px rgba(0,0,0,.12);
white-space: nowrap;
position: absolute;
top: 20px;
left: 50%;
transform: translateX(-50%) translateY(-10px);
font-family: "Syne", sans-serif;
font-size: 2rem;
font-weight: 800;
opacity: 0;
transition:
opacity 0.4s ease,
transform 0.4s ease;
pointer-events: none;
text-shadow: 0 2px 8px rgba(0, 0, 0, 0.12);
white-space: nowrap;
}
/* Sidebar */
.game-sidebar {
display: flex;
flex-direction: column;
gap: 12px;
display: flex;
flex-direction: column;
gap: 12px;
}
.sidebar-card {
background: var(--glass);
backdrop-filter: blur(10px);
border: 1px solid rgba(255,255,255,.85);
border-radius: var(--r-lg);
padding: 18px 20px;
box-shadow: var(--shadow-sm);
background: var(--glass);
backdrop-filter: blur(10px);
border: 1px solid rgba(255, 255, 255, 0.85);
border-radius: var(--r-lg);
padding: 18px 20px;
box-shadow: var(--shadow-sm);
}
.sidebar-card__label {
font-size: .72rem;
font-weight: 700;
letter-spacing: .08em;
text-transform: uppercase;
color: var(--ink-muted);
margin-bottom: 10px;
font-size: 0.72rem;
font-weight: 700;
letter-spacing: 0.08em;
text-transform: uppercase;
color: var(--ink-muted);
margin-bottom: 10px;
}
/* Round scores */
.round-scores { display: grid; gap: 8px; }
.round-scores {
display: grid;
gap: 8px;
}
.rscore-row {
display: flex;
align-items: center;
justify-content: space-between;
gap: 8px;
display: flex;
align-items: center;
justify-content: space-between;
gap: 8px;
}
.rscore-label {
font-size: .82rem;
color: var(--ink-soft);
font-weight: 500;
font-size: 0.82rem;
color: var(--ink-soft);
font-weight: 500;
}
.rscore-val {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: .9rem;
color: var(--ink-muted);
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 0.9rem;
color: var(--ink-muted);
}
.rscore-val.filled { color: var(--sea); }
.rscore-val.filled {
color: var(--sea);
}
/* Player name in sidebar */
.player-name-display {
font-family: 'Syne', sans-serif;
font-weight: 700;
font-size: 1rem;
color: var(--ink);
word-break: break-word;
font-family: "Syne", sans-serif;
font-weight: 700;
font-size: 1rem;
color: var(--ink);
word-break: break-word;
}
/* Controls */
.game-controls {
display: flex;
gap: 10px;
display: flex;
gap: 10px;
}
/* ─── Responsive ─── */
@media (max-width: 900px) {
.canvas-area {
grid-template-columns: 1fr;
grid-template-rows: 1fr auto;
}
.canvas-area {
grid-template-columns: 1fr;
grid-template-rows: 1fr auto;
}
.game-sidebar {
flex-direction: row;
flex-wrap: wrap;
}
.game-sidebar {
flex-direction: row;
flex-wrap: wrap;
}
.sidebar-card { flex: 1 1 140px; }
.sidebar-card {
flex: 1 1 140px;
}
.canvas-wrap { min-height: 340px; }
.canvas-wrap {
min-height: 340px;
}
.game-topbar {
grid-template-columns: auto 1fr auto;
}
.game-topbar {
grid-template-columns: auto 1fr auto;
}
}
@media (max-width: 600px) {
.game-layout { padding: 14px 0 20px; gap: 12px; }
.game-layout {
padding: 14px 0 20px;
gap: 12px;
}
.game-topbar {
grid-template-columns: 1fr;
grid-template-rows: auto auto auto;
gap: 10px;
}
.game-topbar {
grid-template-columns: 1fr;
grid-template-rows: auto auto auto;
gap: 10px;
}
.game-country { text-align: left; }
.game-timer { align-items: flex-start; }
.timer-track { width: 100%; }
.game-country {
text-align: left;
}
.game-timer {
align-items: flex-start;
}
.timer-track {
width: 100%;
}
.canvas-wrap { min-height: 280px; }
.canvas-wrap {
min-height: 280px;
}
.game-controls { flex-direction: column; }
.game-controls .btn { width: 100%; }
.game-controls {
flex-direction: column;
}
.game-controls .btn {
width: 100%;
}
}
@media (max-width: 360px) {
.country-name { font-size: 1.4rem; }
.timer-num { font-size: 1.6rem; }
.canvas-wrap { min-height: 240px; }
.country-name {
font-size: 1.4rem;
}
.timer-num {
font-size: 1.6rem;
}
.canvas-wrap {
min-height: 240px;
}
}

View File

@ -1,206 +1,6 @@
:root {
--ink: #0b1f2a;
--ink-soft: #3d5563;
--ink-muted: #7a9aaa;
--sea: #1a7fc4;
--sea-light: #4faae0;
--sea-dim: rgba(26, 127, 196, 0.12);
--leaf: #41b869;
--leaf-dim: rgba(65, 184, 105, 0.13);
--cream: #f4f9f6;
--white: #ffffff;
--glass: rgba(255, 255, 255, 0.72);
--line: rgba(11, 31, 42, 0.09);
--shadow: 0 24px 60px rgba(11, 31, 42, 0.1);
--r-xl: 32px;
--r-lg: 22px;
--r-md: 14px;
--hh: 80px;
}
*,
*::before,
*::after {
box-sizing: border-box;
margin: 0;
padding: 0;
}
html {
scroll-behavior: smooth;
}
body {
font-family: "DM Sans", sans-serif;
color: var(--ink);
background: var(--cream);
overflow-x: hidden;
position: relative;
}
body::before {
content: "";
position: fixed;
inset: 0;
background:
radial-gradient(
ellipse 60% 50% at 10% 0%,
rgba(65, 184, 105, 0.12) 0%,
transparent 60%
),
radial-gradient(
ellipse 50% 60% at 90% 100%,
rgba(26, 127, 196, 0.12) 0%,
transparent 60%
);
pointer-events: none;
z-index: 0;
}
body::after {
content: "";
position: fixed;
inset: 0;
background-image:
linear-gradient(var(--line) 1px, transparent 1px),
linear-gradient(90deg, var(--line) 1px, transparent 1px);
background-size: 60px 60px;
pointer-events: none;
z-index: 0;
}
a {
color: inherit;
text-decoration: none;
}
img,
svg {
display: block;
max-width: 100%;
}
button,
input {
font: inherit;
}
.wrap {
position: relative;
z-index: 1;
}
.container {
width: 100%;
max-width: 1280px;
margin: 0 auto;
padding: 0 28px;
}
/* ─── HEADER ─── */
.header {
position: sticky;
top: 0;
z-index: 100;
height: var(--hh);
display: flex;
align-items: center;
backdrop-filter: blur(16px) saturate(1.4);
-webkit-backdrop-filter: blur(16px) saturate(1.4);
background: rgba(244, 249, 246, 0.82);
border-bottom: 1px solid var(--line);
}
.header__inner {
display: flex;
align-items: center;
justify-content: space-between;
gap: 24px;
}
.logo {
display: inline-flex;
align-items: center;
gap: 12px;
}
.logo__mark {
width: 44px;
height: 44px;
border-radius: 50%;
background: linear-gradient(135deg, var(--sea) 0%, var(--leaf) 100%);
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
position: relative;
overflow: hidden;
}
.logo__mark svg {
width: 22px;
height: 22px;
}
.logo__text {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 1.25rem;
letter-spacing: -0.02em;
}
.nav {
display: flex;
align-items: center;
gap: 4px;
}
.nav__link {
padding: 9px 16px;
border-radius: 999px;
font-size: 0.92rem;
font-weight: 500;
color: var(--ink-soft);
transition:
background 0.18s,
color 0.18s;
}
.nav__link:hover {
background: var(--sea-dim);
color: var(--sea);
}
.nav__cta {
margin-left: 8px;
padding: 9px 20px;
border-radius: 999px;
font-size: 0.92rem;
font-weight: 600;
color: var(--white);
background: var(--ink);
transition:
opacity 0.18s,
transform 0.18s;
}
.nav__cta:hover {
opacity: 0.82;
transform: translateY(-1px);
}
/* ─── HERO ─── */
.hero {
min-height: calc(100vh - var(--hh));
display: grid;
place-items: center;
padding: 80px 0 60px;
}
.hero__inner {
display: grid;
grid-template-columns: 1fr minmax(320px, 500px);
gap: 64px;
align-items: center;
}
/* index.css — landing page specific styles */
/* ─── EYEBROW ─── */
.eyebrow {
display: inline-flex;
align-items: center;
@ -237,6 +37,21 @@ input {
}
}
/* ─── HERO ─── */
.hero {
min-height: calc(100vh - var(--hh));
display: grid;
place-items: center;
padding: 80px 0 60px;
}
.hero__inner {
display: grid;
grid-template-columns: 1fr minmax(320px, 500px);
gap: 64px;
align-items: center;
}
.hero__title {
font-family: "Syne", sans-serif;
font-weight: 800;
@ -269,73 +84,7 @@ input {
flex-wrap: wrap;
}
.btn {
display: inline-flex;
align-items: center;
justify-content: center;
gap: 8px;
padding: 16px 30px;
border-radius: var(--r-lg);
font-family: "DM Sans", sans-serif;
font-size: 1rem;
font-weight: 600;
border: none;
cursor: pointer;
transition:
transform 0.2s,
box-shadow 0.2s,
opacity 0.2s;
}
.btn:hover {
transform: translateY(-2px);
}
.btn--primary {
color: var(--white);
background: linear-gradient(
135deg,
var(--sea) 0%,
#159fd4 50%,
var(--leaf) 100%
);
background-size: 200% 200%;
box-shadow: 0 8px 24px rgba(26, 127, 196, 0.3);
animation: gradShift 4s ease infinite;
}
@keyframes gradShift {
0% {
background-position: 0% 50%;
}
50% {
background-position: 100% 50%;
}
100% {
background-position: 0% 50%;
}
}
.btn--primary:hover {
box-shadow: 0 12px 32px rgba(26, 127, 196, 0.4);
}
.btn--ghost {
color: var(--ink-soft);
background: transparent;
border: 1.5px solid var(--line);
}
.btn--ghost:hover {
border-color: var(--ink-muted);
color: var(--ink);
}
.btn--full {
width: 100%;
}
/* Globe card */
/* ─── GLOBE CARD ─── */
.globe-card {
background: var(--glass);
backdrop-filter: blur(12px);
@ -494,7 +243,6 @@ input {
height: 8px;
border-radius: 50%;
}
.gb__dot--green {
background: var(--leaf);
}
@ -514,28 +262,6 @@ input {
align-items: center;
}
.section-label {
display: inline-block;
padding: 6px 14px;
border-radius: 999px;
background: var(--sea-dim);
color: var(--sea);
font-size: 0.82rem;
font-weight: 600;
letter-spacing: 0.04em;
text-transform: uppercase;
margin-bottom: 20px;
}
.section-title {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: clamp(2.4rem, 4.5vw, 4rem);
line-height: 1.04;
letter-spacing: -0.03em;
margin-bottom: 20px;
}
.about__desc {
font-size: 1.05rem;
line-height: 1.75;
@ -543,7 +269,35 @@ input {
margin-bottom: 36px;
}
/* Steps */
.mini-globe-wrap {
background: var(--glass);
backdrop-filter: blur(10px);
border: 1px solid rgba(255, 255, 255, 0.8);
border-radius: var(--r-xl);
box-shadow: var(--shadow);
padding: 32px;
display: flex;
align-items: center;
justify-content: center;
aspect-ratio: 1;
max-width: 340px;
position: relative;
overflow: hidden;
}
.mini-globe-wrap::before {
content: "";
position: absolute;
inset: 0;
border-radius: var(--r-xl);
background: radial-gradient(
circle at 60% 40%,
rgba(26, 127, 196, 0.1),
transparent 50%
);
}
/* ─── STEPS ─── */
.steps {
display: grid;
gap: 16px;
@ -633,48 +387,6 @@ input {
margin-bottom: 30px;
}
.form {
display: grid;
gap: 20px;
}
.field {
display: grid;
gap: 8px;
}
.field label {
font-size: 0.86rem;
font-weight: 600;
color: var(--ink-soft);
letter-spacing: 0.02em;
}
.field input {
width: 100%;
height: 54px;
padding: 0 18px;
border: 1.5px solid var(--line);
border-radius: var(--r-md);
background: var(--cream);
color: var(--ink);
font-size: 0.98rem;
transition:
border-color 0.18s,
box-shadow 0.18s;
}
.field input::placeholder {
color: var(--ink-muted);
}
.field input:focus {
outline: none;
border-color: var(--sea-light);
box-shadow: 0 0 0 3px rgba(26, 127, 196, 0.12);
background: var(--white);
}
.form-footer {
margin-top: 8px;
font-size: 0.84rem;
@ -682,94 +394,6 @@ input {
text-align: center;
}
.mini-globe-wrap {
background: var(--glass);
backdrop-filter: blur(10px);
border: 1px solid rgba(255, 255, 255, 0.8);
border-radius: var(--r-xl);
box-shadow: var(--shadow);
padding: 32px;
display: flex;
align-items: center;
justify-content: center;
aspect-ratio: 1;
max-width: 340px;
position: relative;
overflow: hidden;
}
.mini-globe-wrap::before {
content: "";
position: absolute;
inset: 0;
border-radius: var(--r-xl);
background: radial-gradient(
circle at 60% 40%,
rgba(26, 127, 196, 0.1),
transparent 50%
);
}
/* ─── FOOTER ─── */
.footer {
border-top: 1px solid var(--line);
background: rgba(255, 255, 255, 0.5);
backdrop-filter: blur(12px);
padding: 40px 0;
}
.footer__inner {
display: grid;
grid-template-columns: 1fr auto 1fr;
align-items: center;
gap: 24px;
}
.footer__brand {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 1.5rem;
letter-spacing: -0.02em;
}
.footer__sub {
font-size: 0.86rem;
color: var(--ink-muted);
margin-top: 4px;
}
.footer__center {
text-align: center;
}
.footer__label {
font-size: 0.7rem;
font-weight: 700;
letter-spacing: 0.1em;
text-transform: uppercase;
color: var(--ink-muted);
margin-bottom: 10px;
}
.footer__right {
justify-self: end;
display: flex;
gap: 22px;
flex-wrap: wrap;
justify-content: flex-end;
}
.footer__link {
font-size: 0.9rem;
color: var(--ink-muted);
transition: color 0.18s;
}
.footer__link:hover {
color: var(--ink);
}
/* ─── RESPONSIVE ─── */
@media (max-width: 960px) {
.hero__inner,
@ -782,30 +406,12 @@ input {
.globe-card {
min-height: 360px;
}
.footer__inner {
grid-template-columns: 1fr;
}
.footer__center,
.footer__right {
justify-self: start;
text-align: left;
}
.socials {
justify-content: flex-start;
}
.footer__right {
justify-content: flex-start;
}
.mini-globe-wrap {
max-width: 260px;
}
}
@media (max-width: 680px) {
.container {
padding: 0 18px;
}
.hero {
padding: 50px 0 40px;
}
@ -813,9 +419,6 @@ input {
.register {
padding: 70px 0;
}
.nav__link {
display: none;
}
.register__card {
padding: 28px 22px;
}
@ -823,52 +426,15 @@ input {
padding: 24px;
min-height: 280px;
}
.header {
height: auto;
padding: 14px 0;
}
:root {
--hh: 64px;
}
}
@media (max-width: 420px) {
.hero__title {
font-size: 3rem;
}
.section-title {
font-size: 2.2rem;
}
.btn {
padding: 14px 22px;
font-size: 0.95rem;
}
}
@media (max-width: 360px) {
:root {
--r-xl: 22px;
--r-lg: 16px;
}
.container {
padding: 0 14px;
}
.logo__mark {
width: 36px;
height: 36px;
}
.logo__text {
font-size: 1.05rem;
}
.nav__cta {
padding: 8px 14px;
font-size: 0.84rem;
margin-left: 0;
}
.hero {
padding: 36px 0 32px;
min-height: auto;
@ -881,7 +447,6 @@ input {
font-size: 0.95rem;
margin-bottom: 28px;
}
.globe-card {
padding: 16px;
min-height: 240px;
@ -890,18 +455,13 @@ input {
padding: 7px 10px;
font-size: 0.72rem;
}
.about,
.register {
padding: 52px 0;
}
.section-title {
font-size: 2rem;
}
.about__desc {
font-size: 0.95rem;
}
.step {
padding: 16px 18px;
gap: 14px;
@ -912,7 +472,6 @@ input {
font-size: 0.88rem;
border-radius: 12px;
}
.register__card {
padding: 22px 16px;
border-radius: var(--r-lg);
@ -921,54 +480,8 @@ input {
font-size: 1.25rem;
margin-bottom: 22px;
}
.field input {
height: 48px;
font-size: 0.92rem;
}
.btn--full {
padding: 14px 18px;
font-size: 0.95rem;
}
.footer {
padding: 30px 0;
}
.footer__brand {
font-size: 1.25rem;
}
.social {
width: 36px;
height: 36px;
font-size: 0.8rem;
}
.socials {
gap: 8px;
}
.footer__right {
gap: 14px;
}
}
/* ─── SCROLL REVEAL ─── */
.reveal {
opacity: 0;
transform: translateY(28px);
transition:
opacity 0.6s ease,
transform 0.6s ease;
}
.reveal.visible {
opacity: 1;
transform: translateY(0);
}
.reveal-delay-1 {
transition-delay: 0.1s;
}
.reveal-delay-2 {
transition-delay: 0.2s;
}
.reveal-delay-3 {
transition-delay: 0.32s;
}

View File

@ -1,196 +1,247 @@
/* leader.css — leaderboard page styles */
.lb-page {
min-height: calc(100vh - var(--hh));
padding: 60px 0;
}
min-height: calc(100vh - var(--hh));
padding: 60px 0;
}
.lb-inner {
max-width: 760px;
margin: 0 auto;
}
.lb-inner {
max-width: 760px;
margin: 0 auto;
}
.lb-head {
display: flex;
align-items: flex-end;
justify-content: space-between;
gap: 20px;
margin-bottom: 32px;
flex-wrap: wrap;
}
.lb-head {
display: flex;
align-items: flex-end;
justify-content: space-between;
gap: 20px;
margin-bottom: 32px;
flex-wrap: wrap;
}
.lb-title {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: clamp(2rem, 5vw, 3.2rem);
letter-spacing: -.03em;
line-height: 1;
}
.lb-title {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: clamp(2rem, 5vw, 3.2rem);
letter-spacing: -0.03em;
line-height: 1;
}
.lb-title em {
font-style: normal;
background: linear-gradient(135deg, var(--sea), var(--leaf));
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.lb-title em {
font-style: normal;
background: linear-gradient(135deg, var(--sea), var(--leaf));
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.lb-actions { display: flex; gap: 10px; flex-wrap: wrap; }
.lb-actions {
display: flex;
gap: 10px;
flex-wrap: wrap;
}
/* Table */
.lb-table {
background: var(--white);
border: 1px solid var(--line);
border-radius: var(--r-xl);
overflow: hidden;
box-shadow: var(--shadow);
}
/* ─── TABLE ─── */
.lb-table {
background: var(--white);
border: 1px solid var(--line);
border-radius: var(--r-xl);
overflow: hidden;
box-shadow: var(--shadow);
}
.lb-table-head {
display: grid;
grid-template-columns: 56px 1fr 90px 90px 90px;
gap: 0;
padding: 14px 24px;
background: var(--cream);
border-bottom: 1px solid var(--line);
}
.lb-table-head {
display: grid;
grid-template-columns: 56px 1fr 90px 90px 90px;
padding: 14px 24px;
background: var(--cream);
border-bottom: 1px solid var(--line);
}
.lb-th {
font-size: .72rem;
font-weight: 700;
letter-spacing: .08em;
text-transform: uppercase;
color: var(--ink-muted);
}
.lb-th {
font-size: 0.72rem;
font-weight: 700;
letter-spacing: 0.08em;
text-transform: uppercase;
color: var(--ink-muted);
}
.lb-th.right { text-align: right; }
.lb-th.right {
text-align: right;
}
.lb-body { display: grid; }
.lb-body {
display: grid;
}
.lb-row {
display: grid;
grid-template-columns: 56px 1fr 90px 90px 90px;
align-items: center;
padding: 16px 24px;
border-bottom: 1px solid var(--line);
transition: background .15s;
animation: fadeRow .4s ease both;
}
.lb-row {
display: grid;
grid-template-columns: 56px 1fr 90px 90px 90px;
align-items: center;
padding: 16px 24px;
border-bottom: 1px solid var(--line);
transition: background 0.15s;
animation: fadeRow 0.4s ease both;
}
.lb-row:last-child { border-bottom: none; }
.lb-row:last-child {
border-bottom: none;
}
.lb-row:hover {
background: var(--cream);
}
.lb-row:hover { background: var(--cream); }
@keyframes fadeRow {
from {
opacity: 0;
transform: translateY(8px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
@keyframes fadeRow {
from { opacity: 0; transform: translateY(8px); }
to { opacity: 1; transform: translateY(0); }
}
/* ─── TOP 3 HIGHLIGHT ─── */
.lb-row.rank-1 {
background: rgba(240, 180, 40, 0.07);
}
.lb-row.rank-2 {
background: rgba(180, 180, 195, 0.06);
}
.lb-row.rank-3 {
background: rgba(205, 130, 70, 0.05);
}
/* Top 3 highlight */
.lb-row.rank-1 { background: rgba(240,180,40,.07); }
.lb-row.rank-2 { background: rgba(180,180,195,.06); }
.lb-row.rank-3 { background: rgba(205,130,70,.05); }
.lb-rank {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 1rem;
color: var(--ink-muted);
}
.lb-rank {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: 1rem;
color: var(--ink-muted);
}
.lb-row.rank-1 .lb-rank {
color: var(--gold);
}
.lb-row.rank-2 .lb-rank {
color: #a0a0b0;
}
.lb-row.rank-3 .lb-rank {
color: #c07840;
}
.lb-row.rank-1 .lb-rank { color: var(--gold); }
.lb-row.rank-2 .lb-rank { color: #a0a0b0; }
.lb-row.rank-3 .lb-rank { color: #c07840; }
.lb-medal {
font-size: 1.1rem;
}
.lb-medal { font-size: 1.1rem; }
.lb-name {
font-family: "Syne", sans-serif;
font-weight: 700;
font-size: 0.96rem;
color: var(--ink);
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.lb-name {
font-family: 'Syne', sans-serif;
font-weight: 700;
font-size: .96rem;
color: var(--ink);
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.lb-name.is-you::after {
content: " (you)";
font-weight: 400;
font-family: "DM Sans", sans-serif;
color: var(--ink-muted);
font-size: 0.8rem;
}
.lb-name.is-you::after {
content: ' (you)';
font-weight: 400;
font-family: 'DM Sans', sans-serif;
color: var(--ink-muted);
font-size: .8rem;
}
.lb-rounds {
font-size: 0.82rem;
color: var(--ink-muted);
text-align: right;
}
.lb-rounds {
font-size: .82rem;
color: var(--ink-muted);
text-align: right;
}
.lb-date {
font-size: 0.78rem;
color: var(--ink-muted);
text-align: right;
}
.lb-date {
font-size: .78rem;
color: var(--ink-muted);
text-align: right;
}
.lb-score {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 1.05rem;
color: var(--sea);
text-align: right;
}
.lb-score {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: 1.05rem;
color: var(--sea);
text-align: right;
}
.lb-score.gold {
color: var(--gold);
}
.lb-score.silver {
color: #888898;
}
.lb-score.bronze {
color: #c07840;
}
.lb-score.gold { color: var(--gold); }
.lb-score.silver { color: #888898; }
.lb-score.bronze { color: #c07840; }
/* ─── EMPTY STATE ─── */
.lb-empty {
padding: 64px 24px;
text-align: center;
}
/* Empty state */
.lb-empty {
padding: 64px 24px;
text-align: center;
}
.lb-empty__icon {
font-size: 3rem;
margin-bottom: 12px;
}
.lb-empty__icon { font-size: 3rem; margin-bottom: 12px; }
.lb-empty__text {
font-size: 1rem;
color: var(--ink-soft);
margin-bottom: 20px;
}
.lb-empty__text {
font-size: 1rem;
color: var(--ink-soft);
margin-bottom: 20px;
}
/* ─── CURRENT PLAYER BAR ─── */
.your-score-bar {
margin-top: 20px;
padding: 16px 24px;
background: linear-gradient(135deg, var(--sea-dim), var(--leaf-dim));
border: 1px solid rgba(26, 127, 196, 0.18);
border-radius: var(--r-lg);
display: flex;
align-items: center;
justify-content: space-between;
gap: 16px;
flex-wrap: wrap;
}
/* Current player highlight bar */
.your-score-bar {
margin-top: 20px;
padding: 16px 24px;
background: linear-gradient(135deg, var(--sea-dim), var(--leaf-dim));
border: 1px solid rgba(26,127,196,.18);
border-radius: var(--r-lg);
display: flex;
align-items: center;
justify-content: space-between;
gap: 16px;
flex-wrap: wrap;
}
.your-score-bar__text {
font-size: 0.9rem;
color: var(--ink-soft);
}
.your-score-bar__text strong {
color: var(--ink);
}
.your-score-bar__text {
font-size: .9rem;
color: var(--ink-soft);
}
/* ─── RESPONSIVE ─── */
@media (max-width: 640px) {
.lb-page {
padding: 40px 0;
}
.your-score-bar__text strong { color: var(--ink); }
.lb-table-head,
.lb-row {
grid-template-columns: 44px 1fr 72px;
}
@media (max-width: 640px) {
.lb-page { padding: 40px 0; }
.lb-table-head,
.lb-row {
grid-template-columns: 44px 1fr 72px;
}
.lb-th.hide-sm,
.lb-rounds,
.lb-date { display: none; }
}
.lb-th.hide-sm,
.lb-rounds,
.lb-date {
display: none;
}
}
@media (max-width: 360px) {
.lb-table-head,
.lb-row { padding: 12px 16px; }
}
@media (max-width: 360px) {
.lb-table-head,
.lb-row {
padding: 12px 16px;
}
}

View File

@ -1,202 +1,180 @@
/* lobby.css */
.lobby {
min-height: calc(100vh - var(--hh));
display: grid;
place-items: center;
padding: 60px 0;
min-height: calc(100vh - var(--hh));
display: grid;
place-items: center;
padding: 60px 0;
}
.lobby__inner {
display: grid;
grid-template-columns: 1fr minmax(300px, 480px);
gap: 64px;
align-items: center;
width: 100%;
display: grid;
grid-template-columns: 1fr minmax(300px, 480px);
gap: 64px;
align-items: center;
width: 100%;
}
/* ─── Lobby name badge ─── */
.lobby__name-badge {
display: inline-flex;
align-items: center;
gap: 8px;
padding: 10px 18px;
border-radius: var(--r-lg);
background: var(--glass);
border: 1px solid rgba(255,255,255,.9);
box-shadow: var(--shadow-sm);
backdrop-filter: blur(10px);
margin-bottom: 20px;
display: inline-flex;
align-items: center;
gap: 8px;
padding: 10px 18px;
border-radius: var(--r-lg);
background: var(--glass);
border: 1px solid rgba(255, 255, 255, 0.9);
box-shadow: var(--shadow-sm);
backdrop-filter: blur(10px);
margin-bottom: 20px;
}
.lobby__name-icon { font-size: 1.1rem; }
.lobby__name-icon {
font-size: 1.1rem;
}
#lobby-name-display {
font-family: 'Syne', sans-serif;
font-weight: 700;
font-size: 1rem;
color: var(--ink);
letter-spacing: -.01em;
font-family: "Syne", sans-serif;
font-weight: 700;
font-size: 1rem;
color: var(--ink);
letter-spacing: -0.01em;
}
/* Left side promo */
.lobby__promo {}
/* ─── EYEBROW ─── */
.eyebrow {
display: inline-flex;
align-items: center;
gap: 8px;
padding: 6px 14px 6px 10px;
border-radius: 999px;
border: 1px solid var(--leaf-dim);
background: rgba(65,184,105,.08);
color: #289149;
font-size: .78rem;
font-weight: 600;
letter-spacing: .05em;
text-transform: uppercase;
margin-bottom: 22px;
display: inline-flex;
align-items: center;
gap: 8px;
padding: 6px 14px 6px 10px;
border-radius: 999px;
border: 1px solid var(--leaf-dim);
background: rgba(65, 184, 105, 0.08);
color: #289149;
font-size: 0.78rem;
font-weight: 600;
letter-spacing: 0.05em;
text-transform: uppercase;
margin-bottom: 22px;
}
.eyebrow__dot {
width: 7px;
height: 7px;
border-radius: 50%;
background: var(--leaf);
animation: pulse 2s ease-in-out infinite;
width: 7px;
height: 7px;
border-radius: 50%;
background: var(--leaf);
animation: pulse 2s ease-in-out infinite;
}
@keyframes pulse {
0%, 100% { transform: scale(1); opacity: 1; }
50% { transform: scale(1.45); opacity: .7; }
0%,
100% {
transform: scale(1);
opacity: 1;
}
50% {
transform: scale(1.45);
opacity: 0.7;
}
}
.lobby__title {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: clamp(2.8rem, 5.5vw, 5rem);
line-height: .94;
letter-spacing: -.04em;
margin-bottom: 20px;
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: clamp(2.8rem, 5.5vw, 5rem);
line-height: 0.94;
letter-spacing: -0.04em;
margin-bottom: 20px;
}
.lobby__title em {
font-style: normal;
background: linear-gradient(135deg, var(--sea), var(--leaf));
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
font-style: normal;
background: linear-gradient(135deg, var(--sea), var(--leaf));
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.lobby__desc {
font-size: 1.02rem;
line-height: 1.72;
color: var(--ink-soft);
max-width: 420px;
margin-bottom: 32px;
font-size: 1.02rem;
line-height: 1.72;
color: var(--ink-soft);
max-width: 420px;
margin-bottom: 32px;
}
.lobby__badges {
display: flex;
flex-wrap: wrap;
gap: 10px;
display: flex;
flex-wrap: wrap;
gap: 10px;
}
.badge {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 8px 14px;
border-radius: 999px;
background: var(--glass);
border: 1px solid rgba(255,255,255,.9);
box-shadow: var(--shadow-sm);
font-size: .82rem;
font-weight: 600;
color: var(--ink-soft);
backdrop-filter: blur(8px);
display: inline-flex;
align-items: center;
gap: 6px;
padding: 8px 14px;
border-radius: 999px;
background: var(--glass);
border: 1px solid rgba(255, 255, 255, 0.9);
box-shadow: var(--shadow-sm);
font-size: 0.82rem;
font-weight: 600;
color: var(--ink-soft);
backdrop-filter: blur(8px);
}
.badge__icon { font-size: 1rem; }
.badge__icon {
font-size: 1rem;
}
/* Right side card */
.lobby__card {
background: var(--white);
border: 1px solid var(--line);
border-radius: var(--r-xl);
padding: 44px;
box-shadow: var(--shadow);
background: var(--white);
border: 1px solid var(--line);
border-radius: var(--r-xl);
padding: 44px;
box-shadow: var(--shadow);
}
.lobby__card-title {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: 1.5rem;
letter-spacing: -.02em;
margin-bottom: 8px;
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 1.5rem;
letter-spacing: -0.02em;
margin-bottom: 8px;
}
.lobby__card-sub {
font-size: .9rem;
color: var(--ink-muted);
margin-bottom: 30px;
font-size: 0.9rem;
color: var(--ink-muted);
margin-bottom: 30px;
}
.form { display: grid; gap: 20px; }
.field { display: grid; gap: 8px; }
.field label {
font-size: .84rem;
font-weight: 600;
color: var(--ink-soft);
letter-spacing: .02em;
}
.field input {
width: 100%;
height: 54px;
padding: 0 18px;
border: 1.5px solid var(--line);
border-radius: var(--r-md);
background: var(--cream);
color: var(--ink);
font-size: .98rem;
transition: border-color .18s, box-shadow .18s;
}
.field input::placeholder { color: var(--ink-muted); }
.field input:focus {
outline: none;
border-color: var(--sea-light);
box-shadow: 0 0 0 3px rgba(26,127,196,.12);
background: var(--white);
}
.field input.input--error {
border-color: var(--danger);
box-shadow: 0 0 0 3px rgba(224,92,92,.12);
}
.field-error {
font-size: .82rem;
color: var(--danger);
min-height: 1.2em;
transition: opacity .18s;
}
/* Form and field styles are defined in main.css */
/* ─── Responsive ─── */
@media (max-width: 900px) {
.lobby__inner { grid-template-columns: 1fr; gap: 40px; }
.lobby__inner {
grid-template-columns: 1fr;
gap: 40px;
}
}
@media (max-width: 680px) {
.lobby { padding: 44px 0; }
.lobby__card { padding: 28px 22px; }
.lobby {
padding: 44px 0;
}
.lobby__card {
padding: 28px 22px;
}
}
@media (max-width: 360px) {
.lobby__card { padding: 22px 16px; }
.lobby__title { font-size: 2.4rem; }
.lobby__card {
padding: 22px 16px;
}
.lobby__title {
font-size: 2.4rem;
}
}

View File

@ -1,78 +1,107 @@
/* main.css — shared design tokens & base */
@import url('https://fonts.googleapis.com/css2?family=Syne:wght@400;500;700;800&family=DM+Sans:ital,opsz,wght@0,9..40,300;0,9..40,400;0,9..40,500;0,9..40,600&display=swap');
@import url("https://fonts.googleapis.com/css2?family=Syne:wght@400;500;700;800&family=DM+Sans:ital,opsz,wght@0,9..40,300;0,9..40,400;0,9..40,500;0,9..40,600&display=swap");
:root {
--ink: #0b1f2a;
--ink-soft: #3d5563;
--ink-muted: #7a9aaa;
--sea: #1a7fc4;
--sea-light: #4faae0;
--sea-dim: rgba(26,127,196,.12);
--leaf: #41b869;
--leaf-dim: rgba(65,184,105,.13);
--gold: #f0b429;
--danger: #e05c5c;
--cream: #f4f9f6;
--white: #ffffff;
--glass: rgba(255,255,255,.72);
--line: rgba(11,31,42,.09);
--shadow: 0 24px 60px rgba(11,31,42,.10);
--shadow-sm: 0 4px 20px rgba(11,31,42,.07);
--r-xl: 32px;
--r-lg: 22px;
--r-md: 14px;
--r-sm: 8px;
--hh: 72px;
--ink: #0b1f2a;
--ink-soft: #3d5563;
--ink-muted: #7a9aaa;
--sea: #1a7fc4;
--sea-light: #4faae0;
--sea-dim: rgba(26, 127, 196, 0.12);
--leaf: #41b869;
--leaf-dim: rgba(65, 184, 105, 0.13);
--gold: #f0b429;
--danger: #e05c5c;
--cream: #f4f9f6;
--white: #ffffff;
--glass: rgba(255, 255, 255, 0.72);
--line: rgba(11, 31, 42, 0.09);
--shadow: 0 24px 60px rgba(11, 31, 42, 0.1);
--shadow-sm: 0 4px 20px rgba(11, 31, 42, 0.07);
--r-xl: 32px;
--r-lg: 22px;
--r-md: 14px;
--r-sm: 8px;
--hh: 72px;
}
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
*,
*::before,
*::after {
box-sizing: border-box;
margin: 0;
padding: 0;
}
html { scroll-behavior: smooth; }
html {
scroll-behavior: smooth;
}
body {
font-family: 'DM Sans', sans-serif;
color: var(--ink);
background: var(--cream);
overflow-x: hidden;
position: relative;
min-width: 320px;
font-family: "DM Sans", sans-serif;
color: var(--ink);
background: var(--cream);
overflow-x: hidden;
position: relative;
min-width: 320px;
}
body::before {
content: '';
position: fixed;
inset: 0;
background:
radial-gradient(ellipse 60% 50% at 10% 0%, rgba(65,184,105,.10) 0%, transparent 60%),
radial-gradient(ellipse 50% 60% at 90% 100%, rgba(26,127,196,.10) 0%, transparent 60%);
pointer-events: none;
z-index: 0;
content: "";
position: fixed;
inset: 0;
background:
radial-gradient(
ellipse 60% 50% at 10% 0%,
rgba(65, 184, 105, 0.1) 0%,
transparent 60%
),
radial-gradient(
ellipse 50% 60% at 90% 100%,
rgba(26, 127, 196, 0.1) 0%,
transparent 60%
);
pointer-events: none;
z-index: 0;
}
body::after {
content: '';
position: fixed;
inset: 0;
background-image:
linear-gradient(var(--line) 1px, transparent 1px),
linear-gradient(90deg, var(--line) 1px, transparent 1px);
background-size: 60px 60px;
pointer-events: none;
z-index: 0;
content: "";
position: fixed;
inset: 0;
background-image:
linear-gradient(var(--line) 1px, transparent 1px),
linear-gradient(90deg, var(--line) 1px, transparent 1px);
background-size: 60px 60px;
pointer-events: none;
z-index: 0;
}
a { color: inherit; text-decoration: none; }
img, svg { display: block; max-width: 100%; }
button, input { font: inherit; }
a {
color: inherit;
text-decoration: none;
}
img,
svg {
display: block;
max-width: 100%;
}
button,
input {
font: inherit;
}
.wrap { position: relative; z-index: 1; }
.wrap {
position: relative;
z-index: 1;
}
.container {
width: 100%;
max-width: 1280px;
margin: 0 auto;
padding: 0 28px;
width: 100%;
max-width: 1280px;
margin: 0 auto;
padding: 0 28px;
}
.header {
@ -166,91 +195,174 @@ button, input { font: inherit; }
transform: translateY(-1px);
}
/* ─── BUTTONS ─── */
.btn {
display: inline-flex;
align-items: center;
justify-content: center;
gap: 8px;
padding: 15px 28px;
border-radius: var(--r-lg);
font-family: 'DM Sans', sans-serif;
font-size: 1rem;
font-weight: 600;
border: none;
cursor: pointer;
transition: transform .2s, box-shadow .2s, opacity .2s;
line-height: 1;
display: inline-flex;
align-items: center;
justify-content: center;
gap: 8px;
padding: 15px 28px;
border-radius: var(--r-lg);
font-family: "DM Sans", sans-serif;
font-size: 1rem;
font-weight: 600;
border: none;
cursor: pointer;
transition:
transform 0.2s,
box-shadow 0.2s,
opacity 0.2s;
line-height: 1;
}
.btn:hover:not(:disabled) { transform: translateY(-2px); }
.btn:disabled { opacity: .45; cursor: not-allowed; }
.btn:hover:not(:disabled) {
transform: translateY(-2px);
}
.btn:disabled {
opacity: 0.45;
cursor: not-allowed;
}
.btn--primary {
color: var(--white);
background: linear-gradient(135deg, var(--sea) 0%, #159fd4 50%, var(--leaf) 100%);
background-size: 200% 200%;
box-shadow: 0 8px 24px rgba(26,127,196,.28);
animation: gradShift 4s ease infinite;
color: var(--white);
background: linear-gradient(
135deg,
var(--sea) 0%,
#159fd4 50%,
var(--leaf) 100%
);
background-size: 200% 200%;
box-shadow: 0 8px 24px rgba(26, 127, 196, 0.28);
animation: gradShift 4s ease infinite;
}
.btn--primary:hover:not(:disabled) { box-shadow: 0 12px 32px rgba(26,127,196,.4); }
.btn--primary:hover:not(:disabled) {
box-shadow: 0 12px 32px rgba(26, 127, 196, 0.4);
}
@keyframes gradShift {
0% { background-position: 0% 50%; }
50% { background-position: 100% 50%; }
100% { background-position: 0% 50%; }
0% {
background-position: 0% 50%;
}
50% {
background-position: 100% 50%;
}
100% {
background-position: 0% 50%;
}
}
.btn--ghost {
color: var(--ink-soft);
background: transparent;
border: 1.5px solid var(--line);
color: var(--ink-soft);
background: transparent;
border: 1.5px solid var(--line);
}
.btn--ghost:hover:not(:disabled) { border-color: var(--ink-muted); color: var(--ink); }
.btn--ghost:hover:not(:disabled) {
border-color: var(--ink-muted);
color: var(--ink);
}
.btn--danger {
color: var(--white);
background: var(--danger);
box-shadow: 0 6px 18px rgba(224,92,92,.25);
color: var(--white);
background: var(--danger);
box-shadow: 0 6px 18px rgba(224, 92, 92, 0.25);
}
.btn--full { width: 100%; }
.btn--full {
width: 100%;
}
.btn--sm { padding: 10px 20px; font-size: .88rem; border-radius: var(--r-md); }
.btn--sm {
padding: 10px 20px;
font-size: 0.88rem;
border-radius: var(--r-md);
}
/* ─── CARD ─── */
.card {
background: var(--glass);
backdrop-filter: blur(12px);
border: 1px solid rgba(255,255,255,.8);
border-radius: var(--r-xl);
box-shadow: var(--shadow);
background: var(--glass);
backdrop-filter: blur(12px);
border: 1px solid rgba(255, 255, 255, 0.8);
border-radius: var(--r-xl);
box-shadow: var(--shadow);
}
/* ─── SECTION LABELS ─── */
.section-label {
display: inline-block;
padding: 5px 13px;
border-radius: 999px;
background: var(--sea-dim);
color: var(--sea);
font-size: .78rem;
font-weight: 600;
letter-spacing: .06em;
text-transform: uppercase;
margin-bottom: 16px;
display: inline-block;
padding: 5px 13px;
border-radius: 999px;
background: var(--sea-dim);
color: var(--sea);
font-size: 0.78rem;
font-weight: 600;
letter-spacing: 0.06em;
text-transform: uppercase;
margin-bottom: 16px;
}
.section-title {
font-family: 'Syne', sans-serif;
font-weight: 800;
font-size: clamp(2rem, 4vw, 3.6rem);
line-height: 1.04;
letter-spacing: -.03em;
margin-bottom: 16px;
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: clamp(2rem, 4vw, 3.6rem);
line-height: 1.04;
letter-spacing: -0.03em;
margin-bottom: 16px;
}
/* ─── FORM ─── */
.form {
display: grid;
gap: 20px;
}
.field {
display: grid;
gap: 8px;
}
.field label {
font-size: 0.84rem;
font-weight: 600;
color: var(--ink-soft);
letter-spacing: 0.02em;
}
.field input {
width: 100%;
height: 54px;
padding: 0 18px;
border: 1.5px solid var(--line);
border-radius: var(--r-md);
background: var(--cream);
color: var(--ink);
font-size: 0.98rem;
transition:
border-color 0.18s,
box-shadow 0.18s;
}
.field input::placeholder {
color: var(--ink-muted);
}
.field input:focus {
outline: none;
border-color: var(--sea-light);
box-shadow: 0 0 0 3px rgba(26, 127, 196, 0.12);
background: var(--white);
}
.field input.input--error {
border-color: var(--danger);
box-shadow: 0 0 0 3px rgba(224, 92, 92, 0.12);
}
.field-error {
font-size: 0.82rem;
color: var(--danger);
min-height: 1.2em;
}
/* ─── FOOTER ─── */
@ -285,7 +397,6 @@ button, input { font: inherit; }
text-align: center;
}
.footer__label {
font-size: 0.7rem;
font-weight: 700;
@ -315,30 +426,88 @@ button, input { font: inherit; }
/* ─── SCROLL REVEAL ─── */
.reveal {
opacity: 0;
transform: translateY(24px);
transition: opacity .55s ease, transform .55s ease;
opacity: 0;
transform: translateY(24px);
transition:
opacity 0.55s ease,
transform 0.55s ease;
}
.reveal.visible { opacity: 1; transform: translateY(0); }
.reveal-delay-1 { transition-delay: .1s; }
.reveal-delay-2 { transition-delay: .2s; }
.reveal-delay-3 { transition-delay: .32s; }
.reveal.visible {
opacity: 1;
transform: translateY(0);
}
.reveal-delay-1 {
transition-delay: 0.1s;
}
.reveal-delay-2 {
transition-delay: 0.2s;
}
.reveal-delay-3 {
transition-delay: 0.32s;
}
/* ─── NOSCRIPT ─── */
.noscript-msg {
padding: 1.5rem 2rem;
background: var(--danger);
color: var(--white);
font-weight: 600;
text-align: center;
}
/* ─── REDUCED MOTION ─── */
@media (prefers-reduced-motion: reduce) {
*,
*::before,
*::after {
animation-duration: 0.01ms !important;
animation-iteration-count: 1 !important;
transition-duration: 0.01ms !important;
}
}
/* ─── RESPONSIVE BASE ─── */
@media (max-width: 680px) {
.container { padding: 0 18px; }
.nav__link { display: none; }
.footer__inner { grid-template-columns: 1fr; }
.footer__center, .footer__right { justify-self: start; text-align: left; }
.socials { justify-content: flex-start; }
.footer__right { justify-content: flex-start; }
.container {
padding: 0 18px;
}
.nav__link {
display: none;
}
.footer__inner {
grid-template-columns: 1fr;
}
.footer__center,
.footer__right {
justify-self: start;
text-align: left;
}
.socials {
justify-content: flex-start;
}
.footer__right {
justify-content: flex-start;
}
}
@media (max-width: 360px) {
:root { --r-xl: 22px; --r-lg: 16px; }
.container { padding: 0 14px; }
.logo__mark { width: 34px; height: 34px; }
.logo__text { font-size: 1.05rem; }
.nav__cta { padding: 7px 12px; font-size: .82rem; }
:root {
--r-xl: 22px;
--r-lg: 16px;
}
.container {
padding: 0 14px;
}
.logo__mark {
width: 34px;
height: 34px;
}
.logo__text {
font-size: 1.05rem;
}
.nav__cta {
padding: 7px 12px;
font-size: 0.82rem;
}
}

230
frontend/styles/results.css Normal file
View File

@ -0,0 +1,230 @@
/* results.css — results page styles */
.results-page {
min-height: calc(100vh - var(--hh));
display: grid;
place-items: center;
padding: 60px 0;
}
.results-inner {
width: 100%;
max-width: 680px;
display: grid;
gap: 24px;
}
/* ─── HEADER ─── */
.results-header {
text-align: center;
}
.results-emoji {
font-size: 3.5rem;
margin-bottom: 16px;
display: block;
animation: popIn 0.5s cubic-bezier(0.34, 1.56, 0.64, 1) both;
}
@keyframes popIn {
from {
transform: scale(0);
opacity: 0;
}
to {
transform: scale(1);
opacity: 1;
}
}
.results-title {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: clamp(2rem, 5vw, 3.4rem);
letter-spacing: -0.03em;
line-height: 1.05;
margin-bottom: 8px;
}
.results-player {
font-size: 1rem;
color: var(--ink-soft);
}
.results-player strong {
color: var(--ink);
}
/* ─── TOTAL SCORE ─── */
.score-total-card {
background: linear-gradient(
135deg,
var(--sea) 0%,
#159fd4 50%,
var(--leaf) 100%
);
border-radius: var(--r-xl);
padding: 36px 40px;
text-align: center;
color: var(--white);
box-shadow: 0 16px 40px rgba(26, 127, 196, 0.3);
animation: slideUp 0.5s 0.1s cubic-bezier(0.34, 1.56, 0.64, 1) both;
}
@keyframes slideUp {
from {
transform: translateY(30px);
opacity: 0;
}
to {
transform: translateY(0);
opacity: 1;
}
}
.score-total-label {
font-size: 0.78rem;
font-weight: 700;
letter-spacing: 0.1em;
text-transform: uppercase;
opacity: 0.78;
margin-bottom: 8px;
}
.score-total-num {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: clamp(4rem, 10vw, 6rem);
line-height: 1;
letter-spacing: -0.04em;
}
.score-total-max {
font-size: 1.4rem;
opacity: 0.6;
font-weight: 400;
}
.score-grade {
display: inline-block;
margin-top: 12px;
padding: 4px 18px;
border-radius: 999px;
background: rgba(255, 255, 255, 0.22);
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 1.1rem;
letter-spacing: 0.04em;
}
/* ─── ROUND BREAKDOWN ─── */
.rounds-card {
background: var(--white);
border: 1px solid var(--line);
border-radius: var(--r-xl);
padding: 28px 32px;
box-shadow: var(--shadow-sm);
animation: slideUp 0.5s 0.2s cubic-bezier(0.34, 1.56, 0.64, 1) both;
}
.rounds-card-title {
font-family: "Syne", sans-serif;
font-weight: 700;
font-size: 1rem;
letter-spacing: -0.01em;
margin-bottom: 20px;
color: var(--ink-soft);
}
.round-rows {
display: grid;
gap: 14px;
}
.round-row {
display: grid;
grid-template-columns: auto 1fr auto;
align-items: center;
gap: 16px;
}
.round-row__label {
font-size: 0.82rem;
font-weight: 600;
color: var(--ink-soft);
white-space: nowrap;
}
.round-row__bar-wrap {
height: 8px;
background: var(--cream);
border-radius: 999px;
overflow: hidden;
}
.round-row__bar {
height: 100%;
border-radius: 999px;
background: linear-gradient(90deg, var(--sea), var(--leaf));
transition: width 0.8s cubic-bezier(0.34, 1.56, 0.64, 1);
}
.round-row__score {
font-family: "Syne", sans-serif;
font-weight: 800;
font-size: 0.9rem;
color: var(--sea);
min-width: 36px;
text-align: right;
}
/* ─── COUNTRY TAGS ─── */
.countries-row {
display: flex;
flex-wrap: wrap;
gap: 8px;
margin-top: 8px;
}
.country-tag {
padding: 5px 12px;
border-radius: 999px;
background: var(--sea-dim);
color: var(--sea);
font-size: 0.8rem;
font-weight: 600;
}
/* ─── ACTIONS ─── */
.results-actions {
display: flex;
gap: 12px;
flex-wrap: wrap;
animation: slideUp 0.5s 0.3s cubic-bezier(0.34, 1.56, 0.64, 1) both;
}
.results-actions .btn {
flex: 1 1 180px;
}
/* ─── RESPONSIVE ─── */
@media (max-width: 600px) {
.results-page {
padding: 40px 0;
}
.score-total-card {
padding: 28px 24px;
}
.rounds-card {
padding: 22px 20px;
}
.results-actions .btn {
flex-basis: 100%;
}
}
@media (max-width: 360px) {
.score-total-num {
font-size: 3.4rem;
}
}