2025-08-28 10:41:04 +02:00
|
|
|
import { Suspense } from 'react';
|
|
|
|
|
import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
|
|
|
|
|
import { ReactQueryDevtools } from '@tanstack/react-query-devtools';
|
2025-10-16 07:28:04 +02:00
|
|
|
import { BrowserRouter, useNavigate } from 'react-router-dom';
|
2025-09-22 11:04:03 +02:00
|
|
|
import { I18nextProvider } from 'react-i18next';
|
2025-08-03 19:23:20 +02:00
|
|
|
import { Toaster } from 'react-hot-toast';
|
2025-09-26 07:46:25 +02:00
|
|
|
import { ErrorBoundary } from './components/layout/ErrorBoundary';
|
2025-09-26 12:12:17 +02:00
|
|
|
import { LoadingSpinner } from './components/ui';
|
2025-08-28 10:41:04 +02:00
|
|
|
import { AppRouter } from './router/AppRouter';
|
|
|
|
|
import { ThemeProvider } from './contexts/ThemeContext';
|
|
|
|
|
import { AuthProvider } from './contexts/AuthContext';
|
|
|
|
|
import { SSEProvider } from './contexts/SSEContext';
|
2025-10-29 06:58:05 +01:00
|
|
|
import { SubscriptionEventsProvider } from './contexts/SubscriptionEventsContext';
|
2025-09-21 13:27:50 +02:00
|
|
|
import GlobalSubscriptionHandler from './components/auth/GlobalSubscriptionHandler';
|
2025-10-16 07:28:04 +02:00
|
|
|
import { CookieBanner } from './components/ui/CookieConsent';
|
2025-11-07 22:40:08 +00:00
|
|
|
import { useTenantInitializer } from './stores/useTenantInitializer';
|
2025-09-22 11:04:03 +02:00
|
|
|
import i18n from './i18n';
|
2025-08-11 07:01:08 +02:00
|
|
|
|
2025-08-28 10:41:04 +02:00
|
|
|
const queryClient = new QueryClient({
|
|
|
|
|
defaultOptions: {
|
|
|
|
|
queries: {
|
|
|
|
|
staleTime: 5 * 60 * 1000,
|
|
|
|
|
gcTime: 10 * 60 * 1000,
|
|
|
|
|
retry: 3,
|
|
|
|
|
refetchOnWindowFocus: false,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
});
|
2025-08-03 19:23:20 +02:00
|
|
|
|
2025-10-16 07:28:04 +02:00
|
|
|
function AppContent() {
|
|
|
|
|
const navigate = useNavigate();
|
|
|
|
|
|
2025-11-07 22:40:08 +00:00
|
|
|
// Initialize tenant data when user is authenticated or in demo mode
|
|
|
|
|
useTenantInitializer();
|
|
|
|
|
|
2025-10-16 07:28:04 +02:00
|
|
|
return (
|
|
|
|
|
<>
|
|
|
|
|
<Suspense fallback={<LoadingSpinner overlay />}>
|
|
|
|
|
<AppRouter />
|
|
|
|
|
<GlobalSubscriptionHandler />
|
|
|
|
|
<CookieBanner onPreferencesClick={() => navigate('/cookie-preferences')} />
|
|
|
|
|
<Toaster
|
|
|
|
|
position="top-right"
|
|
|
|
|
toastOptions={{
|
|
|
|
|
duration: 4000,
|
|
|
|
|
style: {
|
|
|
|
|
background: '#363636',
|
|
|
|
|
color: '#fff',
|
|
|
|
|
},
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</Suspense>
|
|
|
|
|
</>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-28 10:41:04 +02:00
|
|
|
function App() {
|
2025-08-16 20:13:40 +02:00
|
|
|
return (
|
|
|
|
|
<ErrorBoundary>
|
2025-08-28 10:41:04 +02:00
|
|
|
<QueryClientProvider client={queryClient}>
|
2025-09-22 11:04:03 +02:00
|
|
|
<I18nextProvider i18n={i18n}>
|
|
|
|
|
<BrowserRouter
|
|
|
|
|
future={{
|
|
|
|
|
v7_startTransition: true,
|
|
|
|
|
v7_relativeSplatPath: true,
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<ThemeProvider>
|
|
|
|
|
<AuthProvider>
|
2025-10-16 07:28:04 +02:00
|
|
|
<SSEProvider>
|
2025-10-29 06:58:05 +01:00
|
|
|
<SubscriptionEventsProvider>
|
|
|
|
|
<AppContent />
|
|
|
|
|
</SubscriptionEventsProvider>
|
2025-10-16 07:28:04 +02:00
|
|
|
</SSEProvider>
|
2025-09-22 11:04:03 +02:00
|
|
|
</AuthProvider>
|
|
|
|
|
</ThemeProvider>
|
|
|
|
|
</BrowserRouter>
|
|
|
|
|
<ReactQueryDevtools initialIsOpen={false} />
|
|
|
|
|
</I18nextProvider>
|
2025-08-28 10:41:04 +02:00
|
|
|
</QueryClientProvider>
|
2025-08-16 20:13:40 +02:00
|
|
|
</ErrorBoundary>
|
|
|
|
|
);
|
2025-08-28 10:41:04 +02:00
|
|
|
}
|
2025-08-03 19:23:20 +02:00
|
|
|
|
2025-10-29 06:58:05 +01:00
|
|
|
export default App;
|