mirror of
https://github.com/Radiquum/AniX.git
synced 2025-04-06 00:04:39 +00:00
frontend:
Fix undefined api call Simplify some parts Better theme management Add color picker (dynamic themes wohoo) Add missing lists keys
This commit is contained in:
parent
17b5693f34
commit
79782b4228
8 changed files with 157 additions and 137 deletions
|
@ -1,21 +1,47 @@
|
||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import "beercss";
|
||||||
|
import "material-dynamic-colors";
|
||||||
import { NavigationRail } from "@/app/components/NavigationRail/NavigationRail";
|
import { NavigationRail } from "@/app/components/NavigationRail/NavigationRail";
|
||||||
import { useThemeStore } from "./store/theme-store";
|
import { setTheme, getTheme, setMode, getMode } from "./store/theme-store";
|
||||||
import { useEffect } from "react";
|
import { useEffect, useState } from "react";
|
||||||
// import { useStore } from "./store/app-store";
|
import { ColorPicker } from "@/app/components/ColorPicker/ColorPicker";
|
||||||
|
|
||||||
export const App = (props) => {
|
export const App = (props) => {
|
||||||
const themeStore = useThemeStore();
|
const [colorPicker, setColorPicker] = useState(false);
|
||||||
|
|
||||||
|
const theme = async (from) => {
|
||||||
|
setTheme(from);
|
||||||
|
await ui("theme", from);
|
||||||
|
};
|
||||||
|
|
||||||
|
const mode = () => {
|
||||||
|
let newMode = ui("mode") == "dark" ? "light" : "dark";
|
||||||
|
setMode(newMode);
|
||||||
|
ui("mode", getMode());
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
themeStore.checkTheme();
|
const mode = getMode();
|
||||||
|
const theme = getTheme();
|
||||||
|
if (mode != ui("mode")) {
|
||||||
|
ui("mode", getMode());
|
||||||
|
}
|
||||||
|
if (theme != ui("theme")) {
|
||||||
|
ui("theme", theme);
|
||||||
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<body className={themeStore.theme}>
|
<body>
|
||||||
<NavigationRail />
|
<NavigationRail
|
||||||
<main className="responsive">{props.children}</main>
|
colorPicker={colorPicker}
|
||||||
|
setColorPicker={setColorPicker}
|
||||||
|
/>
|
||||||
|
<main className="responsive">
|
||||||
|
{colorPicker && <ColorPicker mode={mode} theme={theme} />}
|
||||||
|
{props.children}
|
||||||
|
</main>
|
||||||
</body>
|
</body>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
48
frontend/app/components/ColorPicker/ColorPicker.jsx
Normal file
48
frontend/app/components/ColorPicker/ColorPicker.jsx
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
"use client";
|
||||||
|
|
||||||
|
import { useState } from "react";
|
||||||
|
import Styles from "./ColorPicker.module.css"
|
||||||
|
|
||||||
|
export const ColorPicker = (props) => {
|
||||||
|
const colors = [
|
||||||
|
{ hex: "#ffffff", color: "white" },
|
||||||
|
{ hex: "#e91e63", color: "pink" },
|
||||||
|
{ hex: "#ff9800", color: "orange" },
|
||||||
|
{ hex: "#4caf50", color: "green" },
|
||||||
|
{ hex: "#009688", color: "teal" },
|
||||||
|
{ hex: "#9c27b0", color: "purple" },
|
||||||
|
{ hex: "#673ab7", color: "deep-purple" },
|
||||||
|
{ hex: "#ffeb3b", color: "yellow" },
|
||||||
|
{ hex: "#ffc8ff", color: Styles["radiquum-pink"]},
|
||||||
|
{ hex: "#0087c7", color: Styles["fuxigen-blue"]},
|
||||||
|
{ hex: "#e54040", color: Styles["anixart-red"]},
|
||||||
|
];
|
||||||
|
const [mode, setMode] = useState(ui("mode"));
|
||||||
|
|
||||||
|
return (
|
||||||
|
<dialog className="active">
|
||||||
|
<h5>Theme Select</h5>
|
||||||
|
<div className="grid center-align">
|
||||||
|
{colors.map((item) => {
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
key={item.color}
|
||||||
|
className={`circle small ${item.color} s2`}
|
||||||
|
onClick={() => props.theme(item.hex)}
|
||||||
|
></button>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
<div className="medium-divider"></div>
|
||||||
|
<button
|
||||||
|
className={`circle small transparent`}
|
||||||
|
onClick={() => {
|
||||||
|
props.mode();
|
||||||
|
setMode(ui("mode"));
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{mode == "light" ? <i>dark_mode</i> : <i>light_mode</i>}
|
||||||
|
</button>
|
||||||
|
</dialog>
|
||||||
|
);
|
||||||
|
};
|
|
@ -0,0 +1,9 @@
|
||||||
|
.radiquum-pink{
|
||||||
|
background-color: #ffc8ff !important;
|
||||||
|
}
|
||||||
|
.fuxigen-blue{
|
||||||
|
background-color: #0087c7 !important;
|
||||||
|
}
|
||||||
|
.anixart-red{
|
||||||
|
background-color: #e54040 !important;
|
||||||
|
}
|
|
@ -1,12 +1,38 @@
|
||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { usePathname } from "next/navigation";
|
import { usePathname } from "next/navigation";
|
||||||
import { useThemeStore } from "@/app/store/theme-store";
|
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
|
||||||
export const NavigationRail = () => {
|
export const NavigationRail = (props) => {
|
||||||
const pathname = usePathname();
|
const pathname = usePathname();
|
||||||
const themeStore = useThemeStore();
|
|
||||||
|
const items = [
|
||||||
|
{
|
||||||
|
title: "Домашняя",
|
||||||
|
icon: "home",
|
||||||
|
path: "/",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "Поиск",
|
||||||
|
icon: "search",
|
||||||
|
path: "/search",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "Закладки",
|
||||||
|
icon: "bookmark",
|
||||||
|
path: "/bookmarks",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "Избранное",
|
||||||
|
icon: "favorite",
|
||||||
|
path: "/favorites",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "История",
|
||||||
|
icon: "history",
|
||||||
|
path: "/history",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<nav className="left">
|
<nav className="left">
|
||||||
|
@ -14,40 +40,27 @@ export const NavigationRail = () => {
|
||||||
<img className="responsive" src="/favicon.ico"></img>
|
<img className="responsive" src="/favicon.ico"></img>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<Link href="/" className={pathname == "/" ? "active" : ""}>
|
{items.map((item) => {
|
||||||
<i>home</i>
|
return (
|
||||||
<div>Домашняя</div>
|
<Link
|
||||||
</Link>
|
key={item.path}
|
||||||
<Link href="/search" className={pathname == "/search" ? "active" : ""}>
|
href={item.path}
|
||||||
<i>search</i>
|
className={pathname == item.path ? "active" : ""}
|
||||||
<div>Поиск</div>
|
>
|
||||||
</Link>
|
<i>{item.icon}</i>
|
||||||
<Link
|
<div>{item.title}</div>
|
||||||
href="/bookmarks"
|
</Link>
|
||||||
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>История</div>
|
|
||||||
</Link>
|
|
||||||
{/* <a>
|
|
||||||
<i>share</i>
|
|
||||||
<div>share</div>
|
|
||||||
</a> */}
|
|
||||||
<span className="max"></span>
|
<span className="max"></span>
|
||||||
<button className="circle transparent" onClick={() => themeStore.changeTheme(themeStore.theme == "dark" ? "light" : "dark")}>
|
<button
|
||||||
|
className="circle transparent"
|
||||||
|
onClick={() => props.setColorPicker(!props.colorPicker)}
|
||||||
|
>
|
||||||
<i>palette</i>
|
<i>palette</i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
</nav>
|
</nav>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -21,81 +21,3 @@ body, nav.left{
|
||||||
transition: background .2s;
|
transition: background .2s;
|
||||||
transform-origin: left;
|
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;
|
|
||||||
}
|
|
|
@ -1,5 +1,4 @@
|
||||||
import "./globals.css";
|
import "./globals.css";
|
||||||
import "beercss";
|
|
||||||
import {App} from "@/app/App"
|
import {App} from "@/app/App"
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = {
|
||||||
|
|
|
@ -28,7 +28,12 @@ export default function Home() {
|
||||||
|
|
||||||
// set list on initial page load
|
// set list on initial page load
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setList(searchParams.get("list") || "last");
|
const query = searchParams.get("list");
|
||||||
|
if (query) {
|
||||||
|
setList(query);
|
||||||
|
} else {
|
||||||
|
setList("last");
|
||||||
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
async function fetchData(list, page = 0) {
|
async function fetchData(list, page = 0) {
|
||||||
|
@ -44,10 +49,12 @@ export default function Home() {
|
||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
router.push(pathname + "?" + createQueryString("list", list));
|
if (list) {
|
||||||
setReleases(null);
|
router.push(pathname + "?" + createQueryString("list", list));
|
||||||
setPage(0);
|
setReleases(null);
|
||||||
fetchData(list); // Call fetchData here
|
setPage(0);
|
||||||
|
fetchData(list); // Call fetchData here
|
||||||
|
}
|
||||||
}, [list]);
|
}, [list]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -81,6 +88,7 @@ export default function Home() {
|
||||||
{chips.map((item) => {
|
{chips.map((item) => {
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
|
key={item.list}
|
||||||
className={`chip ${list == item.list ? "fill" : ""}`}
|
className={`chip ${list == item.list ? "fill" : ""}`}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setList(item.list);
|
setList(item.list);
|
||||||
|
|
|
@ -1,5 +1,10 @@
|
||||||
"use client";
|
"use client";
|
||||||
import { create } from "zustand";
|
export function setMode(mode) {
|
||||||
|
localStorage.setItem("mode", mode);
|
||||||
|
}
|
||||||
|
export function getMode() {
|
||||||
|
return localStorage.getItem("mode");
|
||||||
|
}
|
||||||
|
|
||||||
export function setTheme(theme) {
|
export function setTheme(theme) {
|
||||||
localStorage.setItem("theme", theme);
|
localStorage.setItem("theme", theme);
|
||||||
|
@ -8,13 +13,3 @@ export function getTheme() {
|
||||||
return localStorage.getItem("theme");
|
return localStorage.getItem("theme");
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useThemeStore = create((set) => ({
|
|
||||||
theme: "light",
|
|
||||||
changeTheme: (theme) => {
|
|
||||||
set({ theme: theme });
|
|
||||||
setTheme(theme);
|
|
||||||
},
|
|
||||||
checkTheme: () => {
|
|
||||||
set({ theme: getTheme() || "light" });
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue