feat: add bottom navigation bar (mobile-first)

Fixed bottom nav with Home/Spiele/Rangliste/Profil tabs.
Desktop keeps header nav. Admin hidden behind gear icon.
Main content padded to avoid overlap with bottom nav.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Ronny
2026-04-11 18:59:00 +02:00
parent dbe71dcb97
commit 636228ff04
5 changed files with 117 additions and 4 deletions
+27
View File
@@ -96,5 +96,32 @@
max-width: 1100px; max-width: 1100px;
margin: 0 auto; margin: 0 auto;
padding: 32px 24px; padding: 32px 24px;
padding-bottom: 70px;
width: 100%; width: 100%;
} }
@media (min-width: 768px) {
.main {
padding-bottom: 32px;
}
}
/* Hide header nav on mobile */
@media (max-width: 767px) {
.nav {
display: none;
}
}
/* Admin link: icon only, subtle */
.adminLink {
display: flex;
align-items: center;
padding: 4px;
color: var(--text-muted);
text-decoration: none;
transition: color 0.2s;
}
.adminLink:hover {
color: var(--text-secondary);
}
+8 -4
View File
@@ -1,10 +1,11 @@
import { useState, useEffect } from 'react'; import { useState, useEffect } from 'react';
import { Routes, Route, NavLink } from 'react-router-dom'; import { Routes, Route, NavLink } from 'react-router-dom';
import { Sun, Moon } from 'lucide-react'; import { Sun, Moon, Settings } from 'lucide-react';
import MatchesPage from './pages/MatchesPage'; import MatchesPage from './pages/MatchesPage';
import LeaderboardPage from './pages/LeaderboardPage'; import LeaderboardPage from './pages/LeaderboardPage';
import ProfilePage from './pages/ProfilePage'; import ProfilePage from './pages/ProfilePage';
import AdminPage from './pages/AdminPage'; import AdminPage from './pages/AdminPage';
import BottomNav from './components/BottomNav';
import styles from './App.module.css'; import styles from './App.module.css';
const IS_DEV = import.meta.env.DEV || import.meta.env.VITE_TEST_MODE === 'true'; const IS_DEV = import.meta.env.DEV || import.meta.env.VITE_TEST_MODE === 'true';
@@ -79,7 +80,7 @@ export default function App() {
)} )}
</div> </div>
<nav className={styles.nav}> <nav className={styles.nav}>
<NavLink to="/" end className={({ isActive }) => isActive ? styles.navLinkActive : styles.navLink}> <NavLink to="/spiele" className={({ isActive }) => isActive ? styles.navLinkActive : styles.navLink}>
Spielplan Spielplan
</NavLink> </NavLink>
<NavLink to="/rangliste" className={({ isActive }) => isActive ? styles.navLinkActive : styles.navLink}> <NavLink to="/rangliste" className={({ isActive }) => isActive ? styles.navLinkActive : styles.navLink}>
@@ -88,8 +89,8 @@ export default function App() {
<NavLink to="/profil" className={({ isActive }) => isActive ? styles.navLinkActive : styles.navLink}> <NavLink to="/profil" className={({ isActive }) => isActive ? styles.navLinkActive : styles.navLink}>
Mein Profil Mein Profil
</NavLink> </NavLink>
<NavLink to="/admin" className={({ isActive }) => isActive ? styles.navLinkActive : styles.navLink}> <NavLink to="/admin" className={styles.adminLink} title="Admin">
Admin <Settings size={16} />
</NavLink> </NavLink>
<button <button
className={styles.themeToggle} className={styles.themeToggle}
@@ -106,12 +107,15 @@ export default function App() {
<main className={styles.main}> <main className={styles.main}>
<Routes> <Routes>
<Route path="/" element={<MatchesPage key={refreshKey} devUser={devUser} />} /> <Route path="/" element={<MatchesPage key={refreshKey} devUser={devUser} />} />
<Route path="/spiele" element={<MatchesPage key={refreshKey} devUser={devUser} />} />
<Route path="/rangliste" element={<LeaderboardPage key={refreshKey} />} /> <Route path="/rangliste" element={<LeaderboardPage key={refreshKey} />} />
<Route path="/profil" element={<ProfilePage key={refreshKey} />} /> <Route path="/profil" element={<ProfilePage key={refreshKey} />} />
<Route path="/admin" element={<AdminPage />} /> <Route path="/admin" element={<AdminPage />} />
</Routes> </Routes>
</main> </main>
<BottomNav />
{IS_DEV && DevPanel && ( {IS_DEV && DevPanel && (
<DevPanel <DevPanel
currentUser={devUser} currentUser={devUser}
@@ -0,0 +1,47 @@
.bottomNav {
position: fixed;
bottom: 0;
left: 0;
right: 0;
display: flex;
justify-content: space-around;
align-items: center;
height: 60px;
padding-bottom: env(safe-area-inset-bottom, 0px);
background: color-mix(in srgb, var(--bg-deep) 92%, transparent);
backdrop-filter: blur(20px);
-webkit-backdrop-filter: blur(20px);
border-top: 1px solid rgba(75, 183, 248, 0.15);
z-index: 100;
}
.tab, .tabActive {
display: flex;
flex-direction: column;
align-items: center;
gap: 2px;
padding: 6px 16px;
font-size: 11px;
text-decoration: none;
color: var(--text-muted);
transition: color 0.2s;
}
.tabActive {
color: var(--primary);
}
.tab:hover {
color: var(--text-secondary);
}
.emojiIcon {
font-size: 20px;
line-height: 1;
}
@media (min-width: 768px) {
.bottomNav {
display: none;
}
}
+29
View File
@@ -0,0 +1,29 @@
import { NavLink } from 'react-router-dom';
import { Home, Trophy, User } from 'lucide-react';
import styles from './BottomNav.module.css';
export default function BottomNav() {
const linkClass = ({ isActive }: { isActive: boolean }) =>
isActive ? styles.tabActive : styles.tab;
return (
<nav className={styles.bottomNav}>
<NavLink to="/" end className={linkClass}>
<Home size={20} />
<span>Home</span>
</NavLink>
<NavLink to="/spiele" className={linkClass}>
<span className={styles.emojiIcon}></span>
<span>Spiele</span>
</NavLink>
<NavLink to="/rangliste" className={linkClass}>
<Trophy size={20} />
<span>Rangliste</span>
</NavLink>
<NavLink to="/profil" className={linkClass}>
<User size={20} />
<span>Profil</span>
</NavLink>
</nav>
);
}
+6
View File
@@ -4,8 +4,14 @@ interface Window {
interface ImportMetaEnv { interface ImportMetaEnv {
readonly VITE_TEST_MODE?: string; readonly VITE_TEST_MODE?: string;
readonly DEV: boolean;
} }
interface ImportMeta { interface ImportMeta {
readonly env: ImportMetaEnv; readonly env: ImportMetaEnv;
} }
declare module '*.module.css' {
const classes: { readonly [key: string]: string };
export default classes;
}