2026-04-05 01:17:47 +02:00
|
|
|
import React, { useState, useEffect, useCallback } from 'react';
|
2026-04-04 21:47:01 +02:00
|
|
|
import ReactDOM from 'react-dom/client';
|
2026-04-05 01:17:47 +02:00
|
|
|
import { LogtoProvider, useLogto } from '@logto/react';
|
2026-04-04 21:55:21 +02:00
|
|
|
import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
|
|
|
|
|
import { BrowserRouter } from 'react-router';
|
2026-04-05 01:30:18 +02:00
|
|
|
import { ThemeProvider, ToastProvider, BreadcrumbProvider, GlobalFilterProvider, CommandPaletteProvider, Spinner } from '@cameleer/design-system';
|
2026-04-04 21:55:21 +02:00
|
|
|
import '@cameleer/design-system/style.css';
|
|
|
|
|
import { AppRouter } from './router';
|
2026-04-05 01:17:47 +02:00
|
|
|
import { fetchConfig } from './config';
|
|
|
|
|
import { setTokenProvider, setLogoutHandler } from './api/client';
|
2026-04-04 21:55:21 +02:00
|
|
|
|
|
|
|
|
const queryClient = new QueryClient({
|
|
|
|
|
defaultOptions: {
|
|
|
|
|
queries: {
|
|
|
|
|
retry: 1,
|
|
|
|
|
staleTime: 10_000,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
});
|
2026-04-04 21:47:01 +02:00
|
|
|
|
2026-04-05 01:17:47 +02:00
|
|
|
function TokenSync({ resource }: { resource: string }) {
|
|
|
|
|
const { getAccessToken, isAuthenticated, signOut } = useLogto();
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
if (isAuthenticated && resource) {
|
|
|
|
|
setTokenProvider(() => getAccessToken(resource));
|
|
|
|
|
} else {
|
|
|
|
|
setTokenProvider(null);
|
|
|
|
|
}
|
|
|
|
|
}, [isAuthenticated, getAccessToken, resource]);
|
|
|
|
|
|
|
|
|
|
const handleLogout = useCallback(() => {
|
|
|
|
|
signOut(window.location.origin + '/login');
|
|
|
|
|
}, [signOut]);
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
setLogoutHandler(handleLogout);
|
|
|
|
|
return () => setLogoutHandler(null);
|
|
|
|
|
}, [handleLogout]);
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function App() {
|
|
|
|
|
const [config, setConfig] = useState<{
|
|
|
|
|
logtoEndpoint: string;
|
|
|
|
|
logtoClientId: string;
|
|
|
|
|
logtoResource: string;
|
|
|
|
|
} | null>(null);
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
fetchConfig().then(setConfig);
|
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
if (!config?.logtoClientId) {
|
|
|
|
|
return (
|
|
|
|
|
<div style={{ display: 'flex', alignItems: 'center', justifyContent: 'center', minHeight: '100vh' }}>
|
|
|
|
|
<Spinner />
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<LogtoProvider
|
|
|
|
|
config={{
|
|
|
|
|
endpoint: config.logtoEndpoint,
|
|
|
|
|
appId: config.logtoClientId,
|
|
|
|
|
resources: config.logtoResource ? [config.logtoResource] : [],
|
|
|
|
|
scopes: ['openid', 'profile', 'email', 'offline_access'],
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<TokenSync resource={config.logtoResource} />
|
|
|
|
|
<QueryClientProvider client={queryClient}>
|
|
|
|
|
<BrowserRouter>
|
|
|
|
|
<AppRouter />
|
|
|
|
|
</BrowserRouter>
|
|
|
|
|
</QueryClientProvider>
|
|
|
|
|
</LogtoProvider>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-04 21:47:01 +02:00
|
|
|
ReactDOM.createRoot(document.getElementById('root')!).render(
|
|
|
|
|
<React.StrictMode>
|
2026-04-04 21:55:21 +02:00
|
|
|
<ThemeProvider>
|
|
|
|
|
<ToastProvider>
|
|
|
|
|
<BreadcrumbProvider>
|
2026-04-05 01:23:36 +02:00
|
|
|
<GlobalFilterProvider>
|
2026-04-05 01:30:18 +02:00
|
|
|
<CommandPaletteProvider>
|
|
|
|
|
<App />
|
|
|
|
|
</CommandPaletteProvider>
|
2026-04-05 01:23:36 +02:00
|
|
|
</GlobalFilterProvider>
|
2026-04-04 21:55:21 +02:00
|
|
|
</BreadcrumbProvider>
|
|
|
|
|
</ToastProvider>
|
|
|
|
|
</ThemeProvider>
|
|
|
|
|
</React.StrictMode>,
|
2026-04-04 21:47:01 +02:00
|
|
|
);
|