f3ec49fe88
- New: Lotus Terminal Mode toggle in Appearance settings - Red phosphor color scheme (bg #0a0000, primary #ff3300, accent #00dd66) - Monospace font override (JetBrains Mono / Fira Code / Cascadia Code) - Retro CRT scanline overlay via CSS pseudo-element - Wired into ThemeManager with dedicated lotusTerminalBodyClass - Branding: replace all user-visible Cinny references with Lotus Chat - WelcomePage, AuthLayout, SplashScreen, index.html meta tags - Device display names in login/register/token flows - System notification brand field - (Preserved internal Matrix protocol CinnySpaces event type) Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
116 lines
3.4 KiB
TypeScript
116 lines
3.4 KiB
TypeScript
import { lightTheme } from 'folds';
|
|
import { createContext, useContext, useEffect, useMemo, useState } from 'react';
|
|
import { onDarkFontWeight, onLightFontWeight } from '../../config.css';
|
|
import { butterTheme, darkTheme, lotusTerminalTheme, silverTheme } from '../../colors.css';
|
|
import { settingsAtom } from '../state/settings';
|
|
import { useSetting } from '../state/hooks/settings';
|
|
|
|
export enum ThemeKind {
|
|
Light = 'light',
|
|
Dark = 'dark',
|
|
}
|
|
|
|
export type Theme = {
|
|
id: string;
|
|
kind: ThemeKind;
|
|
classNames: string[];
|
|
};
|
|
|
|
export const LightTheme: Theme = {
|
|
id: 'light-theme',
|
|
kind: ThemeKind.Light,
|
|
classNames: [lightTheme, onLightFontWeight, 'prism-light'],
|
|
};
|
|
|
|
export const SilverTheme: Theme = {
|
|
id: 'silver-theme',
|
|
kind: ThemeKind.Light,
|
|
classNames: ['silver-theme', silverTheme, onLightFontWeight, 'prism-light'],
|
|
};
|
|
export const DarkTheme: Theme = {
|
|
id: 'dark-theme',
|
|
kind: ThemeKind.Dark,
|
|
classNames: ['dark-theme', darkTheme, onDarkFontWeight, 'prism-dark'],
|
|
};
|
|
export const ButterTheme: Theme = {
|
|
id: 'butter-theme',
|
|
kind: ThemeKind.Dark,
|
|
classNames: ['butter-theme', butterTheme, onDarkFontWeight, 'prism-dark'],
|
|
};
|
|
export const LotusTerminalTheme: Theme = {
|
|
id: 'lotus-terminal-theme',
|
|
kind: ThemeKind.Dark,
|
|
classNames: ['lotus-terminal-theme', lotusTerminalTheme, onDarkFontWeight, 'prism-dark'],
|
|
};
|
|
|
|
export const useThemes = (): Theme[] => {
|
|
const themes: Theme[] = useMemo(() => [LightTheme, SilverTheme, DarkTheme, ButterTheme], []);
|
|
|
|
return themes;
|
|
};
|
|
|
|
export const useThemeNames = (): Record<string, string> =>
|
|
useMemo(
|
|
() => ({
|
|
[LightTheme.id]: 'Light',
|
|
[SilverTheme.id]: 'Silver',
|
|
[DarkTheme.id]: 'Dark',
|
|
[ButterTheme.id]: 'Butter',
|
|
}),
|
|
[]
|
|
);
|
|
|
|
export const useSystemThemeKind = (): ThemeKind => {
|
|
const darkModeQueryList = useMemo(() => window.matchMedia('(prefers-color-scheme: dark)'), []);
|
|
const [themeKind, setThemeKind] = useState<ThemeKind>(
|
|
darkModeQueryList.matches ? ThemeKind.Dark : ThemeKind.Light
|
|
);
|
|
|
|
useEffect(() => {
|
|
const handleMediaQueryChange = () => {
|
|
setThemeKind(darkModeQueryList.matches ? ThemeKind.Dark : ThemeKind.Light);
|
|
};
|
|
|
|
darkModeQueryList.addEventListener('change', handleMediaQueryChange);
|
|
return () => {
|
|
darkModeQueryList.removeEventListener('change', handleMediaQueryChange);
|
|
};
|
|
}, [darkModeQueryList, setThemeKind]);
|
|
|
|
return themeKind;
|
|
};
|
|
|
|
export const useActiveTheme = (): Theme => {
|
|
const systemThemeKind = useSystemThemeKind();
|
|
const themes = useThemes();
|
|
const [systemTheme] = useSetting(settingsAtom, 'useSystemTheme');
|
|
const [themeId] = useSetting(settingsAtom, 'themeId');
|
|
const [lightThemeId] = useSetting(settingsAtom, 'lightThemeId');
|
|
const [darkThemeId] = useSetting(settingsAtom, 'darkThemeId');
|
|
|
|
if (!systemTheme) {
|
|
const selectedTheme = themes.find((theme) => theme.id === themeId) ?? LightTheme;
|
|
|
|
return selectedTheme;
|
|
}
|
|
|
|
const selectedTheme =
|
|
systemThemeKind === ThemeKind.Dark
|
|
? themes.find((theme) => theme.id === darkThemeId) ?? DarkTheme
|
|
: themes.find((theme) => theme.id === lightThemeId) ?? LightTheme;
|
|
|
|
return selectedTheme;
|
|
};
|
|
|
|
const ThemeContext = createContext<Theme | null>(null);
|
|
export const ThemeContextProvider = ThemeContext.Provider;
|
|
|
|
export const useTheme = (): Theme => {
|
|
const theme = useContext(ThemeContext);
|
|
if (!theme) {
|
|
throw new Error('No theme provided!');
|
|
}
|
|
|
|
return theme;
|
|
};
|