Compare commits

...

9 commits

12 changed files with 334 additions and 113 deletions

View file

@ -6,10 +6,10 @@ AniX is an unofficial web client for the Android application Anixart. It allows
## Changelog [RU] ## Changelog [RU]
- [3.5.0](./public/changelog/3.5.0.md)
- [3.4.0](./public/changelog/3.4.0.md) - [3.4.0](./public/changelog/3.4.0.md)
- [3.3.0](./public/changelog/3.3.0.md) - [3.3.0](./public/changelog/3.3.0.md)
- [3.2.3](./public/changelog/3.2.3.md) - [3.2.3](./public/changelog/3.2.3.md)
- [3.2.2](./public/changelog/3.2.2.md)
[other versions](./public/changelog) [other versions](./public/changelog)

16
app/about/page.tsx Normal file
View file

@ -0,0 +1,16 @@
export const metadata = {
title: "О приложении",
openGraph: {
title: "AniX - Неофициальный веб клиент для Anixart",
description:
"AniX - это неофициальный веб-клиент для Android-приложения Anixart. Он позволяет вам получать доступ к своей учетной записи Anixart и управлять ею из веб-браузера. Так-же можно синхронизировать и управлять списками и избранным. И самое главное смотреть все доступные аниме из базы Anixart.",
},
};
export const dynamic = "force-static";
import { AboutPage } from "#/pages/About";
export default function Index() {
return <AboutPage />;
}

View file

@ -1,4 +1,4 @@
export const CURRENT_APP_VERSION = "3.4.0"; export const CURRENT_APP_VERSION = "3.5.0";
export const API_URL = "https://api.anixart.tv"; export const API_URL = "https://api.anixart.tv";
export const API_PREFIX = "/api/proxy"; export const API_PREFIX = "/api/proxy";

View file

@ -87,8 +87,8 @@ export const Navbar = () => {
return ( return (
<> <>
<header className="fixed bottom-0 left-0 z-50 w-full text-white bg-black rounded-t-lg sm:sticky sm:top-0 sm:rounded-t-none sm:rounded-b-lg"> <header className="fixed bottom-0 left-0 z-50 w-full text-white bg-black rounded-t-lg sm:sticky sm:top-0 sm:rounded-t-none sm:rounded-b-lg">
<div className="container flex items-center justify-center gap-4 mx-auto sm:gap-0 sm:justify-between"> <div className={`container flex items-center min-h-[76px] justify-center ${preferenceStore.flags.showFifthButton && preferenceStore.flags.showNavbarTitles == "always" ? "gap-0" : "gap-4"} mx-auto sm:gap-0 sm:justify-between`}>
<div className="flex items-center gap-8 px-2 py-4 sm:gap-4"> <div className={`flex items-center ${preferenceStore.flags.showFifthButton && preferenceStore.flags.showNavbarTitles == "always" ? "gap-4" : "gap-8"} px-2 py-4 sm:gap-4`}>
{menuItems.map((item) => { {menuItems.map((item) => {
return ( return (
<Link <Link
@ -98,7 +98,7 @@ export const Navbar = () => {
item.isAuthRequired && !userStore.isAuth ? "hidden" item.isAuthRequired && !userStore.isAuth ? "hidden"
: item.isShownOnMobile ? "flex" : item.isShownOnMobile ? "flex"
: "hidden sm:flex" : "hidden sm:flex"
} ${[item.href, item.hrefInCategory].includes("/" + pathname.split("/")[1]) ? "font-bold" : "font-medium"}`} } ${[item.href, item.hrefInCategory].includes("/" + pathname.split("/")[1]) ? "font-bold" : "font-medium"} transition-all`}
> >
<span <span
className={`w-6 h-6 iconify ${[item.href, item.hrefInCategory].includes("/" + pathname.split("/")[1]) ? item.icon.active : item.icon.default}`} className={`w-6 h-6 iconify ${[item.href, item.hrefInCategory].includes("/" + pathname.split("/")[1]) ? item.icon.active : item.icon.default}`}
@ -112,13 +112,13 @@ export const Navbar = () => {
); );
})} })}
</div> </div>
<div className="flex items-center gap-8 px-2 py-4 sm:gap-4"> <div className={`flex items-center ${preferenceStore.flags.showFifthButton && preferenceStore.flags.showNavbarTitles == "always" ? "gap-4" : "gap-8"} px-2 py-4 sm:gap-4`}>
{!userStore.isAuth ? {!userStore.isAuth ?
<Link <Link
href={ href={
pathname != "/login" ? `/login?redirect=${pathname}` : "#" pathname != "/login" ? `/login?redirect=${pathname}` : "#"
} }
className={`flex items-center flex-col lg:flex-row gap-1 ${pathname == "/login" ? "font-bold" : "font-medium"}`} className={`flex items-center flex-col lg:flex-row gap-1 ${pathname == "/login" ? "font-bold" : "font-medium"} transition-all`}
> >
<span className="w-6 h-6 iconify material-symbols--login"></span> <span className="w-6 h-6 iconify material-symbols--login"></span>
<span <span
@ -130,7 +130,7 @@ export const Navbar = () => {
: <> : <>
<Link <Link
href={`/profile/${userStore.user.id}`} href={`/profile/${userStore.user.id}`}
className={`hidden lg:flex flex-col lg:flex-row items-center gap-1 ${pathname == `/profile/${userStore.user.id}` ? "font-bold" : "font-medium"}`} className={`hidden lg:flex flex-col lg:flex-row items-center gap-1 ${pathname == `/profile/${userStore.user.id}` ? "font-bold" : "font-medium"} transition-all`}
> >
<Image <Image
src={userStore.user.avatar} src={userStore.user.avatar}
@ -139,13 +139,30 @@ export const Navbar = () => {
width={24} width={24}
height={24} height={24}
/> />
<span className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" || preferenceStore.flags.showNavbarTitles == "links" || (preferenceStore.flags.showNavbarTitles == "selected" && pathname == `/profile/${userStore.user.id}`) ? "block" : "hidden"}`}> <span
className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" || preferenceStore.flags.showNavbarTitles == "links" || (preferenceStore.flags.showNavbarTitles == "selected" && pathname == `/profile/${userStore.user.id}`) ? "block" : "hidden"}`}
>
{userStore.user.login} {userStore.user.login}
</span> </span>
</Link> </Link>
{preferenceStore.flags.showFifthButton ?
<Link
href={menuItems[preferenceStore.flags.showFifthButton].href}
className={`flex flex-col sm:hidden items-center gap-1 ${pathname == menuItems[preferenceStore.flags.showFifthButton].href ? "font-bold" : "font-medium"} transition-all`}
>
<span
className={`w-6 h-6 iconify ${pathname == menuItems[preferenceStore.flags.showFifthButton].href ? menuItems[preferenceStore.flags.showFifthButton].icon.active : menuItems[preferenceStore.flags.showFifthButton].icon.default}`}
></span>
<span
className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" || preferenceStore.flags.showNavbarTitles == "links" || (preferenceStore.flags.showNavbarTitles == "selected" && pathname == menuItems[preferenceStore.flags.showFifthButton].href) ? "block" : "hidden"}`}
>
{menuItems[preferenceStore.flags.showFifthButton].title}
</span>
</Link>
: ""}
<Link <Link
href={`/menu`} href={`/menu`}
className={`flex flex-col lg:hidden items-center gap-1 ${pathname == `/menu` || pathname == `/profile/${userStore.user.id}` ? "font-bold" : "font-medium"}`} className={`flex flex-col lg:hidden items-center gap-1 ${pathname == `/menu` || pathname == `/profile/${userStore.user.id}` ? "font-bold" : "font-medium"} transition-all`}
> >
<Image <Image
src={userStore.user.avatar} src={userStore.user.avatar}
@ -154,7 +171,9 @@ export const Navbar = () => {
width={24} width={24}
height={24} height={24}
/> />
<span className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" || preferenceStore.flags.showNavbarTitles == "links" || (preferenceStore.flags.showNavbarTitles == "selected" && (pathname == `/menu` || pathname == `/profile/${userStore.user.id}`)) ? "block" : "hidden"}`}> <span
className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" || preferenceStore.flags.showNavbarTitles == "links" || (preferenceStore.flags.showNavbarTitles == "selected" && (pathname == `/menu` || pathname == `/profile/${userStore.user.id}`)) ? "block" : "hidden"}`}
>
{userStore.user.login} {userStore.user.login}
</span> </span>
</Link> </Link>
@ -165,7 +184,11 @@ export const Navbar = () => {
onClick={() => setIsSettingModalOpen(true)} onClick={() => setIsSettingModalOpen(true)}
> >
<span className="w-6 h-6 iconify material-symbols--settings-outline-rounded"></span> <span className="w-6 h-6 iconify material-symbols--settings-outline-rounded"></span>
<span className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" ? "block" : "hidden"}`}>Настройки</span> <span
className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" ? "block" : "hidden"}`}
>
Настройки
</span>
</button> </button>
{userStore.isAuth && ( {userStore.isAuth && (
<button <button
@ -173,7 +196,9 @@ export const Navbar = () => {
onClick={() => userStore.logout()} onClick={() => userStore.logout()}
> >
<span className="w-6 h-6 iconify material-symbols--logout"></span> <span className="w-6 h-6 iconify material-symbols--logout"></span>
<span className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" ? "lg:hidden xl:block" : "hidden"}`}> <span
className={`text-xs sm:text-base ${preferenceStore.flags.showNavbarTitles == "always" ? "lg:hidden xl:block" : "hidden"}`}
>
Выйти Выйти
</span> </span>
</button> </button>

View file

@ -342,6 +342,8 @@ export const ReleasePlayerCustom = (props: {
setIsLoading(false); setIsLoading(false);
}; };
if (episode.selected) { if (episode.selected) {
setIsLoading(true);
setPlayerError(null);
__getInfo(); __getInfo();
} }
}, [episode.selected]); }, [episode.selected]);

View file

@ -83,7 +83,7 @@ export const PosterWithStuff = (props: {
return ( return (
<span <span
key={`release_${props.id}_genre_${genre}_${index}`} key={`release_${props.id}_genre_${genre}_${index}`}
className="font-light text-white md:text-sm lg:text-base xl:text-lg" className="font-light leading-none text-white md:text-sm lg:text-base xl:text-lg"
> >
{index > 0 && ", "} {index > 0 && ", "}
{genre} {genre}
@ -91,18 +91,18 @@ export const PosterWithStuff = (props: {
); );
})} })}
{props.title_ru && ( {props.title_ru && (
<p className="text-xl font-bold text-white md:text-2xl"> <p className="py-1 text-xl font-bold leading-none text-white md:text-2xl md:py-0">
{props.title_ru} {props.title_ru}
</p> </p>
)} )}
{props.title_original && ( {props.title_original && (
<p className="text-sm text-gray-300 md:text-base"> <p className="text-sm leading-none text-gray-300 md:text-base">
{props.title_original} {props.title_original}
</p> </p>
)} )}
</div> </div>
{settings.showDescription && props.description && ( {settings.showDescription && props.description && (
<p className="mt-2 text-sm font-light text-white lg:text-base xl:text-lg line-clamp-4"> <p className="mt-2 text-sm font-light leading-none text-white lg:text-base xl:text-lg line-clamp-4">
{props.description} {props.description}
</p> </p>
)} )}

View file

@ -1,5 +1,7 @@
"use client"; "use client";
import { CURRENT_APP_VERSION } from "#/api/config";
import { useUserStore } from "#/store/auth";
import { usePreferencesStore } from "#/store/preferences"; import { usePreferencesStore } from "#/store/preferences";
import { import {
Modal, Modal,
@ -32,10 +34,17 @@ const NavbarTitles = {
links: "Только ссылки", links: "Только ссылки",
selected: "Только выбранные", selected: "Только выбранные",
never: "Никогда", never: "Никогда",
} };
const FifthButton = {
3: "Избранное",
4: "Коллекции",
5: "История",
};
export const SettingsModal = (props: { isOpen: boolean; setIsOpen: any }) => { export const SettingsModal = (props: { isOpen: boolean; setIsOpen: any }) => {
const preferenceStore = usePreferencesStore(); const preferenceStore = usePreferencesStore();
const userStore = useUserStore();
const { computedMode, setMode } = useThemeMode(); const { computedMode, setMode } = useThemeMode();
@ -168,26 +177,66 @@ export const SettingsModal = (props: { isOpen: boolean; setIsOpen: any }) => {
</p> </p>
<Dropdown <Dropdown
color="blue" color="blue"
label={ label={NavbarTitles[preferenceStore.flags.showNavbarTitles]}
NavbarTitles[preferenceStore.flags.showNavbarTitles]
}
> >
{Object.keys(NavbarTitles).map((key: "always" | "links" | "selected" | "never") => { {Object.keys(NavbarTitles).map(
return ( (key: "always" | "links" | "selected" | "never") => {
<Dropdown.Item return (
key={`navbar-titles-${key}`} <Dropdown.Item
onClick={() => className={`${key == "links" ? "hidden lg:flex" : ""}`}
preferenceStore.setFlags({ key={`navbar-titles-${key}`}
showNavbarTitles: key, onClick={() =>
}) preferenceStore.setFlags({
} showNavbarTitles: key,
> })
{NavbarTitles[key]} }
</Dropdown.Item> >
); {NavbarTitles[key]}
})} </Dropdown.Item>
);
}
)}
</Dropdown> </Dropdown>
</div> </div>
{userStore.isAuth ?
<div className="flex items-center justify-between sm:hidden">
<p className=" dark:text-white max-w-96">
Пятый пункт в навигации
</p>
<Dropdown
color="blue"
label={
preferenceStore.flags.showFifthButton ?
FifthButton[preferenceStore.flags.showFifthButton]
: "Нет"
}
>
<Dropdown.Item
onClick={() =>
preferenceStore.setFlags({
showFifthButton: null,
})
}
>
Не показывать
</Dropdown.Item>
{Object.keys(FifthButton).map((key) => {
return (
<Dropdown.Item
key={`navbar-fifthbutton-${key}`}
onClick={() =>
preferenceStore.setFlags({
showFifthButton: Number(key) as 3 | 4 | 5,
})
}
>
{FifthButton[key]}
</Dropdown.Item>
);
})}
</Dropdown>
</div>
: ""}
<HR className="my-4 dark:bg-slate-400" /> <HR className="my-4 dark:bg-slate-400" />
<div className="flex items-center gap-2"> <div className="flex items-center gap-2">
<span className="w-6 h-6 iconify material-symbols--settings-outline"></span> <span className="w-6 h-6 iconify material-symbols--settings-outline"></span>
@ -278,26 +327,15 @@ export const SettingsModal = (props: { isOpen: boolean; setIsOpen: any }) => {
<HR className="my-4 dark:bg-slate-400" /> <HR className="my-4 dark:bg-slate-400" />
<div> <div>
<Link <Link
href={"https://t.me/anix_web"} href={"/about"}
className="flex items-center gap-2 p-2 text-left rounded-md hover:bg-gray-100 dark:hover:bg-gray-900" className="flex items-center gap-2 p-2 text-left rounded-md hover:bg-gray-100 dark:hover:bg-gray-900"
onClick={() => props.setIsOpen(false)}
> >
<span className="w-8 h-8 iconify fa6-brands--telegram"></span> <span className="w-8 h-8 iconify material-symbols--info"></span>
<div> <div>
<p>Телеграм канал</p> <p>О приложении</p>
<p className="text-sm text-gray-400 dark:text-gray-200"> <p className="text-sm text-gray-400 dark:text-gray-200">
@anix_web v{CURRENT_APP_VERSION}
</p>
</div>
</Link>
<Link
href={"https://wah.su/radiquum"}
className="flex items-center gap-2 p-2 text-left rounded-md hover:bg-gray-100 dark:hover:bg-gray-900"
>
<span className="w-8 h-8 iconify mdi--code"></span>
<div>
<p>Разработчик</p>
<p className="text-sm text-gray-400 dark:text-gray-200">
Radiquum
</p> </p>
</div> </div>
</Link> </Link>

136
app/pages/About.tsx Normal file
View file

@ -0,0 +1,136 @@
"use server";
import { Card } from "flowbite-react";
import Image from "next/image";
import * as fs from "node:fs";
import * as path from "node:path";
import { CURRENT_APP_VERSION } from "#/api/config";
import Styles from "../components/ChangelogModal/ChangelogModal.module.css";
import Markdown from "markdown-to-jsx";
import {
Accordion,
AccordionContent,
AccordionPanel,
AccordionTitle,
} from "flowbite-react";
import { version } from "node:os";
import Link from "next/link";
export const AboutPage = () => {
const directoryPath = path.join(process.cwd(), "public/changelog");
const files = fs.readdirSync(directoryPath);
const current = {
version: CURRENT_APP_VERSION,
changelog: `#${CURRENT_APP_VERSION}\r\nНет списка изменений`,
};
const previous = [];
if (files.includes(`${CURRENT_APP_VERSION}.md`)) {
const changelog = fs.readFileSync(
path.join(directoryPath, `${CURRENT_APP_VERSION}.md`),
"utf8"
);
current.changelog = changelog;
}
files.forEach((file) => {
if (file != `${CURRENT_APP_VERSION}.md`) {
const changelog = fs.readFileSync(path.join(directoryPath, file), "utf8");
previous.push({
version: file.replace(".md", ""),
changelog: changelog,
});
}
});
return (
<div className="grid grid-cols-1 gap-2 md:grid-cols-2 lg:grid-cols-3">
<Card className="md:col-span-2 lg:col-span-3">
<div className="flex flex-col items-center gap-4 md:flex-row">
<Image
src="/images/icons/icon-512x512.png"
className="flex-shrink-0 w-32 h-32 rounded-full"
alt="about image"
width={128}
height={128}
/>
<div>
<h1 className="text-xl font-bold">
AniX - Неофициальный веб клиент для Anixart
</h1>
<p className="max-w-[900px]">
AniX - это неофициальный веб-клиент для Android-приложения
Anixart. Он позволяет вам получать доступ к своей учетной записи
Anixart и управлять ею из веб-браузера. Так-же можно
синхронизировать и управлять списками и избранным. И самое главное
смотреть все доступные аниме из базы Anixart.
</p>
</div>
</div>
</Card>
<Link href={"https://wah.su/radiquum"} target="_blank">
<Card>
<div className="flex items-center gap-4">
<Image
src="https://radiquum.wah.su/static/avatar_512.jpg"
className="flex-shrink-0 w-16 h-16 rounded-full"
alt="developer image"
width={128}
height={128}
/>
<div>
<h1 className="text-xl font-bold">Radiquum</h1>
<p className="text-sm text-gray-500 dark:text-gray-200">
Разработчик
</p>
</div>
</div>
</Card>
</Link>
<Link href={"https://t.me/anix_web"} target="_blank">
<Card>
<div className="flex items-center gap-4">
<span className="w-16 h-16 iconify fa6-brands--telegram text-[#001725] dark:text-[#faf8f9]"></span>
<div>
<h1 className="text-xl font-bold">Телеграм канал</h1>
<p className="text-sm text-gray-500 dark:text-gray-200">
@anix_web
</p>
</div>
</div>
</Card>
</Link>
<Link href={"https://github.com/Radiquum/AniX"} target="_blank">
<Card>
<div className="flex items-center gap-4">
<span className="flex-shrink-0 w-16 h-16 iconify fa6-brands--github text-[#001725] dark:text-[#faf8f9]"></span>
<div>
<h1 className="text-xl font-bold">Код на GitHub</h1>
<p className="text-sm text-gray-500 dark:text-gray-200">
github.com/Radiquum/AniX
</p>
</div>
</div>
</Card>
</Link>
<Card className="md:col-span-2 lg:col-span-3">
<h1 className="text-2xl font-bold">Список изменений</h1>
<Markdown className={Styles.markdown}>{current.changelog}</Markdown>
<Accordion collapseAll={true}>
{previous.reverse().map((changelog) => (
<AccordionPanel key={changelog.version}>
<AccordionTitle>v{changelog.version}</AccordionTitle>
<AccordionContent>
<Markdown className={Styles.markdown}>
{changelog.changelog}
</Markdown>
</AccordionContent>
</AccordionPanel>
))}
</Accordion>
</Card>
</div>
);
};

View file

@ -6,9 +6,11 @@ import { useRouter } from "next/navigation";
import { SettingsModal } from "#/components/SettingsModal/SettingsModal"; import { SettingsModal } from "#/components/SettingsModal/SettingsModal";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import Image from "next/image"; import Image from "next/image";
import { usePreferencesStore } from "#/store/preferences";
export const MenuPage = () => { export const MenuPage = () => {
const userStore = useUserStore(); const userStore = useUserStore();
const preferenceStore = usePreferencesStore();
const router = useRouter(); const router = useRouter();
const [isSettingModalOpen, setIsSettingModalOpen] = useState(false); const [isSettingModalOpen, setIsSettingModalOpen] = useState(false);
@ -22,7 +24,7 @@ export const MenuPage = () => {
return ( return (
<> <>
{userStore.user && ( {userStore.user && (
<div className="flex flex-col gap-2"> <div className="fixed flex flex-col justify-end gap-2 left-4 right-4 bottom-24 sm:static">
<div className="flex flex-wrap items-center gap-2"> <div className="flex flex-wrap items-center gap-2">
<Link <Link
href={`/profile/${userStore.user.id}`} href={`/profile/${userStore.user.id}`}
@ -81,62 +83,42 @@ export const MenuPage = () => {
</button> </button>
</div> </div>
</div> </div>
<Link href="/favorites" className="flex-1 sm:hidden"> {preferenceStore.flags.showFifthButton != 3 ?
<Card> <Link href="/favorites" className="flex-1 sm:hidden">
<div className="flex items-center gap-2"> <Card>
<span <div className="flex items-center gap-2">
className={`iconify material-symbols--favorite-outline w-6 h-6`} <span
></span> className={`iconify material-symbols--favorite-outline w-6 h-6`}
<p>Избранное</p> ></span>
</div> <p>Избранное</p>
</Card>
</Link>
<Link href="/collections" className="flex-1 sm:hidden">
<Card>
<div className="flex items-center gap-2">
<span
className={`iconify material-symbols--collections-bookmark-outline w-6 h-6`}
></span>
<p>Коллекции</p>
</div>
</Card>
</Link>
<Link href="/history" className="flex-1 sm:hidden">
<Card>
<div className="flex items-center gap-2">
<span
className={`iconify material-symbols--history w-6 h-6`}
></span>
<p>История</p>
</div>
</Card>
</Link>
<Link href={"https://t.me/anix_web"} className="flex-1">
<Card>
<div className="flex items-center gap-2">
<span className="w-8 h-8 iconify fa6-brands--telegram"></span>
<div>
<p>Телеграм канал</p>
<p className="text-sm text-gray-400 dark:text-gray-200">
@anix_web
</p>
</div> </div>
</div> </Card>
</Card> </Link>
</Link> : ""}
<Link href={"https://wah.su/radiquum"} className="flex-1"> {preferenceStore.flags.showFifthButton != 4 ?
<Card> <Link href="/collections" className="flex-1 sm:hidden">
<div className="flex items-center gap-2"> <Card>
<span className="w-8 h-8 iconify mdi--code"></span> <div className="flex items-center gap-2">
<div> <span
<p>Разработчик</p> className={`iconify material-symbols--collections-bookmark-outline w-6 h-6`}
<p className="text-sm text-gray-400 dark:text-gray-200"> ></span>
Radiquum <p>Коллекции</p>
</p>
</div> </div>
</div> </Card>
</Card> </Link>
</Link> : ""}
{preferenceStore.flags.showFifthButton != 5 ?
<Link href="/history" className="flex-1 sm:hidden">
<Card>
<div className="flex items-center gap-2">
<span
className={`iconify material-symbols--history w-6 h-6`}
></span>
<p>История</p>
</div>
</Card>
</Link>
: ""}
<SettingsModal <SettingsModal
isOpen={isSettingModalOpen} isOpen={isSettingModalOpen}
setIsOpen={setIsSettingModalOpen} setIsOpen={setIsSettingModalOpen}

View file

@ -11,6 +11,7 @@ interface preferencesState {
showChangelog?: boolean; showChangelog?: boolean;
enableAnalytics?: boolean; enableAnalytics?: boolean;
showNavbarTitles?: "always" | "links" | "selected" | "never"; showNavbarTitles?: "always" | "links" | "selected" | "never";
showFifthButton?: null | 3 | 4 | 5;
}; };
params: { params: {
isFirstLaunch?: boolean; isFirstLaunch?: boolean;
@ -22,7 +23,7 @@ interface preferencesState {
}; };
experimental?: { experimental?: {
newPlayer: boolean; newPlayer: boolean;
} };
// color: { // color: {
// primary: string; // primary: string;
// secondary: string; // secondary: string;
@ -44,6 +45,7 @@ export const usePreferencesStore = create<preferencesState>()(
showChangelog: true, showChangelog: true,
enableAnalytics: true, enableAnalytics: true,
showNavbarTitles: "always", showNavbarTitles: "always",
showFifthButton: null,
}, },
params: { params: {
isFirstLaunch: true, isFirstLaunch: true,
@ -54,8 +56,8 @@ export const usePreferencesStore = create<preferencesState>()(
bookmarksCategory: "watching", bookmarksCategory: "watching",
}, },
experimental: { experimental: {
newPlayer: false newPlayer: false,
} },
}, },
setHasHydrated: (state) => { setHasHydrated: (state) => {
set({ set({
@ -74,9 +76,12 @@ export const usePreferencesStore = create<preferencesState>()(
onRehydrateStorage: (state) => { onRehydrateStorage: (state) => {
return () => state.setHasHydrated(true); return () => state.setHasHydrated(true);
}, },
merge: (persistedState , currentState) => { merge: (persistedState, currentState) => {
return deepmerge(currentState as preferencesState, persistedState as preferencesState); return deepmerge(
} currentState as preferencesState,
persistedState as preferencesState
);
},
} }
) )
); );

View file

@ -6,10 +6,10 @@ AniX - это неофициальный веб-клиент для Android-пр
## Список изменений ## Список изменений
- [3.5.0](/public/changelog/3.5.0.md)
- [3.4.0](/public/changelog/3.4.0.md) - [3.4.0](/public/changelog/3.4.0.md)
- [3.3.0](/public/changelog/3.3.0.md) - [3.3.0](/public/changelog/3.3.0.md)
- [3.2.3](/public/changelog/3.2.3.md) - [3.2.3](/public/changelog/3.2.3.md)
- [3.2.2](/public/changelog/3.2.2.md)
[другие версии](/public/changelog) [другие версии](/public/changelog)

17
public/changelog/3.5.0.md Normal file
View file

@ -0,0 +1,17 @@
# 3.5.0
## Добавлено
- Добавлена страница о приложении
- Добавлена возможность добавить пятую кнопку в меню навигации на мобильных устройствах
## Изменено
- Стиль карточек для релизов был изменён на вертикальный
- На мобильных устройствах постер на странице релиза теперь по середине
- Позиция лицензированных сервисов теперь под постером
- На мобильных устройствах позиция меню была смещена вниз
## Исправлено
- Ошибка своего плеера не сбрасывалась, если удалось получить ссылку при переключении серии