gbclient/app/theme-provider.tsx

90 lines
2.9 KiB
TypeScript
Raw Permalink Normal View History

// themes/theme-provider.tsx
'use client'
import React, { createContext, useContext, useEffect, useState } from 'react'
type Theme = {
name: string
label: string
cssFile: string
}
const themes: Theme[] = [
{ name: '3dbevel', label: '3dbevel', cssFile: '/themes/3dbevel.css' },
{ name: 'arcadeflash', label: 'Arcadeflash', cssFile: '/themes/arcadeflash.css' },
{ name: 'jazzage', label: 'Jazzage', cssFile: '/themes/jazzage.css' },
{ name: 'mellowgold', label: 'Mellowgold', cssFile: '/themes/mellowgold.css' },
{ name: 'midcenturymod', label: 'Midcenturymod', cssFile: '/themes/midcenturymod.css' },
{ name: 'polaroidmemories', label: 'Polaroidmemories', cssFile: '/themes/polaroidmemories.css' },
{ name: 'retrowave', label: 'Retrowave', cssFile: '/themes/retrowave.css' },
{ name: 'saturdaycartoons', label: 'Saturdaycartoons', cssFile: '/themes/saturdaycartoons.css' },
{ name: 'typewriter', label: 'Typewriter', cssFile: '/themes/typewriter.css' },
{ name: 'vapordream', label: 'Vapordream', cssFile: '/themes/vapordream.css' },
{ name: 'xeroxui', label: 'Xeroxui', cssFile: '/themes/xeroxui.css' },
{ name: 'y2kglow', label: 'Y2kglow', cssFile: '/themes/y2kglow.css' },
{ name: 'xtreegold', label: 'XTreeGold', cssFile: '/themes/xtreegold.css' }
]
type ThemeContextType = {
theme: Theme
setTheme: (themeName: string) => void
themes: Theme[]
}
const ThemeContext = createContext<ThemeContextType | undefined>(undefined)
export function ThemeProvider({ children }: { children: React.ReactNode }) {
const [currentTheme, setCurrentTheme] = useState<Theme>(themes[0])
const [isLoaded, setIsLoaded] = useState(false)
useEffect(() => {
// Load saved theme from localStorage
const savedThemeName = localStorage.getItem('theme') || themes[0].name
const savedTheme = themes.find(t => t.name === savedThemeName) || themes[0]
setTheme(savedTheme.name)
setIsLoaded(true)
}, [])
const setTheme = (themeName: string) => {
const theme = themes.find(t => t.name === themeName)
if (!theme) return
// Remove any existing theme link
const existingLink = document.getElementById('theme-stylesheet')
if (existingLink) {
document.head.removeChild(existingLink)
}
// Create new link element
const link = document.createElement('link')
link.id = 'theme-stylesheet'
link.rel = 'stylesheet'
link.href = theme.cssFile
link.onload = () => {
setCurrentTheme(theme)
localStorage.setItem('theme', theme.name)
}
document.head.appendChild(link)
}
if (!isLoaded) {
return null // or loading spinner
}
return (
<ThemeContext.Provider value={{ theme: currentTheme, setTheme, themes }}>
{children}
</ThemeContext.Provider>
)
}
export function useTheme() {
const context = useContext(ThemeContext)
if (!context) {
throw new Error('useTheme must be used within a ThemeProvider')
}
return context
}