mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-06-05 20:54:14 +00:00
Sweep deprecated props across the React tree: - Modal: destroyOnClose -> destroyOnHidden, maskClosable -> mask.closable - Space: direction -> orientation (or removed when redundant) - Input.Group compact -> Space.Compact block - Drawer: width -> size - Spin: tip -> description - Progress: trailColor -> railColor - Alert: message -> title - Popover: overlayClassName -> rootClassName - BackTop -> FloatButton.BackTop Also refresh dashboard theming for v6: rename dark/ultra Layout and Menu tokens (siderBg, darkItemBg, darkSubMenuItemBg, darkPopupBg), tweak gauge size/stroke, add font-size overrides for Statistic and Progress so the overview numbers stay legible under v6 defaults.
137 lines
4.2 KiB
TypeScript
137 lines
4.2 KiB
TypeScript
import { createContext, useCallback, useContext, useEffect, useMemo, useState } from 'react';
|
|
import type { ReactNode } from 'react';
|
|
import { theme as antdTheme } from 'antd';
|
|
import type { ThemeConfig } from 'antd';
|
|
|
|
const STORAGE_DARK = 'dark-mode';
|
|
const STORAGE_ULTRA = 'isUltraDarkThemeEnabled';
|
|
|
|
function readBool(key: string, fallback: boolean): boolean {
|
|
const raw = localStorage.getItem(key);
|
|
if (raw === null) return fallback;
|
|
return raw === 'true';
|
|
}
|
|
|
|
function applyDom(isDark: boolean, isUltra: boolean) {
|
|
document.body.setAttribute('class', isDark ? 'dark' : 'light');
|
|
if (isUltra) {
|
|
document.documentElement.setAttribute('data-theme', 'ultra-dark');
|
|
} else {
|
|
document.documentElement.removeAttribute('data-theme');
|
|
}
|
|
const msg = document.getElementById('message');
|
|
if (msg) msg.className = isDark ? 'dark' : 'light';
|
|
}
|
|
|
|
// Mirror the Vue useTheme module: apply current localStorage state at
|
|
// module load so the document is in the right theme before React mounts.
|
|
const initialDark = readBool(STORAGE_DARK, true);
|
|
const initialUltra = readBool(STORAGE_ULTRA, false);
|
|
applyDom(initialDark, initialUltra);
|
|
|
|
const DARK_TOKENS = {
|
|
colorBgBase: '#1e1e1e',
|
|
colorBgLayout: '#1e1e1e',
|
|
colorBgContainer: '#252526',
|
|
colorBgElevated: '#2d2d30',
|
|
};
|
|
const ULTRA_DARK_TOKENS = {
|
|
colorBgBase: '#000',
|
|
colorBgLayout: '#000',
|
|
colorBgContainer: '#0a0a0a',
|
|
colorBgElevated: '#141414',
|
|
};
|
|
const DARK_LAYOUT_TOKENS = {
|
|
bodyBg: '#1e1e1e',
|
|
headerBg: '#252526',
|
|
headerColor: '#ffffff',
|
|
footerBg: '#1e1e1e',
|
|
siderBg: '#252526',
|
|
triggerBg: '#333333',
|
|
triggerColor: '#ffffff',
|
|
};
|
|
const ULTRA_DARK_LAYOUT_TOKENS = {
|
|
bodyBg: '#000',
|
|
headerBg: '#0a0a0a',
|
|
headerColor: '#ffffff',
|
|
footerBg: '#000',
|
|
siderBg: '#0a0a0a',
|
|
triggerBg: '#141414',
|
|
triggerColor: '#ffffff',
|
|
};
|
|
const DARK_MENU_TOKENS = {
|
|
darkItemBg: '#252526',
|
|
darkSubMenuItemBg: '#1e1e1e',
|
|
darkPopupBg: '#252526',
|
|
};
|
|
const ULTRA_DARK_MENU_TOKENS = {
|
|
darkItemBg: '#0a0a0a',
|
|
darkSubMenuItemBg: '#000',
|
|
darkPopupBg: '#0a0a0a',
|
|
};
|
|
|
|
export function buildAntdThemeConfig(isDark: boolean, isUltra: boolean): ThemeConfig {
|
|
if (!isDark) {
|
|
return { algorithm: antdTheme.defaultAlgorithm };
|
|
}
|
|
return {
|
|
algorithm: antdTheme.darkAlgorithm,
|
|
token: isUltra ? ULTRA_DARK_TOKENS : DARK_TOKENS,
|
|
components: {
|
|
Layout: isUltra ? ULTRA_DARK_LAYOUT_TOKENS : DARK_LAYOUT_TOKENS,
|
|
Menu: isUltra ? ULTRA_DARK_MENU_TOKENS : DARK_MENU_TOKENS,
|
|
},
|
|
};
|
|
}
|
|
|
|
export function pauseAnimationsUntilLeave(elementId: string): void {
|
|
document.documentElement.setAttribute('data-theme-animations', 'off');
|
|
const el = document.getElementById(elementId);
|
|
if (!el) return;
|
|
const restore = () => {
|
|
document.documentElement.removeAttribute('data-theme-animations');
|
|
el.removeEventListener('mouseleave', restore);
|
|
el.removeEventListener('touchend', restore);
|
|
};
|
|
el.addEventListener('mouseleave', restore);
|
|
el.addEventListener('touchend', restore);
|
|
}
|
|
|
|
interface ThemeContextValue {
|
|
isDark: boolean;
|
|
isUltra: boolean;
|
|
toggleTheme: () => void;
|
|
toggleUltra: () => void;
|
|
antdThemeConfig: ThemeConfig;
|
|
}
|
|
|
|
const ThemeContext = createContext<ThemeContextValue | null>(null);
|
|
|
|
export function ThemeProvider({ children }: { children: ReactNode }) {
|
|
const [isDark, setIsDark] = useState<boolean>(initialDark);
|
|
const [isUltra, setIsUltra] = useState<boolean>(initialUltra);
|
|
|
|
useEffect(() => {
|
|
applyDom(isDark, isUltra);
|
|
localStorage.setItem(STORAGE_DARK, String(isDark));
|
|
localStorage.setItem(STORAGE_ULTRA, String(isUltra));
|
|
}, [isDark, isUltra]);
|
|
|
|
const toggleTheme = useCallback(() => setIsDark((v) => !v), []);
|
|
const toggleUltra = useCallback(() => setIsUltra((v) => !v), []);
|
|
|
|
const antdThemeConfig = useMemo(() => buildAntdThemeConfig(isDark, isUltra), [isDark, isUltra]);
|
|
|
|
const value = useMemo<ThemeContextValue>(
|
|
() => ({ isDark, isUltra, toggleTheme, toggleUltra, antdThemeConfig }),
|
|
[isDark, isUltra, toggleTheme, toggleUltra, antdThemeConfig],
|
|
);
|
|
|
|
return <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>;
|
|
}
|
|
|
|
export function useTheme(): ThemeContextValue {
|
|
const ctx = useContext(ThemeContext);
|
|
if (!ctx) throw new Error('useTheme must be used inside <ThemeProvider>');
|
|
return ctx;
|
|
}
|