work on light theme
This commit is contained in:
@ -2,11 +2,12 @@ import React, { createContext } from "react";
|
|||||||
import { ConfigProvider, message } from "antd";
|
import { ConfigProvider, message } from "antd";
|
||||||
import "./App.css";
|
import "./App.css";
|
||||||
import HeaderComponent from "./components/HeaderComponent";
|
import HeaderComponent from "./components/HeaderComponent";
|
||||||
import { theme } from "./config/style";
|
import { darkTheme, lightTheme, theme } from "./config/style";
|
||||||
import { Provider } from "react-redux";
|
import { Provider, useSelector } from "react-redux";
|
||||||
import { store } from "./config/store";
|
import { StorePrototype, store } from "./config/store";
|
||||||
import { MessageInstance } from "antd/es/message/interface";
|
import { MessageInstance } from "antd/es/message/interface";
|
||||||
import AppRoutes from "./pages/AppRoutes";
|
import AppRoutes from "./pages/AppRoutes";
|
||||||
|
import ThemeProviderWrapper from "./config/ThemeProviderWrapper";
|
||||||
|
|
||||||
export const MessageContext = createContext({} as MessageInstance);
|
export const MessageContext = createContext({} as MessageInstance);
|
||||||
|
|
||||||
@ -17,7 +18,7 @@ const App = () => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Provider store={store}>
|
<Provider store={store}>
|
||||||
<ConfigProvider theme={theme}>
|
<ThemeProviderWrapper>
|
||||||
<MessageContext.Provider value={messageApi}>
|
<MessageContext.Provider value={messageApi}>
|
||||||
<div className="content">
|
<div className="content">
|
||||||
{contextHolder}
|
{contextHolder}
|
||||||
@ -26,7 +27,7 @@ const App = () => {
|
|||||||
</AppRoutes>
|
</AppRoutes>
|
||||||
</div>
|
</div>
|
||||||
</MessageContext.Provider>
|
</MessageContext.Provider>
|
||||||
</ConfigProvider>
|
</ThemeProviderWrapper>
|
||||||
</Provider>
|
</Provider>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@ -3,15 +3,23 @@ import {
|
|||||||
GlobalOutlined,
|
GlobalOutlined,
|
||||||
LogoutOutlined,
|
LogoutOutlined,
|
||||||
MenuOutlined,
|
MenuOutlined,
|
||||||
|
MoonOutlined,
|
||||||
PicCenterOutlined,
|
PicCenterOutlined,
|
||||||
SettingOutlined,
|
SettingOutlined,
|
||||||
|
SunOutlined,
|
||||||
UserOutlined,
|
UserOutlined,
|
||||||
} from "@ant-design/icons";
|
} from "@ant-design/icons";
|
||||||
import { Drawer, Layout, Menu, MenuProps } from "antd";
|
import { Drawer, Layout, Menu, MenuProps, Switch } from "antd";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import AuthModal from "./AuthModal";
|
import AuthModal from "./AuthModal";
|
||||||
import "./styles.css";
|
import "./styles.css";
|
||||||
import { StorePrototype, logOut, setLanguage, store } from "../config/store";
|
import {
|
||||||
|
StorePrototype,
|
||||||
|
logOut,
|
||||||
|
setLanguage,
|
||||||
|
setTheme,
|
||||||
|
store,
|
||||||
|
} from "../config/store";
|
||||||
import { useSelector } from "react-redux";
|
import { useSelector } from "react-redux";
|
||||||
import tr from "../config/translation";
|
import tr from "../config/translation";
|
||||||
import { Link, useNavigate } from "react-router-dom";
|
import { Link, useNavigate } from "react-router-dom";
|
||||||
@ -43,6 +51,10 @@ const HeaderComponent = () => {
|
|||||||
(state: StorePrototype) => state.auth.user
|
(state: StorePrototype) => state.auth.user
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const currentTheme: string | undefined = useSelector(
|
||||||
|
(state: StorePrototype) => state.settings.theme
|
||||||
|
);
|
||||||
|
|
||||||
const [selectedKeys, setSelectedKeys] = useState<string[]>([]);
|
const [selectedKeys, setSelectedKeys] = useState<string[]>([]);
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const keys = [];
|
const keys = [];
|
||||||
@ -113,6 +125,18 @@ const HeaderComponent = () => {
|
|||||||
children: languageSelectItems,
|
children: languageSelectItems,
|
||||||
style: { background: "#001529" },
|
style: { background: "#001529" },
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: (
|
||||||
|
<Switch
|
||||||
|
onChange={(v: boolean) =>
|
||||||
|
store.dispatch(setTheme(v ? "light" : "dark"))
|
||||||
|
}
|
||||||
|
defaultChecked={currentTheme === "light"}
|
||||||
|
/>
|
||||||
|
),
|
||||||
|
key: "theme",
|
||||||
|
icon: currentTheme === "dark" ? <MoonOutlined /> : <SunOutlined />,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: user ? user.username : tr("Log in"),
|
label: user ? user.username : tr("Log in"),
|
||||||
key: "login",
|
key: "login",
|
||||||
@ -167,6 +191,18 @@ const HeaderComponent = () => {
|
|||||||
onClick: () => !user && setAuthModalOpen(true),
|
onClick: () => !user && setAuthModalOpen(true),
|
||||||
...(user ? { children: userMenuItems } : {}),
|
...(user ? { children: userMenuItems } : {}),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: (
|
||||||
|
<Switch
|
||||||
|
onChange={(v: boolean) =>
|
||||||
|
store.dispatch(setTheme(v ? "light" : "dark"))
|
||||||
|
}
|
||||||
|
defaultChecked={currentTheme === "light"}
|
||||||
|
/>
|
||||||
|
),
|
||||||
|
key: "theme",
|
||||||
|
icon: currentTheme === "dark" ? <MoonOutlined /> : <SunOutlined />,
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
22
frontend/app/src/config/ThemeProviderWrapper.tsx
Normal file
22
frontend/app/src/config/ThemeProviderWrapper.tsx
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import React, { ReactNode } from "react";
|
||||||
|
import { useSelector } from "react-redux";
|
||||||
|
import { StorePrototype } from "./store";
|
||||||
|
import { ConfigProvider } from "antd";
|
||||||
|
import { darkTheme, lightTheme } from "./style";
|
||||||
|
import PropTypes from "prop-types";
|
||||||
|
|
||||||
|
const ThemeProviderWrapper = ({ children }: { children: ReactNode }) => {
|
||||||
|
const theme = useSelector((state: StorePrototype) => state.settings.theme);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ConfigProvider theme={theme === "dark" ? darkTheme : lightTheme}>
|
||||||
|
{children}
|
||||||
|
</ConfigProvider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
ThemeProviderWrapper.propTypes = {
|
||||||
|
children: PropTypes.node,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ThemeProviderWrapper;
|
||||||
@ -23,10 +23,12 @@ const initialAuthDataState: AuthDataType = {
|
|||||||
|
|
||||||
export type SettingsType = {
|
export type SettingsType = {
|
||||||
language: string | undefined;
|
language: string | undefined;
|
||||||
|
theme: string | undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
const initialSettingsState: SettingsType = {
|
const initialSettingsState: SettingsType = {
|
||||||
language: undefined,
|
language: undefined,
|
||||||
|
theme: undefined,
|
||||||
};
|
};
|
||||||
|
|
||||||
export type StorePrototype = {
|
export type StorePrototype = {
|
||||||
@ -44,6 +46,8 @@ export const logOut = createAction("auth/logOut");
|
|||||||
|
|
||||||
export const setLanguage = createAction<string>("settings/setLanguage");
|
export const setLanguage = createAction<string>("settings/setLanguage");
|
||||||
export const loadLanguage = createAction("settings/loadLanguage");
|
export const loadLanguage = createAction("settings/loadLanguage");
|
||||||
|
export const setTheme = createAction<string>("settings/setTheme");
|
||||||
|
export const loadTheme = createAction("settings/loadTheme");
|
||||||
|
|
||||||
export const store = configureStore({
|
export const store = configureStore({
|
||||||
reducer: {
|
reducer: {
|
||||||
@ -94,6 +98,19 @@ export const store = configureStore({
|
|||||||
state.language = "en";
|
state.language = "en";
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
builder.addCase(setTheme, (state, action) => {
|
||||||
|
state.theme = action.payload || "dark";
|
||||||
|
localStorage.setItem("theme", action.payload || "dark");
|
||||||
|
});
|
||||||
|
builder.addCase(loadTheme, (state) => {
|
||||||
|
const theme: string | null = localStorage.getItem("theme");
|
||||||
|
if (theme) {
|
||||||
|
state.theme = theme;
|
||||||
|
} else {
|
||||||
|
const darkThemeMq = window.matchMedia("(prefers-color-scheme: dark)");
|
||||||
|
state.theme = darkThemeMq.matches ? "dark" : "light";
|
||||||
|
}
|
||||||
|
});
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
// Adding the api middleware enables caching, invalidation, polling,
|
// Adding the api middleware enables caching, invalidation, polling,
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import { ThemeConfig } from "antd";
|
import { ThemeConfig } from "antd";
|
||||||
|
|
||||||
export const theme: ThemeConfig = {
|
export const darkTheme: ThemeConfig = {
|
||||||
token: {
|
token: {
|
||||||
colorText: "white",
|
colorText: "white",
|
||||||
colorIcon: "white",
|
colorIcon: "white",
|
||||||
@ -18,3 +18,13 @@ export const theme: ThemeConfig = {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const lightTheme: ThemeConfig = {
|
||||||
|
token: {
|
||||||
|
colorPrimary: "#00d8a4",
|
||||||
|
colorIconHover: "#00d8a4",
|
||||||
|
borderRadius: 5,
|
||||||
|
fontFamily: "Comfortaa",
|
||||||
|
// colorWarningBg: "#001529",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|||||||
@ -5,6 +5,7 @@ import {
|
|||||||
getLocalClient,
|
getLocalClient,
|
||||||
getLocalToken,
|
getLocalToken,
|
||||||
loadLanguage,
|
loadLanguage,
|
||||||
|
loadTheme,
|
||||||
store,
|
store,
|
||||||
} from "../config/store";
|
} from "../config/store";
|
||||||
import DashboardPage from "./DashboardPage";
|
import DashboardPage from "./DashboardPage";
|
||||||
@ -19,6 +20,7 @@ const AppRoutes = ({ children }: { children: ReactNode }) => {
|
|||||||
store.dispatch(getLocalToken());
|
store.dispatch(getLocalToken());
|
||||||
store.dispatch(getLocalClient());
|
store.dispatch(getLocalClient());
|
||||||
store.dispatch(loadLanguage());
|
store.dispatch(loadLanguage());
|
||||||
|
store.dispatch(loadTheme());
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<BrowserRouter>
|
<BrowserRouter>
|
||||||
|
|||||||
Reference in New Issue
Block a user