2022-11-09 12:28:04 +08:00
|
|
|
import React, { type FC, useLayoutEffect } from 'react';
|
|
|
|
import { useOutlet } from 'dumi';
|
|
|
|
import { ConfigProvider, theme as antdTheme } from 'antd';
|
|
|
|
import { ThemeConfig } from 'antd/es/config-provider/context';
|
|
|
|
import ThemeContext, { ThemeContextProps } from '../slots/ThemeContext';
|
2022-11-22 17:00:28 +08:00
|
|
|
import ThemeSwitch from '../common/ThemeSwitch';
|
|
|
|
import useLocation from '../../hooks/useLocation';
|
2022-11-09 12:28:04 +08:00
|
|
|
|
|
|
|
const ANT_DESIGN_SITE_THEME = 'antd-site-theme';
|
|
|
|
|
2022-11-22 17:00:28 +08:00
|
|
|
const getAlgorithm = (theme: string) => {
|
|
|
|
if (theme === 'dark') {
|
|
|
|
return antdTheme.darkAlgorithm;
|
|
|
|
}
|
|
|
|
if (theme === 'compact') {
|
|
|
|
return antdTheme.compactAlgorithm;
|
|
|
|
}
|
|
|
|
return antdTheme.defaultAlgorithm;
|
|
|
|
};
|
|
|
|
|
|
|
|
const getThemeString = (algorithm: typeof antdTheme.defaultAlgorithm) => {
|
|
|
|
if (algorithm === antdTheme.darkAlgorithm) {
|
|
|
|
return 'dark';
|
|
|
|
}
|
|
|
|
if (algorithm === antdTheme.compactAlgorithm) {
|
|
|
|
return 'compact';
|
|
|
|
}
|
|
|
|
return 'light';
|
|
|
|
};
|
|
|
|
|
2022-11-09 12:28:04 +08:00
|
|
|
const GlobalLayout: FC = () => {
|
|
|
|
const outlet = useOutlet();
|
2022-11-22 17:00:28 +08:00
|
|
|
const { pathname } = useLocation();
|
2022-11-09 12:28:04 +08:00
|
|
|
|
2022-11-22 17:00:28 +08:00
|
|
|
const [theme, setTheme] = React.useState<ThemeConfig>({
|
|
|
|
algorithm: [antdTheme.defaultAlgorithm],
|
|
|
|
});
|
|
|
|
|
|
|
|
const handleThemeChange = (newTheme: ThemeConfig, ignoreAlgorithm: boolean = true) => {
|
|
|
|
const nextTheme = { ...newTheme };
|
|
|
|
if (ignoreAlgorithm) {
|
|
|
|
nextTheme.algorithm = theme.algorithm;
|
|
|
|
}
|
|
|
|
setTheme(nextTheme);
|
|
|
|
localStorage.setItem(
|
|
|
|
ANT_DESIGN_SITE_THEME,
|
|
|
|
JSON.stringify(nextTheme, (key, value) => {
|
|
|
|
if (key === 'algorithm') {
|
|
|
|
return Array.isArray(value) ? value.map((item) => getThemeString(item)) : ['light'];
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
};
|
2022-11-09 12:28:04 +08:00
|
|
|
|
|
|
|
const contextValue = React.useMemo<ThemeContextProps>(
|
|
|
|
() => ({
|
|
|
|
theme,
|
2022-11-22 17:00:28 +08:00
|
|
|
setTheme: handleThemeChange,
|
2022-11-09 12:28:04 +08:00
|
|
|
}),
|
|
|
|
[theme],
|
|
|
|
);
|
|
|
|
|
|
|
|
useLayoutEffect(() => {
|
|
|
|
const localTheme = localStorage.getItem(ANT_DESIGN_SITE_THEME);
|
|
|
|
if (localTheme) {
|
|
|
|
try {
|
|
|
|
const themeConfig = JSON.parse(localTheme);
|
2022-11-22 17:00:28 +08:00
|
|
|
if (themeConfig.algorithm) {
|
|
|
|
themeConfig.algorithm = themeConfig.algorithm.map((item: string) => getAlgorithm(item));
|
2022-11-09 12:28:04 +08:00
|
|
|
}
|
|
|
|
setTheme(themeConfig);
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<ThemeContext.Provider value={contextValue}>
|
|
|
|
<ConfigProvider
|
|
|
|
theme={{
|
|
|
|
...theme,
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{outlet}
|
2022-11-22 17:00:28 +08:00
|
|
|
{!pathname.startsWith('/~demos') && (
|
|
|
|
<ThemeSwitch
|
|
|
|
value={theme.algorithm as []}
|
|
|
|
onChange={(value) => handleThemeChange({ ...theme, algorithm: value }, false)}
|
|
|
|
/>
|
|
|
|
)}
|
2022-11-09 12:28:04 +08:00
|
|
|
</ConfigProvider>
|
|
|
|
</ThemeContext.Provider>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default GlobalLayout;
|