frontend: add color theme & theme switcher

This commit is contained in:
Kentai Radiquum 2024-04-19 22:58:45 +05:00
parent 84b13a2c80
commit b8878c4fb8
Signed by: Radiquum
GPG key ID: 858E8EE696525EED
7 changed files with 192 additions and 19 deletions

21
frontend/app/App.jsx Normal file
View file

@ -0,0 +1,21 @@
"use client";
import { NavigationRail } from "@/app/components/NavigationRail/NavigationRail";
import { useThemeStore } from "./store/theme-store";
import { useEffect } from "react";
// import { useStore } from "./store/app-store";
export const App = (props) => {
const themeStore = useThemeStore();
useEffect(() => {
themeStore.checkTheme();
}, []);
return (
<body className={themeStore.theme}>
<NavigationRail />
<main className="responsive">{props.children}</main>
</body>
);
};

View file

@ -1,10 +1,13 @@
"use client";
import { usePathname } from "next/navigation";
import { useThemeStore } from "@/app/store/theme-store";
import Link from "next/link";
export const NavigationRail = () => {
const pathname = usePathname();
const themeStore = useThemeStore();
return (
<nav className="left">
<button className="circle transparent ">
@ -13,24 +16,38 @@ export const NavigationRail = () => {
<Link href="/" className={pathname == "/" ? "active" : ""}>
<i>home</i>
<div>Home</div>
</Link>
<Link href="/favorites" className={pathname == "/favorites" ? "active" : ""}>
<i>favorite</i>
<div>Favorites</div>
<div>Домашняя</div>
</Link>
<Link href="/search" className={pathname == "/search" ? "active" : ""}>
<i>search</i>
<div>Search</div>
<div>Поиск</div>
</Link>
<Link
href="/bookmarks"
className={pathname == "/bookmarks" ? "active" : ""}
>
<i>bookmark</i>
<div>Закладки</div>
</Link>
<Link
href="/favorites"
className={pathname == "/favorites" ? "active" : ""}
>
<i>favorite</i>
<div>Избранное</div>
</Link>
<Link href="/history" className={pathname == "/history" ? "active" : ""}>
<i>history</i>
<div>History</div>
<div>История</div>
</Link>
{/* <a>
<i>share</i>
<div>share</div>
</a> */}
<span className="max"></span>
<button className="circle transparent end-align" onClick={() => themeStore.changeTheme(themeStore.theme == "dark" ? "light" : "dark")}>
<i>palette</i>
</button>
</nav>
);
};

View file

@ -11,3 +11,86 @@ body {
text-wrap: balance;
}
}
body, nav.left{
transition: background .2s;
transform-origin: left;
}
body.light.light {
--primary: #bc004b;
--on-primary: #ffffff;
--primary-container: #ffd9de;
--on-primary-container: #400014;
--secondary: #75565b;
--on-secondary: #ffffff;
--secondary-container: #ffd9de;
--on-secondary-container: #2c1519;
--tertiary: #795831;
--on-tertiary: #ffffff;
--tertiary-container: #ffddba;
--on-tertiary-container: #2b1700;
--error: #ba1a1a;
--on-error: #ffffff;
--error-container: #ffdad6;
--on-error-container: #410002;
--background: #fffbff;
--on-background: #201a1b;
--surface: #fff8f7;
--on-surface: #201a1b;
--surface-variant: #f3dddf;
--on-surface-variant: #524345;
--outline: #847375;
--outline-variant: #d6c2c3;
--shadow: #000000;
--scrim: #000000;
--inverse-surface: #362f2f;
--inverse-on-surface: #fbeeee;
--inverse-primary: #ffb2be;
--surface-dim: #e3d7d8;
--surface-bright: #fff8f7;
--surface-container-lowest: #ffffff;
--surface-container-low: #fdf1f1;
--surface-container: #f8ebeb;
--surface-container-high: #f2e5e6;
--surface-container-highest: #ece0e0;
}
body.dark.dark {
--primary: #ffb2be;
--on-primary: #660025;
--primary-container: #900038;
--on-primary-container: #ffd9de;
--secondary: #e5bdc2;
--on-secondary: #43292d;
--secondary-container: #5c3f43;
--on-secondary-container: #ffd9de;
--tertiary: #ebbf90;
--on-tertiary: #452b08;
--tertiary-container: #5f411c;
--on-tertiary-container: #ffddba;
--error: #ffb4ab;
--on-error: #690005;
--error-container: #93000a;
--on-error-container: #ffb4ab;
--background: #201a1b;
--on-background: #ece0e0;
--surface: #181213;
--on-surface: #ece0e0;
--surface-variant: #524345;
--on-surface-variant: #d6c2c3;
--outline: #9f8c8e;
--outline-variant: #524345;
--shadow: #000000;
--scrim: #000000;
--inverse-surface: #ece0e0;
--inverse-on-surface: #362f2f;
--inverse-primary: #bc004b;
--surface-dim: #181213;
--surface-bright: #3f3738;
--surface-container-lowest: #120d0d;
--surface-container-low: #201a1b;
--surface-container: #241e1f;
--surface-container-high: #2f2829;
--surface-container-highest: #3a3334;
}

View file

@ -1,21 +1,16 @@
import "./globals.css";
import "beercss";
import "material-dynamic-colors";
import { NavigationRail } from "@/components/NavigationRail";
import {App} from "@/app/App"
export const metadata = {
title: "AniX",
description: "Unofficial web app for anixart",
description: "Неофициальное веб приложение для anixart",
};
export default function RootLayout({ children }) {
export default async function RootLayout({ children }) {
return (
<html lang="en">
<body>
<NavigationRail />
<main className="responsive">{children}</main>
</body>
<html lang="ru">
<App>{children}</App>
</html>
);
}

View file

@ -0,0 +1,20 @@
"use client";
import { create } from "zustand";
export function setTheme(theme) {
localStorage.setItem("theme", theme);
}
export function getTheme() {
return localStorage.getItem("theme");
}
export const useThemeStore = create((set) => ({
theme: "light",
changeTheme: (theme) => {
set({ theme: theme });
setTheme(theme);
},
checkTheme: () => {
set({ theme: getTheme() || "light" });
}
}));

View file

@ -12,7 +12,8 @@
"material-dynamic-colors": "^1.1.0",
"next": "14.2.2",
"react": "^18",
"react-dom": "^18"
"react-dom": "^18",
"zustand": "^4.5.2"
},
"devDependencies": {
"postcss": "^8",
@ -1764,6 +1765,14 @@
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz",
"integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q=="
},
"node_modules/use-sync-external-store": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.2.0.tgz",
"integrity": "sha512-eEgnFxGQ1Ife9bzYs6VLi8/4X6CObHMw9Qr9tPY43iKwsPw8xE8+EFsf/2cFZ5S3esXgpWgtSCtLNS41F+sKPA==",
"peerDependencies": {
"react": "^16.8.0 || ^17.0.0 || ^18.0.0"
}
},
"node_modules/util-deprecate": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
@ -1907,6 +1916,33 @@
"engines": {
"node": ">= 14"
}
},
"node_modules/zustand": {
"version": "4.5.2",
"resolved": "https://registry.npmjs.org/zustand/-/zustand-4.5.2.tgz",
"integrity": "sha512-2cN1tPkDVkwCy5ickKrI7vijSjPksFRfqS6237NzT0vqSsztTNnQdHw9mmN7uBdk3gceVXU0a+21jFzFzAc9+g==",
"dependencies": {
"use-sync-external-store": "1.2.0"
},
"engines": {
"node": ">=12.7.0"
},
"peerDependencies": {
"@types/react": ">=16.8",
"immer": ">=9.0.6",
"react": ">=16.8"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
},
"immer": {
"optional": true
},
"react": {
"optional": true
}
}
}
}
}

View file

@ -13,7 +13,8 @@
"material-dynamic-colors": "^1.1.0",
"next": "14.2.2",
"react": "^18",
"react-dom": "^18"
"react-dom": "^18",
"zustand": "^4.5.2"
},
"devDependencies": {
"postcss": "^8",