feat: initial commit
This commit is contained in:
@@ -0,0 +1,70 @@
|
||||
import React, { useEffect } from 'react'
|
||||
import ReactDOM from 'react-dom/client'
|
||||
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
|
||||
import { Toaster } from 'sonner'
|
||||
import App from './App'
|
||||
import './index.css'
|
||||
import { useThemeStore, applyTheme, resolveTheme, type ThemeMode, type AccentKey } from './store/theme'
|
||||
|
||||
/* ---------------------------------------------------------------
|
||||
Flash prevention: apply theme BEFORE React hydrates.
|
||||
Reads directly from localStorage to avoid the Zustand wrapper.
|
||||
--------------------------------------------------------------- */
|
||||
;(function () {
|
||||
try {
|
||||
const raw = localStorage.getItem('schaeffler-theme')
|
||||
if (raw) {
|
||||
const { state } = JSON.parse(raw) as { state: { mode: ThemeMode; accent: AccentKey } }
|
||||
applyTheme(state.mode ?? 'light', state.accent ?? 'green')
|
||||
}
|
||||
} catch {
|
||||
// ignore — default theme already applied by CSS
|
||||
}
|
||||
})()
|
||||
|
||||
const queryClient = new QueryClient({
|
||||
defaultOptions: {
|
||||
queries: {
|
||||
staleTime: 30_000,
|
||||
retry: 1,
|
||||
},
|
||||
},
|
||||
})
|
||||
|
||||
/** Subscribes to store changes and system preference changes */
|
||||
function ThemeProvider({ children }: { children: React.ReactNode }) {
|
||||
const mode = useThemeStore((s) => s.mode)
|
||||
const accent = useThemeStore((s) => s.accent)
|
||||
const resolvedTheme = resolveTheme(mode)
|
||||
|
||||
// Apply whenever mode or accent changes
|
||||
useEffect(() => {
|
||||
applyTheme(mode, accent)
|
||||
}, [mode, accent])
|
||||
|
||||
// Listen to system preference changes when mode='system'
|
||||
useEffect(() => {
|
||||
if (mode !== 'system') return
|
||||
const mq = window.matchMedia('(prefers-color-scheme: dark)')
|
||||
const handler = () => applyTheme('system', accent)
|
||||
mq.addEventListener('change', handler)
|
||||
return () => mq.removeEventListener('change', handler)
|
||||
}, [mode, accent])
|
||||
|
||||
return (
|
||||
<>
|
||||
{children}
|
||||
<Toaster position="top-right" richColors theme={resolvedTheme} />
|
||||
</>
|
||||
)
|
||||
}
|
||||
|
||||
ReactDOM.createRoot(document.getElementById('root')!).render(
|
||||
<React.StrictMode>
|
||||
<QueryClientProvider client={queryClient}>
|
||||
<ThemeProvider>
|
||||
<App />
|
||||
</ThemeProvider>
|
||||
</QueryClientProvider>
|
||||
</React.StrictMode>,
|
||||
)
|
||||
Reference in New Issue
Block a user