2eab705a8a
- Added 'custom' accent option with color picker + hex input - Pipette icon on the custom swatch (switches to solid when active) - Color picker appears inline when custom is selected - Generates hover/light variants automatically from hex (darken/lighten) - Dark mode accent-light uses rgba for translucency - Persisted in localStorage, applied before React hydration (no flash) Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
72 lines
2.3 KiB
TypeScript
72 lines
2.3 KiB
TypeScript
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; customHex?: string } }
|
|
applyTheme(state.mode ?? 'light', state.accent ?? 'green', state.customHex)
|
|
}
|
|
} 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 customHex = useThemeStore((s) => s.customHex)
|
|
const resolvedTheme = resolveTheme(mode)
|
|
|
|
// Apply whenever mode, accent, or customHex changes
|
|
useEffect(() => {
|
|
applyTheme(mode, accent, customHex)
|
|
}, [mode, accent, customHex])
|
|
|
|
// 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, customHex)
|
|
mq.addEventListener('change', handler)
|
|
return () => mq.removeEventListener('change', handler)
|
|
}, [mode, accent, customHex])
|
|
|
|
return (
|
|
<>
|
|
{children}
|
|
<Toaster position="top-left" richColors theme={resolvedTheme} />
|
|
</>
|
|
)
|
|
}
|
|
|
|
ReactDOM.createRoot(document.getElementById('root')!).render(
|
|
<React.StrictMode>
|
|
<QueryClientProvider client={queryClient}>
|
|
<ThemeProvider>
|
|
<App />
|
|
</ThemeProvider>
|
|
</QueryClientProvider>
|
|
</React.StrictMode>,
|
|
)
|