Files
VibeFinance/frontend/src/App.jsx
T
2026-04-17 17:43:07 +02:00

178 lines
7.0 KiB
React

import "./App.css";
import { AppProvider, useApp } from "./context/AppContext.jsx";
import { useEffect, useRef, useState } from "react";
import LoginPage from "./components/LoginPage.jsx";
import NavBar from "./components/NavBar.jsx";
import GebruikersBeheer from "./components/GebruikersBeheer.jsx";
import DashboardTab from "./pages/DashboardTab.jsx";
import EigenVermogenTab from "./pages/EigenVermogenTab.jsx";
import SchuldTab from "./pages/SchuldTab.jsx";
import VoortgangTab from "./pages/VoortgangTab.jsx";
import { RED, PURPLE, PURPLE_LIGHT } from "./constants/index.js";
import ProfielPopup from "./components/ProfielPopup.jsx";
import InstellingenModal from "./components/InstellingenModal.jsx";
function initials(naam = "") {
const parts = naam.trim().split(/\s+/);
if (parts.length >= 2) return (parts[0][0] + parts[parts.length - 1][0]).toUpperCase();
return naam.slice(0, 2).toUpperCase();
}
const PAGES = [DashboardTab, EigenVermogenTab, SchuldTab, VoortgangTab];
function AppInner() {
const { loggedIn, login, logout, locked, tab, showUsers, setShowUsers, showInstellingen, setShowInstellingen, loading, T, currentUser, avatar, idleMinutes } = useApp();
const [profielOpen, setProfielOpen] = useState(false);
const idleTimer = useRef(null);
const IDLE_MS = (idleMinutes ?? 30) * 60 * 1000;
const [sessionVerlopen, setSessionVerlopen] = useState(false);
useEffect(() => {
if (!loggedIn) return;
const reset = () => {
clearTimeout(idleTimer.current);
idleTimer.current = setTimeout(() => {
logout();
setSessionVerlopen(true);
}, IDLE_MS);
};
const events = ["mousemove", "mousedown", "keydown", "touchstart", "scroll"];
events.forEach((e) => window.addEventListener(e, reset, { passive: true }));
reset();
return () => {
events.forEach((e) => window.removeEventListener(e, reset));
clearTimeout(idleTimer.current);
};
}, [loggedIn]);
if (!loggedIn) {
if (sessionVerlopen) {
return (
<div style={{
minHeight: "100vh", background: T.bg, color: T.text,
display: "flex", alignItems: "center", justifyContent: "center",
fontFamily: "'Inter', system-ui, sans-serif",
}}>
<div style={{
background: T.card, border: `1px solid ${T.border}`,
borderRadius: 20, padding: "40px 36px", maxWidth: 360, width: "100%",
textAlign: "center", boxShadow: "0 16px 48px rgba(0,0,0,0.4)",
}}>
<div style={{ fontSize: 52, marginBottom: 16 }}></div>
<div style={{ fontSize: 20, fontWeight: 800, marginBottom: 10 }}>Sessie verlopen</div>
<div style={{ fontSize: 14, color: T.muted, marginBottom: 28, lineHeight: 1.6 }}>
Je bent automatisch uitgelogd na 30 minuten inactiviteit. Klik hieronder om opnieuw in te loggen.
</div>
<button onClick={() => { setSessionVerlopen(false); }} style={{
width: "100%", padding: "14px",
background: "linear-gradient(135deg, #8b5cf6, #a855f7)",
border: "none", borderRadius: 12,
color: "#fff", fontWeight: 700, fontSize: 15, cursor: "pointer",
boxShadow: "0 4px 16px rgba(139,92,246,0.4)",
}}>
Opnieuw inloggen
</button>
</div>
</div>
);
}
return <LoginPage onLogin={login} />;
}
if (loading) {
return (
<div style={{
minHeight: "100vh", background: T.bg, color: T.text,
display: "flex", alignItems: "center", justifyContent: "center",
flexDirection: "column", gap: 16,
fontFamily: "'Inter', system-ui, sans-serif",
}}>
<div style={{
width: 40, height: 40, borderRadius: "50%",
border: `3px solid rgba(139,92,246,0.2)`,
borderTop: `3px solid ${PURPLE_LIGHT}`,
animation: "spin 0.8s linear infinite",
}} />
<style>{`@keyframes spin { to { transform: rotate(360deg); } }`}</style>
<div style={{ color: T.muted, fontSize: 14 }}>Data laden</div>
</div>
);
}
const ActivePage = PAGES[tab] ?? DashboardTab;
return (
<div style={{ display: "flex", minHeight: "100vh", background: T.bg, color: T.text, fontFamily: "'Inter', system-ui, sans-serif" }}>
<style>{`
* { box-sizing: border-box; }
input[type=number]::-webkit-inner-spin-button { opacity: 0.4; }
`}</style>
<NavBar />
{/* Hoofdinhoud */}
<div style={{ flex: 1, overflowY: "auto", maxHeight: "100vh" }}>
{/* Topbalk rechts */}
<div style={{ display: "flex", justifyContent: "flex-end", alignItems: "center", gap: 8, padding: "12px 20px 0" }}>
<div style={{ position: "relative" }}>
<button onClick={() => setProfielOpen((o) => !o)} style={{
display: "flex", alignItems: "center", gap: 6,
background: "transparent", border: "none", cursor: "pointer", padding: 0,
}}>
<div style={{
width: 44, height: 44, borderRadius: "50%", overflow: "hidden", flexShrink: 0,
background: avatar ? "transparent" : (profielOpen ? `linear-gradient(135deg, ${PURPLE}, #a855f7)` : `${PURPLE}33`),
border: `1px solid ${profielOpen ? PURPLE : PURPLE + "55"}`,
display: "flex", alignItems: "center", justifyContent: "center",
fontSize: 13, fontWeight: 800, color: profielOpen ? "#fff" : PURPLE_LIGHT,
}}>
{avatar
? <img src={avatar} alt="avatar" style={{ width: "100%", height: "100%", objectFit: "cover" }} />
: initials(currentUser?.naam || "")
}
</div>
</button>
{profielOpen && <ProfielPopup onClose={() => setProfielOpen(false)} />}
</div>
</div>
{/* Vergrendeld banner */}
{locked && (
<div style={{
position: "sticky", top: 16, zIndex: 150,
margin: "16px auto 0", maxWidth: 400,
background: "rgba(239,68,68,0.15)",
border: "1px solid rgba(239,68,68,0.4)",
borderRadius: 99, padding: "8px 20px",
display: "flex", alignItems: "center", gap: 8,
backdropFilter: "blur(8px)",
}}>
<span style={{ fontSize: 14 }}>🔒</span>
<span style={{ fontSize: 13, color: RED, fontWeight: 600 }}>
Vergrendeld ga naar Gebruikers om te ontgrendelen
</span>
</div>
)}
{/* Modals */}
{showUsers && <GebruikersBeheer onClose={() => setShowUsers(false)} />}
{showInstellingen && <InstellingenModal onClose={() => setShowInstellingen(false)} />}
{/* Actieve pagina */}
<div style={{ maxWidth: 1200, margin: "0 auto", padding: "24px 16px" }}>
<ActivePage />
</div>
</div>
</div>
);
}
export default function App() {
return (
<AppProvider>
<AppInner />
</AppProvider>
);
}