import React, { Suspense, lazy, useEffect, useMemo, useState } from "react"; import { Box, Button, CssBaseline, IconButton, Typography } from "@mui/material"; import useMediaQuery from "@mui/material/useMediaQuery"; import { CssVarsProvider } from "@mui/material/styles"; import DashboardIcon from "@mui/icons-material/Dashboard"; import WorkOutlineIcon from "@mui/icons-material/WorkOutline"; import ViewKanbanIcon from "@mui/icons-material/ViewKanban"; import BusinessIcon from "@mui/icons-material/Business"; import DeleteOutlineIcon from "@mui/icons-material/DeleteOutline"; import SettingsIcon from "@mui/icons-material/Settings"; import AlarmIcon from "@mui/icons-material/Alarm"; import AccountCircleIcon from "@mui/icons-material/AccountCircle"; import ShieldIcon from "@mui/icons-material/Shield"; import SearchIcon from "@mui/icons-material/Search"; import MailOutlineIcon from "@mui/icons-material/MailOutline"; import MemoryIcon from "@mui/icons-material/Memory"; import { Navigate, Route, Routes, useLocation, useNavigate, createBrowserRouter, RouterProvider } from "react-router-dom"; import { getTheme } from "./theme"; import { ToastProvider } from "./toast"; import { ConfirmProvider } from "./confirm"; import { PromptProvider } from "./prompt"; import JobTable from "./components/JobTable"; import type { JobTableColumns } from "./components/JobTable"; import { I18nProvider, useI18n } from "./i18n/I18nProvider"; import LoginPage from "./pages/LoginPage"; import ForgotPasswordPage from "./pages/ForgotPasswordPage"; import ResetPasswordPage from "./pages/ResetPasswordPage"; import RouteErrorPage from "./pages/RouteErrorPage"; import { api } from "./api"; import { clearAuthToken, getAuthToken } from "./auth"; import AppShell, { NavItem } from "./layout/AppShell"; import { clearAccentColor, getAccentColor, getThemeModePref, setAccentColor, setThemeModePref, ThemeModePref } from "./themePrefs"; const AddJobModal = lazy(() => import("./components/AddJobModal")); const KanbanBoard = lazy(() => import("./components/KanbanBoard")); const DashboardView = lazy(() => import("./components/DashboardView")); const CompaniesTable = lazy(() => import("./components/CompaniesTable")); const SettingsView = lazy(() => import("./components/SettingsView")); const RemindersView = lazy(() => import("./components/RemindersView")); const QuickCommandDialog = lazy(() => import("./components/QuickCommandDialog")); const ProfilePage = lazy(() => import("./pages/ProfilePage")); const AdminAuditPage = lazy(() => import("./pages/AdminAuditPage")); const AdminUsersPage = lazy(() => import("./pages/AdminUsersPage")); const AdminSystemPage = lazy(() => import("./pages/AdminSystemPage")); const CorrespondenceInboxPage = lazy(() => import("./pages/CorrespondenceInboxPage")); const GmailReviewPage = lazy(() => import("./pages/GmailReviewPage")); const NotFoundPage = lazy(() => import("./pages/NotFoundPage")); type AuthConfig = { requireAuth: boolean }; type MeResponse = { provider?: "local" | "google" | "external"; id?: string; email?: string; userName?: string; firstName?: string; lastName?: string; displayName?: string; avatarImageDataUrl?: string; roles?: string[]; }; function breadcrumbsFor(path: string, t: (k: any) => string): string[] { if (path.startsWith("/dashboard")) return [t("home"), t("analytics"), t("overview")]; if (path.startsWith("/jobs")) return [t("home"), t("jobApplications")]; if (path.startsWith("/reminders")) return [t("home"), t("reminders")]; if (path.startsWith("/kanban")) return [t("home"), t("kanbanBoard")]; if (path.startsWith("/companies")) return [t("home"), t("companies")]; if (path.startsWith("/correspondence/review")) return [t("home"), "Gmail review queue"]; if (path.startsWith("/correspondence")) return [t("home"), "Correspondence inbox"]; if (path.startsWith("/trash")) return [t("home"), t("trash")]; if (path.startsWith("/settings")) return [t("home"), t("settings")]; if (path.startsWith("/profile")) return [t("home"), t("account"), t("profile")]; if (path.startsWith("/admin/audit")) return [t("home"), t("admin"), t("auditLog")]; if (path.startsWith("/admin/users")) return [t("home"), t("admin"), t("users")]; if (path.startsWith("/admin/system")) return [t("home"), t("admin"), t("system")]; return [t("home")]; } function titleFor(path: string, t: (k: any) => string): string { if (path === "/dashboard") return t("dashboard"); if (path.startsWith("/reminders")) return t("reminders"); if (path.startsWith("/jobs")) return t("jobApplications"); if (path.startsWith("/kanban")) return t("kanbanBoard"); if (path.startsWith("/companies")) return t("companies"); if (path.startsWith("/correspondence/review")) return "Gmail review queue"; if (path.startsWith("/correspondence")) return "Correspondence inbox"; if (path.startsWith("/trash")) return t("trash"); if (path.startsWith("/settings")) return t("settings"); if (path.startsWith("/profile")) return t("profile"); if (path.startsWith("/admin/audit")) return t("auditLog"); if (path.startsWith("/admin/users")) return t("users"); if (path.startsWith("/admin/system")) return t("systemStatus"); return t("appTitle"); } function PageLoader() { return Loading...; } function Shell({ jobPageSize, setJobPageSize, jobColumns, setJobColumns, themeMode, onThemeModeChange, accentColor, onAccentColorChange, onResetAccentColor }: { jobPageSize: 15 | 20 | 25; setJobPageSize: (n: 15 | 20 | 25) => void; jobColumns: JobTableColumns; setJobColumns: (c: JobTableColumns) => void; themeMode: ThemeModePref; onThemeModeChange: (v: ThemeModePref) => void; accentColor: string; onAccentColorChange: (v: string) => void; onResetAccentColor: () => void; }) { const location = useLocation(); const navigate = useNavigate(); const { t } = useI18n(); const compactHeaderActions = useMediaQuery("(max-width:767.95px)"); const [addOpen, setAddOpen] = useState(false); const [quickOpen, setQuickOpen] = useState(false); const [refreshToken, setRefreshToken] = useState(0); const [requireAuth, setRequireAuth] = useState(null); const [isAdmin, setIsAdmin] = useState(false); const [me, setMe] = useState(null); const [mobileDrawerOpen, setMobileDrawerOpen] = useState(false); const [notifCount, setNotifCount] = useState(0); const path = location.pathname; const isJobs = path.startsWith("/jobs"); useEffect(() => { api.get("/auth/config").then((r) => setRequireAuth(Boolean(r.data?.requireAuth))).catch(() => setRequireAuth(false)); }, []); useEffect(() => { api.get("/auth/me").then((r) => { setMe(r.data); setIsAdmin(Boolean(r.data?.roles?.includes("Admin"))); }).catch(() => { setMe(null); setIsAdmin(false); }); }, []); useEffect(() => { const load = () => { api.get("/jobapplications/reminders", { params: { upcomingDays: 14 } }).then((r) => setNotifCount(Array.isArray(r.data) ? r.data.length : 0)).catch(() => setNotifCount(0)); }; load(); const id = window.setInterval(load, 60000); return () => window.clearInterval(id); }, []); useEffect(() => { const onKeyDown = (e: KeyboardEvent) => { if ((e.ctrlKey || e.metaKey) && e.key.toLowerCase() === "k") { e.preventDefault(); setQuickOpen(true); } }; window.addEventListener("keydown", onKeyDown); return () => window.removeEventListener("keydown", onKeyDown); }, []); const token = getAuthToken(); if (requireAuth === null) return Loading...; if (requireAuth && !token) return ; const pageTitle = titleFor(path, t); const breadcrumbs = breadcrumbsFor(path, t); const setAndPersistPageSize = (n: 15 | 20 | 25) => { setJobPageSize(n); window.localStorage.setItem("jobPageSize", String(n)); }; const setAndPersistColumns = (next: JobTableColumns) => { setJobColumns(next); window.localStorage.setItem("jobColumns", JSON.stringify(next)); }; const fullName = [me?.firstName, me?.lastName].filter(Boolean).join(" "); const nav: NavItem[] = [ { to: "/dashboard", label: t("dashboard"), icon: , section: t("manage") }, { to: "/jobs", label: t("jobApplications"), icon: , section: t("manage") }, { to: "/reminders", label: t("reminders"), icon: , badgeCount: notifCount, section: t("manage") }, { to: "/kanban", label: t("kanbanBoard"), icon: , section: t("manage") }, { to: "/companies", label: t("companies"), icon: , section: t("manage") }, { to: "/correspondence", label: "Correspondence", icon: , section: t("manage") }, { to: "/correspondence/review", label: "Gmail review", icon: , section: t("manage") }, { to: "/trash", label: t("trash"), icon: , section: t("manage") }, ]; const navBottom: NavItem[] = [ { to: "/admin/audit", label: t("auditLog"), icon: , hidden: !isAdmin, section: t("admin") }, { to: "/admin/users", label: t("users"), icon: , hidden: !isAdmin, section: t("admin") }, { to: "/admin/system", label: t("system"), icon: , hidden: !isAdmin, section: t("admin") }, { to: "/profile", label: t("profile"), icon: , section: t("account") }, { to: "/settings", label: t("settings"), icon: , section: t("account") }, ]; const rightActions = ( {compactHeaderActions ? ( setQuickOpen(true)} sx={{ border: "1px solid", borderColor: "divider", borderRadius: 2.5, width: 42, height: 42, flex: "0 0 auto" }} > ) : ( )} {isJobs ? ( ) : null} ); return ( <> { setMobileDrawerOpen(false); navigate(to); }} user={{ email: me?.email, userName: me?.userName, displayName: me?.displayName || fullName || undefined, avatarImageDataUrl: me?.avatarImageDataUrl, roleLabel: isAdmin ? t("superAdmin") : t("user") }} notificationsCount={notifCount} onOpenNotifications={() => navigate("/reminders")} onOpenSettings={() => navigate("/settings")} onOpenProfile={() => navigate("/profile")} onSignOut={() => { clearAuthToken(); navigate("/login"); }} rightActions={rightActions} > }> } /> } /> } /> } /> } /> } /> } /> } /> } /> } /> } /> } /> } /> } /> } /> setAddOpen(false)} onCreated={() => { setRefreshToken((t) => t + 1); }} /> setQuickOpen(false)} onNavigate={(to) => navigate(to)} onOpenAddJob={() => setAddOpen(true)} /> ); } export default function App() { const systemPrefersDark = useMediaQuery("(prefers-color-scheme: dark)", { defaultMatches: true, noSsr: true }); const [themeMode, setThemeMode] = useState(() => getThemeModePref()); const [accentColor, setAccentColorState] = useState(() => getAccentColor()); const effectiveMode: "light" | "dark" = themeMode === "light" ? "light" : themeMode === "dark" ? "dark" : systemPrefersDark ? "dark" : "light"; const theme = useMemo(() => getTheme(effectiveMode, accentColor), [effectiveMode, accentColor]); useEffect(() => { const sync = () => { setThemeMode(getThemeModePref()); setAccentColorState(getAccentColor()); }; window.addEventListener("auth-changed", sync); return () => window.removeEventListener("auth-changed", sync); }, []); const onThemeModeChange = (v: ThemeModePref) => { setThemeModePref(v); setThemeMode(v); }; const onAccentColorChange = (v: string) => { setAccentColor(v); setAccentColorState(getAccentColor()); }; const onResetAccentColor = () => { clearAccentColor(); setAccentColorState(getAccentColor()); }; const [jobPageSize, setJobPageSize] = useState<15 | 20 | 25>(() => { const raw = window.localStorage.getItem("jobPageSize"); const n = raw ? Number(raw) : 15; return (n === 20 || n === 25 ? n : 15) as 15 | 20 | 25; }); const [jobColumns, setJobColumns] = useState(() => { const raw = window.localStorage.getItem("jobColumns"); if (!raw) return { status: true, dateApplied: true, daysSince: true, jobUrl: false }; try { const p = JSON.parse(raw) as Partial; return { status: p.status ?? true, dateApplied: p.dateApplied ?? true, daysSince: p.daysSince ?? true, jobUrl: p.jobUrl ?? false }; } catch { return { status: true, dateApplied: true, daysSince: true, jobUrl: false }; } }); const router = useMemo(() => createBrowserRouter([ { path: "/login", element: , errorElement: }, { path: "/forgot-password", element: , errorElement: }, { path: "/reset-password", element: , errorElement: }, { path: "/*", element: , errorElement: }, ], { future: { v7_relativeSplatPath: true } }), [jobColumns, jobPageSize, themeMode, accentColor]); return ( ); }