The single 12-card SettingsPage is replaced by a hub at /settings linking
to three thematic sub-pages mounted via a shared SettingsLayout (Outlet):
/settings SettingsHomePage (3 cards-cluster + PageHelp)
/settings/users UsersSettingsPage (Account, License, DocsContent)
/settings/data DataSettingsPage (Categories, DataManagement,
PriceFetchConsentToggle)
/settings/systems SystemsSettingsPage (Version, UpdateCard,
ChangelogContent, LogViewer)
DocsPage and ChangelogPage are extracted into reusable DocsContent /
ChangelogContent components and the standalone /docs and /changelog
routes become Navigate redirects to preserve external bookmarks and
release-note links. UpdateCard is extracted from the inline updater
block for symmetry and testability.
TokenStoreFallbackBanner is mounted once in SettingsLayout, surfacing
the OS-keychain-fallback warning across the four main routes only.
The two existing /settings/categories/{standard,migrate} sub-routes
stay flat (siblings of SettingsLayout) to keep their focused flows
free of the banner — their internal back-links now point to
/settings/data.
i18n FR/EN gain settings.{home, users, data, systems, backToHome};
docs/architecture.md and CHANGELOG{,.fr}.md updated. Pure refactor of
presentation: no new business logic, no Tauri commands, no SQL
migrations.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
146 lines
6 KiB
TypeScript
146 lines
6 KiB
TypeScript
import { BrowserRouter, Routes, Route } from "react-router-dom";
|
|
import { useEffect, useState, useRef } from "react";
|
|
import { useTranslation } from "react-i18next";
|
|
import { useProfile } from "./contexts/ProfileContext";
|
|
import AppShell from "./components/layout/AppShell";
|
|
import DashboardPage from "./pages/DashboardPage";
|
|
import ImportPage from "./pages/ImportPage";
|
|
import TransactionsPage from "./pages/TransactionsPage";
|
|
import CategoriesPage from "./pages/CategoriesPage";
|
|
import AdjustmentsPage from "./pages/AdjustmentsPage";
|
|
import BudgetPage from "./pages/BudgetPage";
|
|
import ReportsPage from "./pages/ReportsPage";
|
|
import ReportsHighlightsPage from "./pages/ReportsHighlightsPage";
|
|
import ReportsTrendsPage from "./pages/ReportsTrendsPage";
|
|
import ReportsComparePage from "./pages/ReportsComparePage";
|
|
import ReportsCategoryPage from "./pages/ReportsCategoryPage";
|
|
import ReportsCartesPage from "./pages/ReportsCartesPage";
|
|
import SettingsLayout from "./pages/settings/SettingsLayout";
|
|
import SettingsHomePage from "./pages/settings/SettingsHomePage";
|
|
import UsersSettingsPage from "./pages/settings/UsersSettingsPage";
|
|
import DataSettingsPage from "./pages/settings/DataSettingsPage";
|
|
import SystemsSettingsPage from "./pages/settings/SystemsSettingsPage";
|
|
import AccountsPage from "./pages/AccountsPage";
|
|
import BalancePage from "./pages/BalancePage";
|
|
import SnapshotEditPage from "./pages/SnapshotEditPage";
|
|
import CategoriesStandardGuidePage from "./pages/CategoriesStandardGuidePage";
|
|
import CategoriesMigrationPage from "./pages/CategoriesMigrationPage";
|
|
import DocsPage from "./pages/DocsPage";
|
|
import ChangelogPage from "./pages/ChangelogPage";
|
|
import ProfileSelectionPage from "./pages/ProfileSelectionPage";
|
|
import ErrorPage from "./components/shared/ErrorPage";
|
|
|
|
const STARTUP_TIMEOUT_MS = 10_000;
|
|
const MAX_RETRIES = 3;
|
|
const RETRY_DELAY_MS = 1_000;
|
|
|
|
export default function App() {
|
|
const { t } = useTranslation();
|
|
const { activeProfile, isLoading, refreshKey, connectActiveProfile } = useProfile();
|
|
const [dbReady, setDbReady] = useState(false);
|
|
const [startupError, setStartupError] = useState<string | null>(null);
|
|
const timeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
|
const cancelledRef = useRef(false);
|
|
|
|
useEffect(() => {
|
|
if (activeProfile && !isLoading) {
|
|
setDbReady(false);
|
|
setStartupError(null);
|
|
cancelledRef.current = false;
|
|
|
|
timeoutRef.current = setTimeout(() => {
|
|
setStartupError(t("error.startupTimeout"));
|
|
}, STARTUP_TIMEOUT_MS);
|
|
|
|
const attemptConnect = async (attempt: number): Promise<void> => {
|
|
try {
|
|
await connectActiveProfile();
|
|
if (cancelledRef.current) return;
|
|
if (timeoutRef.current) clearTimeout(timeoutRef.current);
|
|
setDbReady(true);
|
|
} catch (err) {
|
|
if (cancelledRef.current) return;
|
|
console.error(`Failed to connect profile (attempt ${attempt}/${MAX_RETRIES}):`, err);
|
|
if (attempt < MAX_RETRIES) {
|
|
await new Promise((r) => setTimeout(r, RETRY_DELAY_MS));
|
|
if (!cancelledRef.current) return attemptConnect(attempt + 1);
|
|
} else {
|
|
if (timeoutRef.current) clearTimeout(timeoutRef.current);
|
|
setStartupError(err instanceof Error ? err.message : String(err));
|
|
}
|
|
}
|
|
};
|
|
|
|
attemptConnect(1);
|
|
}
|
|
|
|
return () => {
|
|
cancelledRef.current = true;
|
|
if (timeoutRef.current) clearTimeout(timeoutRef.current);
|
|
};
|
|
}, [activeProfile, isLoading, connectActiveProfile, t]);
|
|
|
|
if (isLoading) {
|
|
return (
|
|
<div className="flex items-center justify-center h-screen bg-[var(--background)]">
|
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-[var(--primary)]" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (startupError) {
|
|
return <ErrorPage error={startupError} />;
|
|
}
|
|
|
|
if (!activeProfile) {
|
|
return <ProfileSelectionPage />;
|
|
}
|
|
|
|
if (!dbReady) {
|
|
return (
|
|
<div className="flex items-center justify-center h-screen bg-[var(--background)]">
|
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-[var(--primary)]" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<BrowserRouter key={refreshKey}>
|
|
<Routes>
|
|
<Route element={<AppShell />}>
|
|
<Route path="/" element={<DashboardPage />} />
|
|
<Route path="/import" element={<ImportPage />} />
|
|
<Route path="/transactions" element={<TransactionsPage />} />
|
|
<Route path="/categories" element={<CategoriesPage />} />
|
|
<Route path="/adjustments" element={<AdjustmentsPage />} />
|
|
<Route path="/budget" element={<BudgetPage />} />
|
|
<Route path="/reports" element={<ReportsPage />} />
|
|
<Route path="/reports/highlights" element={<ReportsHighlightsPage />} />
|
|
<Route path="/reports/trends" element={<ReportsTrendsPage />} />
|
|
<Route path="/reports/compare" element={<ReportsComparePage />} />
|
|
<Route path="/reports/category" element={<ReportsCategoryPage />} />
|
|
<Route path="/reports/cartes" element={<ReportsCartesPage />} />
|
|
<Route path="/settings" element={<SettingsLayout />}>
|
|
<Route index element={<SettingsHomePage />} />
|
|
<Route path="users" element={<UsersSettingsPage />} />
|
|
<Route path="data" element={<DataSettingsPage />} />
|
|
<Route path="systems" element={<SystemsSettingsPage />} />
|
|
</Route>
|
|
<Route path="/balance" element={<BalancePage />} />
|
|
<Route path="/balance/accounts" element={<AccountsPage />} />
|
|
<Route path="/balance/snapshot" element={<SnapshotEditPage />} />
|
|
<Route
|
|
path="/settings/categories/standard"
|
|
element={<CategoriesStandardGuidePage />}
|
|
/>
|
|
<Route
|
|
path="/settings/categories/migrate"
|
|
element={<CategoriesMigrationPage />}
|
|
/>
|
|
<Route path="/docs" element={<DocsPage />} />
|
|
<Route path="/changelog" element={<ChangelogPage />} />
|
|
</Route>
|
|
</Routes>
|
|
</BrowserRouter>
|
|
);
|
|
}
|