From 25e31a7799f2adabc703056461c3c5062e6a58d7 Mon Sep 17 00:00:00 2001 From: Kentai Radiquum Date: Mon, 29 Jul 2024 21:39:37 +0500 Subject: [PATCH] Merge remote-tracking branch 'origin/feat_player' --- TODO.md | 18 +- app/{App.jsx => App.tsx} | 0 app/api/[...endpoint]/route.ts | 38 ++ app/api/bookmarks/route.js | 43 --- app/api/common.js | 8 - app/api/config.js | 32 -- app/api/config.ts | 32 ++ app/api/favorites/route.js | 26 -- app/api/history/route.js | 19 - app/api/home/route.js | 50 --- app/api/profile/[id]/route.js | 21 - app/api/profile/login/route.js | 15 - app/api/profile/login/route.ts | 14 + app/api/search/route.js | 21 - app/api/search/route.ts | 39 ++ app/api/utils.js | 126 ------ app/api/utils.ts | 249 ++++++++++++ app/bookmarks/[slug]/{page.js => page.tsx} | 2 +- app/bookmarks/{page.js => page.tsx} | 2 +- app/components/Chip/Chip.jsx | 12 - app/components/Chip/Chip.tsx | 16 + .../Navbar/{Navbar.jsx => Navbar.tsx} | 7 +- ...{RelatedSection.jsx => RelatedSection.tsx} | 4 +- ...eleaseCourusel.jsx => ReleaseCourusel.tsx} | 30 +- .../ReleaseInfo/ReleaseInfo.SearchLink.tsx | 21 + .../{ReleaseLink.jsx => ReleaseLink.16_9.tsx} | 25 +- .../ReleaseLink/ReleaseLink.Poster.tsx | 82 ++++ app/components/ReleaseLink/ReleaseLink.tsx | 13 + .../ReleasePlayer/ReleasePlayer.tsx | 174 +++++++++ ...{ReleaseSection.jsx => ReleaseSection.tsx} | 2 +- .../Spinner/{Spinner.jsx => Spinner.tsx} | 0 app/favorites/{page.js => page.tsx} | 2 +- app/history/{page.js => page.tsx} | 2 +- app/home/[slug]/{page.js => page.tsx} | 2 +- ...ScrollPosition.js => useScrollPosition.ts} | 0 app/{layout.js => layout.tsx} | 2 +- app/login/{page.js => page.tsx} | 2 +- app/{page.js => page.tsx} | 0 app/pages/{Bookmarks.jsx => Bookmarks.tsx} | 36 +- ...arksCategory.jsx => BookmarksCategory.tsx} | 23 +- app/pages/{Favorites.jsx => Favorites.tsx} | 20 +- app/pages/{History.jsx => History.tsx} | 20 +- app/pages/Index.jsx | 64 ---- app/pages/Index.tsx | 94 +++++ app/pages/IndexCategory.jsx | 91 ----- app/pages/IndexCategory.tsx | 77 ++++ app/pages/{Login.jsx => Login.tsx} | 16 +- app/pages/{Profile.jsx => Profile.tsx} | 47 +-- app/pages/Release.tsx | 359 ++++++++++++++++++ app/pages/{Search.jsx => Search.tsx} | 35 +- app/pages/{common.js => common.ts} | 6 + app/profile/[id]/page.js | 17 - app/profile/[id]/page.tsx | 16 + app/profile/page.js | 14 - app/release/[id]/page.tsx | 16 + app/search/{page.js => page.tsx} | 2 +- app/store/{auth.js => auth.ts} | 15 +- app/store/{preferences.js => preferences.ts} | 0 jsconfig.json | 7 - package-lock.json | 39 ++ package.json | 2 + tsconfig.json | 42 ++ 62 files changed, 1508 insertions(+), 701 deletions(-) rename app/{App.jsx => App.tsx} (100%) create mode 100644 app/api/[...endpoint]/route.ts delete mode 100644 app/api/bookmarks/route.js delete mode 100644 app/api/common.js delete mode 100644 app/api/config.js create mode 100644 app/api/config.ts delete mode 100644 app/api/favorites/route.js delete mode 100644 app/api/history/route.js delete mode 100644 app/api/home/route.js delete mode 100644 app/api/profile/[id]/route.js delete mode 100644 app/api/profile/login/route.js create mode 100644 app/api/profile/login/route.ts delete mode 100644 app/api/search/route.js create mode 100644 app/api/search/route.ts delete mode 100644 app/api/utils.js create mode 100644 app/api/utils.ts rename app/bookmarks/[slug]/{page.js => page.tsx} (88%) rename app/bookmarks/{page.js => page.tsx} (71%) delete mode 100644 app/components/Chip/Chip.jsx create mode 100644 app/components/Chip/Chip.tsx rename app/components/Navbar/{Navbar.jsx => Navbar.tsx} (96%) rename app/components/RelatedSection/{RelatedSection.jsx => RelatedSection.tsx} (93%) rename app/components/ReleaseCourusel/{ReleaseCourusel.jsx => ReleaseCourusel.tsx} (75%) create mode 100644 app/components/ReleaseInfo/ReleaseInfo.SearchLink.tsx rename app/components/ReleaseLink/{ReleaseLink.jsx => ReleaseLink.16_9.tsx} (83%) create mode 100644 app/components/ReleaseLink/ReleaseLink.Poster.tsx create mode 100644 app/components/ReleaseLink/ReleaseLink.tsx create mode 100644 app/components/ReleasePlayer/ReleasePlayer.tsx rename app/components/ReleaseSection/{ReleaseSection.jsx => ReleaseSection.tsx} (94%) rename app/components/Spinner/{Spinner.jsx => Spinner.tsx} (100%) rename app/favorites/{page.js => page.tsx} (69%) rename app/history/{page.js => page.tsx} (70%) rename app/home/[slug]/{page.js => page.tsx} (90%) rename app/hooks/{useScrollPosition.js => useScrollPosition.ts} (100%) rename app/{layout.js => layout.tsx} (89%) rename app/login/{page.js => page.tsx} (56%) rename app/{page.js => page.tsx} (100%) rename app/pages/{Bookmarks.jsx => Bookmarks.tsx} (70%) rename app/pages/{BookmarksCategory.jsx => BookmarksCategory.tsx} (82%) rename app/pages/{Favorites.jsx => Favorites.tsx} (85%) rename app/pages/{History.jsx => History.tsx} (81%) delete mode 100644 app/pages/Index.jsx create mode 100644 app/pages/Index.tsx delete mode 100644 app/pages/IndexCategory.jsx create mode 100644 app/pages/IndexCategory.tsx rename app/pages/{Login.jsx => Login.tsx} (93%) rename app/pages/{Profile.jsx => Profile.tsx} (90%) create mode 100644 app/pages/Release.tsx rename app/pages/{Search.jsx => Search.tsx} (85%) rename app/pages/{common.js => common.ts} (91%) delete mode 100644 app/profile/[id]/page.js create mode 100644 app/profile/[id]/page.tsx delete mode 100644 app/profile/page.js create mode 100644 app/release/[id]/page.tsx rename app/search/{page.js => page.tsx} (87%) rename app/store/{auth.js => auth.ts} (63%) rename app/store/{preferences.js => preferences.ts} (100%) delete mode 100644 jsconfig.json create mode 100644 tsconfig.json diff --git a/TODO.md b/TODO.md index 2f0eae9..67eabe0 100644 --- a/TODO.md +++ b/TODO.md @@ -12,7 +12,7 @@ - [X] Просмотр последних аниме в списках - [X] Просмотр всех аниме в списках - [X] Сортировка аниме в списках -- [ ] Добавление \ Удаление аниме из списков +- [X] Добавление \ Удаление аниме из списков ### Поиск @@ -21,12 +21,14 @@ - [ ] Просмотр страницы франшизы - [ ] Фильтры поиска - [ ] История поиска +- [X] Поиск по тегам со страницы тайтла ### Закладки - [X] Просмотр всех аниме в списке - [X] Сортировка аниме в списке -- [ ] Добавление \ Удаление аниме из списка +- [X] Добавление \ Удаление аниме из списка +- [ ] Поиск в списке ### Профиль @@ -48,12 +50,16 @@ ### Страница аниме тайтла -- [ ] Описание тайтла +- [X] Описание тайтла +- [ ] Скриншоты тайтла +- [ ] Видео тайтла - [ ] Просмотр тайтла -- [ ] Просмотр комментариев -- [ ] Комментирование +- [ ] Просмотр комментариев и комментирование - [ ] Сохранение эпизода в историю просмотров -- [ ] Добавление \ Удаление аниме в\из списков закладок и избранных +- [X] Добавление \ Удаление аниме в\из списков закладок и избранных +- [X] Связанные релизы +- [ ] Просмотр страницы всех вязанных релизов +- [ ] Оценка тайтла ## Баги diff --git a/app/App.jsx b/app/App.tsx similarity index 100% rename from app/App.jsx rename to app/App.tsx diff --git a/app/api/[...endpoint]/route.ts b/app/api/[...endpoint]/route.ts new file mode 100644 index 0000000..a67d193 --- /dev/null +++ b/app/api/[...endpoint]/route.ts @@ -0,0 +1,38 @@ +import { NextResponse, NextRequest } from "next/server"; +import { fetchDataViaGet, fetchDataViaPost } from "../utils"; +import { API_URL } from "../config"; + +export async function GET( + req: NextRequest, + { params }: { params: { endpoint: Array } } +) { + const { endpoint } = params; + let API_V2: boolean | string = + req.nextUrl.searchParams.get("API_V2") || false; + if (API_V2 === "true") { + req.nextUrl.searchParams.delete("API_V2"); + } + const query = req.nextUrl.searchParams.toString(); + const url = `${API_URL}/${endpoint.join("/")}${query ? `?${query}` : ""}`; + + const response = await fetchDataViaGet(url, API_V2); + return NextResponse.json(response); +} + +export async function POST( + req: NextRequest, + { params }: { params: { endpoint: Array } } +) { + const { endpoint } = params; + let API_V2: boolean | string = + req.nextUrl.searchParams.get("API_V2") || false; + if (API_V2 === "true") { + req.nextUrl.searchParams.delete("API_V2"); + } + const query = req.nextUrl.searchParams.toString(); + const url = `${API_URL}/${endpoint.join("/")}${query ? `?${query}` : ""}`; + const body = JSON.stringify( await req.json()); + + const response = await fetchDataViaPost(url, body, API_V2); + return NextResponse.json(response); +} diff --git a/app/api/bookmarks/route.js b/app/api/bookmarks/route.js deleted file mode 100644 index d53f467..0000000 --- a/app/api/bookmarks/route.js +++ /dev/null @@ -1,43 +0,0 @@ -import { NextResponse } from "next/server"; -import { fetchDataViaGet } from "../utils"; -import { ENDPOINTS } from "../config"; -import { sort } from "../common"; - -const list = { - watching: 1, - planned: 2, - watched: 3, - delayed: 4, - abandoned: 5, -}; - -export async function GET(request) { - const page = parseInt(request.nextUrl.searchParams.get(["page"])) || 0; - const listName = request.nextUrl.searchParams.get(["list"]) || null; - const token = request.nextUrl.searchParams.get(["token"]) || null; - const sortName = - request.nextUrl.searchParams.get(["sort"]) || "adding_descending"; - - if (!token || token == "null") { - return NextResponse.json({ message: "No token provided" }, { status: 403 }); - } - - if (!listName || listName == "null") { - return NextResponse.json({ message: "No list provided" }, { status: 400 }); - } - - if (!list[listName]) { - return NextResponse.json({ message: "Unknown list" }, { status: 400 }); - } - - if (!sort[sortName]) { - return NextResponse.json({ message: "Unknown sort" }, { status: 400 }); - } - - let url = new URL(`${ENDPOINTS.user.bookmark}/${list[listName]}/${page}`); - url.searchParams.set("token", token); - url.searchParams.set("sort", sort[sortName]); - - const response = await fetchDataViaGet(url.toString()); - return NextResponse.json(response); -} diff --git a/app/api/common.js b/app/api/common.js deleted file mode 100644 index b48983a..0000000 --- a/app/api/common.js +++ /dev/null @@ -1,8 +0,0 @@ -export const sort = { - adding_descending: 1, - adding_ascending: 2, - year_descending: 3, - year_ascending: 4, - alphabet_descending: 5, - alphabet_ascending: 6, -}; diff --git a/app/api/config.js b/app/api/config.js deleted file mode 100644 index 6e65034..0000000 --- a/app/api/config.js +++ /dev/null @@ -1,32 +0,0 @@ -export const API_URL = "https://api.anixart.tv"; -export const USER_AGENT = - "AnixartApp/8.2.1-23121216 (Android 11; SDK 30; arm64-v8a;)"; - -export const ENDPOINTS = { - release: { - info: `${API_URL}/release`, - episode: `${API_URL}/episode`, - }, - user: { - profile: `${API_URL}/profile`, - bookmark: `${API_URL}/profile/list/all`, - history: `${API_URL}/history`, - favorite: `${API_URL}/favorite/all`, - }, - filter: `${API_URL}/filter`, - auth: `${API_URL}/auth/signIn`, - // user: { - // history: `${API_URL}/history`, - // watching: `${API_URL}/profile/list/all/1`, - // planned: `${API_URL}/profile/list/all/2`, - // watched: `${API_URL}/profile/list/all/3`, - // delayed: `${API_URL}/profile/list/all/4`, - // abandoned: `${API_URL}/profile/list/all/5`, - // favorite: `${API_URL}/favorite`, - // }, - search: `${API_URL}/search/releases`, - statistic: { - addHistory: `${API_URL}/history/add`, - markWatched: `${API_URL}/episode/watch`, - }, -}; diff --git a/app/api/config.ts b/app/api/config.ts new file mode 100644 index 0000000..0b91a5d --- /dev/null +++ b/app/api/config.ts @@ -0,0 +1,32 @@ +export const API_URL = "https://api.anixart.tv"; +export const API_PREFIX = "/api"; +export const USER_AGENT = + "AnixartApp/8.2.1-23121216 (Android 11; SDK 30; arm64-v8a;)"; + +export const ENDPOINTS = { + release: { + info: `${API_PREFIX}/release`, + episode: `${API_PREFIX}/episode`, + }, + user: { + profile: `${API_PREFIX}/profile`, + bookmark: `${API_PREFIX}/profile/list`, + history: `${API_PREFIX}/history`, + favorite: `${API_PREFIX}/favorite`, + }, + filter: `${API_PREFIX}/filter`, + // user: { + // history: `${API_PREFIX}/history`, + // watching: `${API_PREFIX}/profile/list/all/1`, + // planned: `${API_PREFIX}/profile/list/all/2`, + // watched: `${API_PREFIX}/profile/list/all/3`, + // delayed: `${API_PREFIX}/profile/list/all/4`, + // abandoned: `${API_PREFIX}/profile/list/all/5`, + // favorite: `${API_PREFIX}/favorite`, + // }, + search: `${API_URL}/search`, + statistic: { + addHistory: `${API_PREFIX}/history/add`, + markWatched: `${API_PREFIX}/episode/watch`, + }, +}; diff --git a/app/api/favorites/route.js b/app/api/favorites/route.js deleted file mode 100644 index b241b82..0000000 --- a/app/api/favorites/route.js +++ /dev/null @@ -1,26 +0,0 @@ -import { NextResponse } from "next/server"; -import { fetchDataViaGet } from "../utils"; -import { ENDPOINTS } from "../config"; -import { sort } from "../common"; - - -export async function GET(request) { - const page = parseInt(request.nextUrl.searchParams.get(["page"])) || 0; - const token = request.nextUrl.searchParams.get(["token"]) || null; - const sortName = request.nextUrl.searchParams.get(["sort"]) || "adding_descending"; - - if (!token || token == "null") { - return NextResponse.json({ message: "No token provided" }, { status: 403 }); - } - - if (!sort[sortName]) { - return NextResponse.json({ message: "Unknown sort" }, { status: 400 }); - } - - let url = new URL(`${ENDPOINTS.user.favorite}/${page}`); - url.searchParams.set("token", token); - url.searchParams.set("sort", sort[sortName]); - - const response = await fetchDataViaGet(url.toString()); - return NextResponse.json(response); -} diff --git a/app/api/history/route.js b/app/api/history/route.js deleted file mode 100644 index 75e650a..0000000 --- a/app/api/history/route.js +++ /dev/null @@ -1,19 +0,0 @@ -import { NextResponse } from "next/server"; -import { fetchDataViaGet } from "../utils"; -import { ENDPOINTS } from "../config"; - -export async function GET(request) { - const page = parseInt(request.nextUrl.searchParams.get(["page"])) || 0; - const token = request.nextUrl.searchParams.get(["token"]) || null; - const sortName = request.nextUrl.searchParams.get(["sort"]) || "adding_descending"; - - if (!token || token == "null") { - return NextResponse.json({ message: "No token provided" }, { status: 403 }); - } - - let url = new URL(`${ENDPOINTS.user.history}/${page}`); - url.searchParams.set("token", token); - - const response = await fetchDataViaGet(url.toString()); - return NextResponse.json(response); -} diff --git a/app/api/home/route.js b/app/api/home/route.js deleted file mode 100644 index 1a1d96b..0000000 --- a/app/api/home/route.js +++ /dev/null @@ -1,50 +0,0 @@ -import { NextResponse } from "next/server"; -import { fetchDataViaPost } from "../utils"; -import { ENDPOINTS } from "../config"; - -export async function GET(request) { - const page = parseInt(request.nextUrl.searchParams.get(["page"])) || 0; - const status = request.nextUrl.searchParams.get(["status"]) || null; - const token = request.nextUrl.searchParams.get(["token"]) || null; - - let statusId; - if (status == "last" || !status) { - statusId = null; - } else if (status == "finished") { - statusId = 1; - } else if (status == "ongoing") { - statusId = 2; - } else if (status == "announce") { - statusId = 3; - } else { - return NextResponse.json({ message: "Bad status" }, { status: 400 }); - } - - let url = new URL(`${ENDPOINTS.filter}/${page}`); - if (token) { - url.searchParams.set("token", token); - } - - const data = { - country: null, - season: null, - sort: 0, - studio: null, - age_ratings: [], - category_id: null, - end_year: null, - episode_duration_from: null, - episode_duration_to: null, - episodes_from: null, - episodes_to: null, - genres: [], - profile_list_exclusions: [], - start_year: null, - status_id: statusId, - types: [], - is_genres_exclude_mode_enabled: false, - }; - - const response = await fetchDataViaPost(url.toString(), data); - return NextResponse.json(response); -} diff --git a/app/api/profile/[id]/route.js b/app/api/profile/[id]/route.js deleted file mode 100644 index 1799d0f..0000000 --- a/app/api/profile/[id]/route.js +++ /dev/null @@ -1,21 +0,0 @@ -import { NextResponse } from "next/server"; -import { fetchDataViaGet } from "@/app/api/utils"; -import { ENDPOINTS } from "@/app/api/config"; - -export async function GET(request, params) { - const token = request.nextUrl.searchParams.get(["token"]) || null; - let url = new URL(`${ENDPOINTS.user.profile}/${params["params"]["id"]}`); - if (token) { - url.searchParams.set("token", token); - } - - const response = await fetchDataViaGet(url.toString()); - if (!response) { - return NextResponse.json({ message: "Server Error" }, { status: 500 }); - } - if (!response.profile) { - return NextResponse.json({ message: "Profile not found" }, { status: 404 }); - } - - return NextResponse.json(response); -} diff --git a/app/api/profile/login/route.js b/app/api/profile/login/route.js deleted file mode 100644 index aabbee9..0000000 --- a/app/api/profile/login/route.js +++ /dev/null @@ -1,15 +0,0 @@ -import { NextResponse } from "next/server"; -import { authorize } from "@/app/api/utils"; -import { ENDPOINTS } from "@/app/api/config"; - -export async function POST(request) { - const response = await authorize(ENDPOINTS.auth, await request.json()); - if (!response) { - return NextResponse.json({ message: "Server Error" }, { status: 500 }); - } - if (!response.profile) { - return NextResponse.json({ message: "Profile not found" }, { status: 404 }); - } - - return NextResponse.json(response); - } \ No newline at end of file diff --git a/app/api/profile/login/route.ts b/app/api/profile/login/route.ts new file mode 100644 index 0000000..21d5a51 --- /dev/null +++ b/app/api/profile/login/route.ts @@ -0,0 +1,14 @@ +import { NextResponse, NextRequest } from "next/server"; +import { authorize } from "#/api/utils"; +import { API_URL } from "#/api/config"; + +export async function POST(request: NextRequest) { + const response = await authorize(`${API_URL}/auth/signIn`, await request.json()); + if (!response) { + return NextResponse.json({ message: "Server Error" }, { status: 500 }); + } + if (!response.profile) { + return NextResponse.json({ message: "Profile not found" }, { status: 404 }); + } + return NextResponse.json(response); +} diff --git a/app/api/search/route.js b/app/api/search/route.js deleted file mode 100644 index 6f3912f..0000000 --- a/app/api/search/route.js +++ /dev/null @@ -1,21 +0,0 @@ -import { NextResponse } from "next/server"; -import { fetchDataViaPost } from "../utils"; -import { ENDPOINTS } from "../config"; - -export async function GET(request) { - const page = parseInt(request.nextUrl.searchParams.get(["page"])) || 0; - const query = request.nextUrl.searchParams.get(["q"]) || null; - const token = request.nextUrl.searchParams.get(["token"]) || null; - let url = new URL(`${ENDPOINTS.search}/${page}`); - if (token) { - url.searchParams.set("token", token); - } - const data = { query, searchBy: 0 }; - - const response = await fetchDataViaPost(url.toString(), data, true); - if (!response) { - return NextResponse.json({ message: "Bad request" }, { status: 400 }); - } - - return NextResponse.json(response); -} diff --git a/app/api/search/route.ts b/app/api/search/route.ts new file mode 100644 index 0000000..61eeacd --- /dev/null +++ b/app/api/search/route.ts @@ -0,0 +1,39 @@ +import { NextResponse } from "next/server"; +import { NextRequest } from "next/server"; +import { fetchDataViaPost } from "../utils"; +import { ENDPOINTS } from "../config"; + +export async function GET(request: NextRequest) { + const page = parseInt(request.nextUrl.searchParams.get("page")) || 0; + const query = request.nextUrl.searchParams.get("q") || null; + const token = request.nextUrl.searchParams.get("token") || null; + + const where = request.nextUrl.searchParams.get("where") || "releases" + const searchBy = parseInt(request.nextUrl.searchParams.get("searchBy")) || 0 + const list = parseInt(request.nextUrl.searchParams.get("list")) || null + + let url: URL; + + if (where == "releases") { + url = new URL(`${ENDPOINTS.search}/releases/${page}`); + } else if (where == "list") { + if (!list) { return NextResponse.json({ message: "List ID required" }, { status: 400 })} + if (!token) { return NextResponse.json({ message: "token required" }, { status: 400 })} + url = new URL(`${ENDPOINTS.search}/profile/list/${list}/${page}`); + } + if (token) { + url.searchParams.set("token", token); + } + const data = { query, searchBy }; + + const response = await fetchDataViaPost( + url.toString(), + JSON.stringify(data), + true + ); + if (!response) { + return NextResponse.json({ message: "Bad request" }, { status: 400 }); + } + + return NextResponse.json(response); +} diff --git a/app/api/utils.js b/app/api/utils.js deleted file mode 100644 index c106e54..0000000 --- a/app/api/utils.js +++ /dev/null @@ -1,126 +0,0 @@ -import { USER_AGENT } from "./config"; -export const HEADERS = { - "User-Agent": USER_AGENT, - "Content-Type": "application/json; charset=UTF-8", -}; - -export const fetchDataViaGet = async (url, API_V2) => { - if (API_V2) { - HEADERS["API-Version"] = "v2"; - } - try { - const response = await fetch(url, { - headers: HEADERS, - }); - if (response.status !== 200) { - throw new Error("Error fetching data"); - } - const data = await response.json(); - return data; - } catch (error) { - console.log(error); - } -}; - -export const fetchDataViaPost = async (url, body, API_V2) => { - if (API_V2) { - HEADERS["API-Version"] = "v2"; - } - try { - const response = await fetch(url, { - method: "POST", - headers: HEADERS, - body: JSON.stringify(body), - }); - if (response.status !== 200) { - throw new Error("Error fetching data"); - } - const data = await response.json(); - return data; - } catch (error) { - console.log(error); - } -}; - -export const authorize = async (url, data) => { - try { - const response = await fetch( - `${url}?login=${data.login}&password=${data.password}`, - { - method: "POST", - headers: { - "User-Agent": USER_AGENT, - Sign: "9aa5c7af74e8cd70c86f7f9587bde23d", - "Content-Type": "application/x-www-form-urlencoded", - }, - } - ); - if (response.status !== 200) { - throw new Error("Error authorizing user"); - } - return await response.json(); - } catch (error) { - return error; - } -}; - -export function setJWT(user_id, jwt) { - const data = { jwt: jwt, user_id: user_id }; - localStorage.setItem("JWT", JSON.stringify(data)); -} -export function getJWT() { - const data = localStorage.getItem("JWT"); - return JSON.parse(data); -} -export function removeJWT() { - localStorage.removeItem("JWT"); -} - -export function numberDeclension(number, one, two, five) { - if (number > 10 && [11, 12, 13, 14].includes(number % 100)) return five; - let last_num = number % 10; - if (last_num == 1) return one; - if ([2, 3, 4].includes(last_num)) return two; - if ([5, 6, 7, 8, 9, 0].includes(last_num)) return five; -} - -export function unixToDate(unix) { - const date = new Date(unix * 1000); - return date.toLocaleString("ru-RU"); -} - -export function sinceUnixDate(unixInSeconds) { - const unix = Math.floor(unixInSeconds * 1000); - const date = new Date(unix); - const currentDate = new Date().valueOf(); - const dateDifferenceSeconds = new Date(currentDate - unix) / 1000; - - const minutes = Math.floor(dateDifferenceSeconds / 60) - const hours = Math.floor(dateDifferenceSeconds / 3600); - const days = Math.floor(dateDifferenceSeconds / 86400); - - const minutesName = numberDeclension(minutes, "минута", "минуты", "минут"); - const hoursName = numberDeclension(hours, "час", "часа", "часов"); - const daysName = numberDeclension(days, "день", "дня", "дней"); - - if (dateDifferenceSeconds < 60) return "менее минуты назад"; - if (dateDifferenceSeconds < 3600) - return `${minutes} ${minutesName} назад`; - if (dateDifferenceSeconds < 86400) - return `${hours} ${hoursName} назад`; - if (dateDifferenceSeconds < 2592000) - return `${days} ${daysName} назад`; - - return date.toLocaleString("ru-RU").split(",")[0]; -} - -export function minutesToTime(min) { - const d = Math.floor(min / 1440); // 60*24 - const h = Math.floor((min - d * 1440) / 60); - const m = Math.round(min % 60); - - var dDisplay = d > 0 ? `${d} ${numberDeclension(d, "день", "дня", "дней")}, ` : ""; - var hDisplay = h > 0 ? `${h} ${numberDeclension(h, "час", "часа", "часов")}, ` : ""; - var mDisplay = m > 0 ? `${m} ${numberDeclension(m, "минута", "минуты", "минут")}` : ""; - return dDisplay + hDisplay + mDisplay; -} diff --git a/app/api/utils.ts b/app/api/utils.ts new file mode 100644 index 0000000..a4c0f3f --- /dev/null +++ b/app/api/utils.ts @@ -0,0 +1,249 @@ +import { USER_AGENT, ENDPOINTS } from "./config"; +export const HEADERS = { + "User-Agent": USER_AGENT, + "Content-Type": "application/json; charset=UTF-8", +}; + +export const fetchDataViaGet = async ( + url: string, + API_V2: string | boolean = false +) => { + if (API_V2) { + HEADERS["API-Version"] = "v2"; + } + try { + const response = await fetch(url, { + headers: HEADERS, + }); + if (response.status !== 200) { + throw new Error("Error fetching data"); + } + const data = await response.json(); + return data; + } catch (error) { + console.log(error); + } +}; + +export const fetchDataViaPost = async ( + url: string, + body: string, + API_V2: string | boolean = false +) => { + if (API_V2) { + HEADERS["API-Version"] = "v2"; + } + try { + const response = await fetch(url, { + method: "POST", + headers: HEADERS, + body: body, + }); + if (response.status !== 200) { + throw new Error("Error fetching data"); + } + const data = await response.json(); + return data; + } catch (error) { + console.log(error); + } +}; + +export const authorize = async ( + url: string, + data: { login: string; password: string } +) => { + try { + const response = await fetch( + `${url}?login=${data.login}&password=${data.password}`, + { + method: "POST", + headers: { + "User-Agent": USER_AGENT, + Sign: "9aa5c7af74e8cd70c86f7f9587bde23d", + "Content-Type": "application/x-www-form-urlencoded", + }, + } + ); + if (response.status !== 200) { + throw new Error("Error authorizing user"); + } + return await response.json(); + } catch (error) { + return error; + } +}; + +export function setJWT(user_id: number | string, jwt: string) { + const data = { jwt: jwt, user_id: user_id }; + localStorage.setItem("JWT", JSON.stringify(data)); +} +export function getJWT() { + const data = localStorage.getItem("JWT"); + return JSON.parse(data); +} +export function removeJWT() { + localStorage.removeItem("JWT"); +} + +export function numberDeclension( + number: number, + one: string, + two: string, + five: string +) { + if (number > 10 && [11, 12, 13, 14].includes(number % 100)) return five; + let last_num = number % 10; + if (last_num == 1) return one; + if ([2, 3, 4].includes(last_num)) return two; + if ([5, 6, 7, 8, 9, 0].includes(last_num)) return five; +} + +const months = [ + "янв.", + "фев.", + "мар.", + "апр.", + "мая", + "июня", + "июля", + "авг.", + "сен.", + "окт.", + "ноя.", + "дек.", +]; + +export function unixToDate(unix: number) { + const date = new Date(unix * 1000); + return date.getDate() + " " + months[date.getMonth()] + " " + date.getFullYear(); +} + +export const getSeasonFromUnix = (unix: number) => { + const date = new Date(unix * 1000); + const month = date.getMonth(); + if (month >= 3 && month <= 5) return "весна"; + if (month >= 6 && month <= 8) return "лето"; + if (month >= 9 && month <= 11) return "осень"; + return "зима"; +}; + +export function sinceUnixDate(unixInSeconds: number) { + const unix = Math.floor(unixInSeconds * 1000); + const date = new Date(unix); + const currentDate = new Date().valueOf(); + const dateDifferenceSeconds = new Date(currentDate - unix).getTime() / 1000; + + const minutes = Math.floor(dateDifferenceSeconds / 60); + const hours = Math.floor(dateDifferenceSeconds / 3600); + const days = Math.floor(dateDifferenceSeconds / 86400); + + const minutesName = numberDeclension(minutes, "минута", "минуты", "минут"); + const hoursName = numberDeclension(hours, "час", "часа", "часов"); + const daysName = numberDeclension(days, "день", "дня", "дней"); + + if (dateDifferenceSeconds < 60) return "менее минуты назад"; + if (dateDifferenceSeconds < 3600) return `${minutes} ${minutesName} назад`; + if (dateDifferenceSeconds < 86400) return `${hours} ${hoursName} назад`; + if (dateDifferenceSeconds < 2592000) return `${days} ${daysName} назад`; + + return date.getDate() + " " + months[date.getMonth()] + " " + date.getFullYear(); +} + +export function minutesToTime(min: number) { + const d = Math.floor(min / 1440); // 60*24 + const h = Math.floor((min - d * 1440) / 60); + const m = Math.round(min % 60); + + var dDisplay = + d > 0 ? `${d} ${numberDeclension(d, "день", "дня", "дней")}, ` : ""; + var hDisplay = + h > 0 ? `${h} ${numberDeclension(h, "час", "часа", "часов")}, ` : ""; + var mDisplay = + m > 0 ? `${m} ${numberDeclension(m, "минута", "минуты", "минут")}` : ""; + return dDisplay + hDisplay + mDisplay; +} + +const StatusList: Record = { + last: null, + finished: 1, + ongoing: 2, + announce: 3, +}; + +export async function _FetchHomePageReleases( + status: string, + token: string | null, + page: string | number = 0 +) { + let statusId: null | number = null; + let categoryId: null | number = null; + if (status == "films") { + categoryId = 2; + } else { + statusId = StatusList[status]; + } + + const body = { + country: null, + season: null, + sort: 0, + studio: null, + age_ratings: [], + category_id: categoryId, + end_year: null, + episode_duration_from: null, + episode_duration_to: null, + episodes_from: null, + episodes_to: null, + genres: [], + profile_list_exclusions: [], + start_year: null, + status_id: statusId, + types: [], + is_genres_exclude_mode_enabled: false, + }; + + let url: string; + url = `${ENDPOINTS.filter}/${page}`; + if (token) { + url += `?token=${token}`; + } + + const data: Object = fetch(url, { + method: "POST", + body: JSON.stringify(body), + }) + .then((response) => { + if (response.ok) { + return response.json(); + } else { + throw new Error("Error fetching data"); + } + }) + .then((data: Object) => { + return data; + }) + .catch((error) => { + console.log(error); + return null; + }); + return data; +} + +export const BookmarksList = { + watching: 1, + planned: 2, + watched: 3, + delayed: 4, + abandoned: 5, +}; + +export const SortList = { + adding_descending: 1, + adding_ascending: 2, + year_descending: 3, + year_ascending: 4, + alphabet_descending: 5, + alphabet_ascending: 6, +}; diff --git a/app/bookmarks/[slug]/page.js b/app/bookmarks/[slug]/page.tsx similarity index 88% rename from app/bookmarks/[slug]/page.js rename to app/bookmarks/[slug]/page.tsx index 2387475..dd6ecf3 100644 --- a/app/bookmarks/[slug]/page.js +++ b/app/bookmarks/[slug]/page.tsx @@ -1,4 +1,4 @@ -import { BookmarksCategoryPage } from "@/app/pages/BookmarksCategory"; +import { BookmarksCategoryPage } from "#/pages/BookmarksCategory"; const SectionTitleMapping = { watching: "Смотрю", diff --git a/app/bookmarks/page.js b/app/bookmarks/page.tsx similarity index 71% rename from app/bookmarks/page.js rename to app/bookmarks/page.tsx index dea39e9..5edef18 100644 --- a/app/bookmarks/page.js +++ b/app/bookmarks/page.tsx @@ -2,7 +2,7 @@ export const metadata = { title: "Закладки", }; - import { BookmarksPage } from "@/app/pages/Bookmarks"; + import { BookmarksPage } from "#/pages/Bookmarks"; export default function Index() { return ; diff --git a/app/components/Chip/Chip.jsx b/app/components/Chip/Chip.jsx deleted file mode 100644 index 34ebb55..0000000 --- a/app/components/Chip/Chip.jsx +++ /dev/null @@ -1,12 +0,0 @@ -export const Chip = (props) => { - return ( -
-

- {props.name} - {props.name && props.devider ? props.devider : " "} - {props.name_2} -

-
- ); - }; - \ No newline at end of file diff --git a/app/components/Chip/Chip.tsx b/app/components/Chip/Chip.tsx new file mode 100644 index 0000000..c3c9e5e --- /dev/null +++ b/app/components/Chip/Chip.tsx @@ -0,0 +1,16 @@ +export const Chip = (props: { + name?: string; + name_2?: string; + devider?: string; + bg_color?: string; +}) => { + return ( +
+

+ {props.name} + {props.name && props.devider ? props.devider : " "} + {props.name_2} +

+
+ ); +}; diff --git a/app/components/Navbar/Navbar.jsx b/app/components/Navbar/Navbar.tsx similarity index 96% rename from app/components/Navbar/Navbar.jsx rename to app/components/Navbar/Navbar.tsx index 9049fa0..04e55bb 100644 --- a/app/components/Navbar/Navbar.jsx +++ b/app/components/Navbar/Navbar.tsx @@ -1,14 +1,13 @@ "use client"; import Link from "next/link"; import { usePathname } from "next/navigation"; -import { useUserStore } from "@/app/store/auth"; +import { useUserStore } from "#/store/auth"; import { Dropdown } from "flowbite-react"; export const Navbar = () => { const pathname = usePathname(); - const userStore = useUserStore((state) => state); + const userStore: any = useUserStore((state) => state); - const isNotAuthorizedStyle = "text-gray-700"; const navLinks = [ { id: 1, @@ -107,7 +106,7 @@ export const Navbar = () => { }} > - + diff --git a/app/components/RelatedSection/RelatedSection.jsx b/app/components/RelatedSection/RelatedSection.tsx similarity index 93% rename from app/components/RelatedSection/RelatedSection.jsx rename to app/components/RelatedSection/RelatedSection.tsx index 1ee3d86..e9a75b2 100644 --- a/app/components/RelatedSection/RelatedSection.jsx +++ b/app/components/RelatedSection/RelatedSection.tsx @@ -1,7 +1,7 @@ -import { numberDeclension } from "@/app/api/utils"; +import { numberDeclension } from "#/api/utils"; import Link from "next/link"; -export const RelatedSection = (props) => { +export const RelatedSection = (props: any) => { const declension = numberDeclension( props.release_count, "релиз", diff --git a/app/components/ReleaseCourusel/ReleaseCourusel.jsx b/app/components/ReleaseCourusel/ReleaseCourusel.tsx similarity index 75% rename from app/components/ReleaseCourusel/ReleaseCourusel.jsx rename to app/components/ReleaseCourusel/ReleaseCourusel.tsx index 8eaaabd..c3b5482 100644 --- a/app/components/ReleaseCourusel/ReleaseCourusel.jsx +++ b/app/components/ReleaseCourusel/ReleaseCourusel.tsx @@ -9,9 +9,13 @@ import "swiper/css"; import "swiper/css/navigation"; import { Navigation } from "swiper/modules"; -export const ReleaseCourusel = (props) => { +export const ReleaseCourusel = (props: { + sectionTitle: string; + showAllLink?: string; + content: any; +}) => { useEffect(() => { - const options = { + const options: any = { direction: "horizontal", spaceBetween: 8, allowTouchMove: true, @@ -39,12 +43,14 @@ export const ReleaseCourusel = (props) => {

{props.sectionTitle}

- -
-

Показать все

- -
- + {props.showAllLink && ( + +
+

Показать все

+ +
+ + )}
@@ -65,11 +71,15 @@ export const ReleaseCourusel = (props) => {
diff --git a/app/components/ReleaseInfo/ReleaseInfo.SearchLink.tsx b/app/components/ReleaseInfo/ReleaseInfo.SearchLink.tsx new file mode 100644 index 0000000..db19124 --- /dev/null +++ b/app/components/ReleaseInfo/ReleaseInfo.SearchLink.tsx @@ -0,0 +1,21 @@ +import Link from "next/link"; + +// const searchBy = { +// title: 0, +// studio: 1, +// director: 2, +// author: 3, +// genre: 4 +// } + +// TODO: сделать какую-нибудь анимацию на ссылке при наведении и фокусе +export const ReleaseInfoSearchLink = (props: { title: string, searchBy: string | number | null }) => { + return ( + + {props.title} + + ); +}; diff --git a/app/components/ReleaseLink/ReleaseLink.jsx b/app/components/ReleaseLink/ReleaseLink.16_9.tsx similarity index 83% rename from app/components/ReleaseLink/ReleaseLink.jsx rename to app/components/ReleaseLink/ReleaseLink.16_9.tsx index 80fd24b..737d256 100644 --- a/app/components/ReleaseLink/ReleaseLink.jsx +++ b/app/components/ReleaseLink/ReleaseLink.16_9.tsx @@ -1,18 +1,18 @@ import Link from "next/link"; -import { sinceUnixDate } from "@/app/api/utils"; -import { Chip } from "@/app/components/Chip/Chip"; +import { sinceUnixDate } from "#/api/utils"; +import { Chip } from "#/components/Chip/Chip"; -export const ReleaseLink = (props) => { +const profile_lists = { + // 0: "Не смотрю", + 1: { name: "Смотрю", bg_color: "bg-green-500" }, + 2: { name: "В планах", bg_color: "bg-purple-500" }, + 3: { name: "Просмотрено", bg_color: "bg-blue-500" }, + 4: { name: "Отложено", bg_color: "bg-yellow-500" }, + 5: { name: "Брошено", bg_color: "bg-red-500" }, +}; + +export const ReleaseLink169 = (props: any) => { const grade = props.grade.toFixed(1); - const profile_lists = { - // 0: "Не смотрю", - 1: { name: "Смотрю", bg_color: "bg-green-500" }, - 2: { name: "В планах", bg_color: "bg-purple-500" }, - 3: { name: "Просмотрено", bg_color: "bg-blue-500" }, - 4: { name: "Отложено", bg_color: "bg-yellow-500" }, - 5: { name: "Брошено", bg_color: "bg-red-500" }, - }; - const profile_list_status = props.profile_list_status; let user_list = null; if (profile_list_status != null || profile_list_status != 0) { @@ -79,6 +79,7 @@ export const ReleaseLink = (props) => { devider=", " /> )} + {props.category && } {props.is_favorite && (
diff --git a/app/components/ReleaseLink/ReleaseLink.Poster.tsx b/app/components/ReleaseLink/ReleaseLink.Poster.tsx new file mode 100644 index 0000000..a46ba67 --- /dev/null +++ b/app/components/ReleaseLink/ReleaseLink.Poster.tsx @@ -0,0 +1,82 @@ +import Link from "next/link"; +import { sinceUnixDate } from "#/api/utils"; +import { Chip } from "#/components/Chip/Chip"; + +const profile_lists = { + // 0: "Не смотрю", + 1: { name: "Смотрю", bg_color: "bg-green-500" }, + 2: { name: "В планах", bg_color: "bg-purple-500" }, + 3: { name: "Просмотрено", bg_color: "bg-blue-500" }, + 4: { name: "Отложено", bg_color: "bg-yellow-500" }, + 5: { name: "Брошено", bg_color: "bg-red-500" }, +}; + +export const ReleaseLinkPoster = (props: any) => { + const grade = props.grade.toFixed(1); + const profile_list_status = props.profile_list_status; + let user_list = null; + if (profile_list_status != null || profile_list_status != 0) { + user_list = profile_lists[profile_list_status]; + } + return ( + +
+
+
+ + {props.status ? ( + + ) : ( + + )} + +
+
+ {props.title_ru && ( +

+ {props.title_ru} +

+ )} + {props.title_original && ( +

+ {props.title_original} +

+ )} +
+
+
+ + ); +}; diff --git a/app/components/ReleaseLink/ReleaseLink.tsx b/app/components/ReleaseLink/ReleaseLink.tsx new file mode 100644 index 0000000..335f95b --- /dev/null +++ b/app/components/ReleaseLink/ReleaseLink.tsx @@ -0,0 +1,13 @@ +import { ReleaseLink169 } from "./ReleaseLink.16_9"; +import { ReleaseLinkPoster } from "./ReleaseLink.Poster"; + +export const ReleaseLink = (props: any) => { + const type = props.type || "16_9"; + + if (type == "16_9") { + return ; + } + if (type == "poster") { + return ; + } +}; diff --git a/app/components/ReleasePlayer/ReleasePlayer.tsx b/app/components/ReleasePlayer/ReleasePlayer.tsx new file mode 100644 index 0000000..2669497 --- /dev/null +++ b/app/components/ReleasePlayer/ReleasePlayer.tsx @@ -0,0 +1,174 @@ +"use client"; + +import { Spinner } from "#/components/Spinner/Spinner"; +import { useUserStore } from "#/store/auth"; +import { Card, Dropdown, Button } from "flowbite-react"; +import { ENDPOINTS } from "#/api/config"; +import { useState, useEffect } from "react"; + +const DropdownTheme = { + floating: { + target: + "w-full md:w-[256px] bg-blue-600 enabled:hover:bg-blue-700 focus:ring-4 focus:outline-none focus:ring-blue-300 text-center dark:bg-blue-600 dark:hover:bg-blue-700 dark:focus:ring-blue-800", + }, +}; + +const ButtonThemeInactive = + "bg-blue-600 enabled:hover:bg-blue-700 focus:ring-4 focus:outline-none focus:ring-blue-300 dark:bg-blue-600 dark:hover:bg-blue-700 dark:focus:ring-blue-800"; +const ButtonThemeActive = + "bg-blue-800 dark:bg-blue-600 disabled:opacity-100 dark:disabled:opacity-100"; + +async function _fetch(url: string) { + const data = fetch(url) + .then((res) => { + if (res.ok) { + return res.json(); + } else { + throw new Error("Error fetching data"); + } + }) + .catch((err) => console.log(err)); + return data; +} + +export const ReleasePlayer = (props: { id: number }) => { + const token = useUserStore((state) => state.token); + const [voiceoverInfo, setVoiceoverInfo] = useState(null); + const [selectedVoiceover, setSelectedVoiceover] = useState(null); + const [sourcesInfo, setSourcesInfo] = useState(null); + const [selectedSource, setSelectedSource] = useState(null); + const [episodeInfo, setEpisodeInfo] = useState(null); + const [selectedEpisode, setSelectedEpisode] = useState(null); + const [isFirstLoad, setIsFirstLoad] = useState(true); + + useEffect(() => { + async function _fetchInfo() { + const voiceover = await _fetch( + `${ENDPOINTS.release.episode}/${props.id}` + ); + setVoiceoverInfo(voiceover.types); + setSelectedVoiceover(voiceover.types[0]); + } + _fetchInfo(); + // eslint-disable-next-line react-hooks/exhaustive-deps + }, []); + + useEffect(() => { + async function _fetchInfo() { + const sources = await _fetch( + `${ENDPOINTS.release.episode}/${props.id}/${selectedVoiceover.id}` + ); + setSourcesInfo(sources.sources); + setSelectedSource(sources.sources[0]); + } + if (selectedVoiceover) { + _fetchInfo(); + } + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [selectedVoiceover]); + + useEffect(() => { + async function _fetchInfo(url: string) { + const episodes = await _fetch(url); + + setEpisodeInfo(episodes.episodes); + setSelectedEpisode(episodes.episodes[0]); + } + if (selectedSource) { + let url = `${ENDPOINTS.release.episode}/${props.id}/${selectedVoiceover.id}/${selectedSource.id}`; + if (token) { + url = `${ENDPOINTS.release.episode}/${props.id}/${selectedVoiceover.id}/${selectedSource.id}?token=${token}`; + } + _fetchInfo(url); + } + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [selectedSource, token]); + + useEffect(() => { + async function _fetchInfo() { + _fetch(`${ENDPOINTS.statistic.addHistory}/${props.id}/${selectedVoiceover.id}/${selectedSource.id}?token=${token}`); + _fetch(`${ENDPOINTS.statistic.markWatched}/${props.id}/${selectedVoiceover.id}/${selectedSource.id}?token=${token}`); + } + if (selectedEpisode && !isFirstLoad && token) { + _fetchInfo(); + } + + if (isFirstLoad) { + setIsFirstLoad(false); + } + + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [selectedEpisode]); + + return ( + + {!voiceoverInfo || !sourcesInfo || !episodeInfo ? ( +
+ +
+ ) : ( + <> +
+ + {voiceoverInfo.map((voiceover: any) => ( + setSelectedVoiceover(voiceover)} + > + {voiceover.name} + + ))} + + + {sourcesInfo.map((source: any) => ( + setSelectedSource(source)} + > + {source.name} + + ))} + +
+
+ +
+
+
+ {episodeInfo.map((episode: any) => ( + + ))} +
+
+ + )} +
+ ); +}; diff --git a/app/components/ReleaseSection/ReleaseSection.jsx b/app/components/ReleaseSection/ReleaseSection.tsx similarity index 94% rename from app/components/ReleaseSection/ReleaseSection.jsx rename to app/components/ReleaseSection/ReleaseSection.tsx index b4d4b88..083a8f1 100644 --- a/app/components/ReleaseSection/ReleaseSection.jsx +++ b/app/components/ReleaseSection/ReleaseSection.tsx @@ -1,6 +1,6 @@ import { ReleaseLink } from "../ReleaseLink/ReleaseLink"; -export const ReleaseSection = (props) => { +export const ReleaseSection = (props: any) => { return (
{props.sectionTitle && ( diff --git a/app/components/Spinner/Spinner.jsx b/app/components/Spinner/Spinner.tsx similarity index 100% rename from app/components/Spinner/Spinner.jsx rename to app/components/Spinner/Spinner.tsx diff --git a/app/favorites/page.js b/app/favorites/page.tsx similarity index 69% rename from app/favorites/page.js rename to app/favorites/page.tsx index a86a845..b2f5394 100644 --- a/app/favorites/page.js +++ b/app/favorites/page.tsx @@ -2,7 +2,7 @@ export const metadata = { title: "Избранное", }; -import { FavoritesPage } from "@/app/pages/Favorites"; +import { FavoritesPage } from "#/pages/Favorites"; export default function Index() { return ; diff --git a/app/history/page.js b/app/history/page.tsx similarity index 70% rename from app/history/page.js rename to app/history/page.tsx index 79e029d..2815aa3 100644 --- a/app/history/page.js +++ b/app/history/page.tsx @@ -2,7 +2,7 @@ export const metadata = { title: "История", }; -import { HistoryPage } from "@/app/pages/History"; +import { HistoryPage } from "#/pages/History"; export default function Index() { return ; diff --git a/app/home/[slug]/page.js b/app/home/[slug]/page.tsx similarity index 90% rename from app/home/[slug]/page.js rename to app/home/[slug]/page.tsx index e982c2b..37845d4 100644 --- a/app/home/[slug]/page.js +++ b/app/home/[slug]/page.tsx @@ -1,4 +1,4 @@ -import { IndexCategoryPage } from "@/app/pages/IndexCategory"; +import { IndexCategoryPage } from "#/pages/IndexCategory"; const SectionTitleMapping = { last: "Последние релизы", diff --git a/app/hooks/useScrollPosition.js b/app/hooks/useScrollPosition.ts similarity index 100% rename from app/hooks/useScrollPosition.js rename to app/hooks/useScrollPosition.ts diff --git a/app/layout.js b/app/layout.tsx similarity index 89% rename from app/layout.js rename to app/layout.tsx index b13a990..a116406 100644 --- a/app/layout.js +++ b/app/layout.tsx @@ -1,5 +1,5 @@ import "./globals.css"; -import { App } from "@/app/App"; +import { App } from "./App"; export const metadata = { title: { diff --git a/app/login/page.js b/app/login/page.tsx similarity index 56% rename from app/login/page.js rename to app/login/page.tsx index 2eb561a..a3c1924 100644 --- a/app/login/page.js +++ b/app/login/page.tsx @@ -1,4 +1,4 @@ -import { LoginPage } from "@/app/pages/Login"; +import { LoginPage } from "#/pages/Login"; export default function Login() { return ; } diff --git a/app/page.js b/app/page.tsx similarity index 100% rename from app/page.js rename to app/page.tsx diff --git a/app/pages/Bookmarks.jsx b/app/pages/Bookmarks.tsx similarity index 70% rename from app/pages/Bookmarks.jsx rename to app/pages/Bookmarks.tsx index 9778711..0e03385 100644 --- a/app/pages/Bookmarks.jsx +++ b/app/pages/Bookmarks.tsx @@ -1,16 +1,22 @@ "use client"; import useSWR from "swr"; -import { ReleaseCourusel } from "@/app/components/ReleaseCourusel/ReleaseCourusel"; -import { Spinner } from "@/app/components/Spinner/Spinner"; -const fetcher = (...args) => fetch(...args).then((res) => res.json()); -import { useUserStore } from "@/app/store/auth"; +import { ReleaseCourusel } from "#/components/ReleaseCourusel/ReleaseCourusel"; +import { Spinner } from "#/components/Spinner/Spinner"; +const fetcher = (...args: any) => + fetch([...args] as any).then((res) => res.json()); +import { useUserStore } from "#/store/auth"; +import { BookmarksList } from "#/api/utils"; +import { ENDPOINTS } from "#/api/config"; export function BookmarksPage() { const token = useUserStore((state) => state.token); - function useFetchReleases(list) { - let url; - url = `/api/bookmarks?list=${list}&token=${token}`; + function useFetchReleases(listName: string) { + let url: string; + + if (token) { + url = `${ENDPOINTS.user.bookmark}/all/${BookmarksList[listName]}/0?token=${token}`; + } const { data } = useSWR(url, fetcher); return [data]; @@ -58,15 +64,13 @@ export function BookmarksPage() { content={watchedData.content} /> )} - {delayedData && - delayedData.content && - delayedData.content.length > 0 && ( - - )} + {delayedData && delayedData.content && delayedData.content.length > 0 && ( + + )} {abandonedData && abandonedData.content && abandonedData.content.length > 0 && ( diff --git a/app/pages/BookmarksCategory.jsx b/app/pages/BookmarksCategory.tsx similarity index 82% rename from app/pages/BookmarksCategory.jsx rename to app/pages/BookmarksCategory.tsx index 843907a..3cc26e2 100644 --- a/app/pages/BookmarksCategory.jsx +++ b/app/pages/BookmarksCategory.tsx @@ -1,34 +1,37 @@ "use client"; import useSWRInfinite from "swr/infinite"; -import { ReleaseSection } from "@/app/components/ReleaseSection/ReleaseSection"; -import { Spinner } from "@/app/components/Spinner/Spinner"; +import { ReleaseSection } from "#/components/ReleaseSection/ReleaseSection"; +import { Spinner } from "#/components/Spinner/Spinner"; import { useState, useEffect } from "react"; -import { useScrollPosition } from "@/app/hooks/useScrollPosition"; +import { useScrollPosition } from "#/hooks/useScrollPosition"; import { useUserStore } from "../store/auth"; import { Dropdown } from "flowbite-react"; import { sort } from "./common"; +import { ENDPOINTS } from "#/api/config"; +import { BookmarksList, SortList } from "#/api/utils"; -const fetcher = async (url) => { +const fetcher = async (url: string) => { const res = await fetch(url); if (!res.ok) { - const error = new Error("An error occurred while fetching the data."); - error.info = await res.json(); - error.status = res.status; + const error = new Error(`An error occurred while fetching the data. status: ${res.status}`); + error.message = await res.json(); throw error; } return res.json(); }; -export function BookmarksCategoryPage(props) { +export function BookmarksCategoryPage(props: any) { const token = useUserStore((state) => state.token); const [selectedSort, setSelectedSort] = useState(0); const [isLoadingEnd, setIsLoadingEnd] = useState(false); - const getKey = (pageIndex, previousPageData) => { + const getKey = (pageIndex: number, previousPageData: any) => { if (previousPageData && !previousPageData.content.length) return null; - return `/api/bookmarks?list=${props.slug}&page=${pageIndex}&token=${token}&sort=${sort.values[selectedSort].value}`; + if (token) { + return `${ENDPOINTS.user.bookmark}/all/${BookmarksList[props.slug]}/${pageIndex}?token=${token}&sort=${sort.values[selectedSort].id}`; + } }; const { data, error, isLoading, size, setSize } = useSWRInfinite( diff --git a/app/pages/Favorites.jsx b/app/pages/Favorites.tsx similarity index 85% rename from app/pages/Favorites.jsx rename to app/pages/Favorites.tsx index aadd0b3..563b95b 100644 --- a/app/pages/Favorites.jsx +++ b/app/pages/Favorites.tsx @@ -1,20 +1,20 @@ "use client"; import useSWRInfinite from "swr/infinite"; -import { ReleaseSection } from "@/app/components/ReleaseSection/ReleaseSection"; -import { Spinner } from "@/app/components/Spinner/Spinner"; +import { ReleaseSection } from "#/components/ReleaseSection/ReleaseSection"; +import { Spinner } from "#/components/Spinner/Spinner"; import { useState, useEffect } from "react"; -import { useScrollPosition } from "@/app/hooks/useScrollPosition"; +import { useScrollPosition } from "#/hooks/useScrollPosition"; import { useUserStore } from "../store/auth"; import { Dropdown } from "flowbite-react"; import { sort } from "./common"; +import { ENDPOINTS } from "#/api/config"; -const fetcher = async (url) => { +const fetcher = async (url: string) => { const res = await fetch(url); if (!res.ok) { - const error = new Error("An error occurred while fetching the data."); - error.info = await res.json(); - error.status = res.status; + const error = new Error(`An error occurred while fetching the data. status: ${res.status}`); + error.message = await res.json(); throw error; } @@ -26,9 +26,11 @@ export function FavoritesPage() { const [selectedSort, setSelectedSort] = useState(0); const [isLoadingEnd, setIsLoadingEnd] = useState(false); - const getKey = (pageIndex, previousPageData) => { + const getKey = (pageIndex: number, previousPageData: any) => { if (previousPageData && !previousPageData.content.length) return null; - return `/api/favorites?page=${pageIndex}&token=${token}&sort=${sort.values[selectedSort].value}`; + if (token) { + return `${ENDPOINTS.user.favorite}/all/${pageIndex}?token=${token}&sort=${sort.values[selectedSort].id}`; + } }; const { data, error, isLoading, size, setSize } = useSWRInfinite( diff --git a/app/pages/History.jsx b/app/pages/History.tsx similarity index 81% rename from app/pages/History.jsx rename to app/pages/History.tsx index 26367a0..81c76da 100644 --- a/app/pages/History.jsx +++ b/app/pages/History.tsx @@ -1,18 +1,18 @@ "use client"; import useSWRInfinite from "swr/infinite"; -import { ReleaseSection } from "@/app/components/ReleaseSection/ReleaseSection"; -import { Spinner } from "@/app/components/Spinner/Spinner"; +import { ReleaseSection } from "#/components/ReleaseSection/ReleaseSection"; +import { Spinner } from "#/components/Spinner/Spinner"; import { useState, useEffect } from "react"; -import { useScrollPosition } from "@/app/hooks/useScrollPosition"; +import { useScrollPosition } from "#/hooks/useScrollPosition"; import { useUserStore } from "../store/auth"; +import { ENDPOINTS } from "#/api/config"; -const fetcher = async (url) => { +const fetcher = async (url: string) => { const res = await fetch(url); if (!res.ok) { - const error = new Error("An error occurred while fetching the data."); - error.info = await res.json(); - error.status = res.status; + const error = new Error(`An error occurred while fetching the data. status: ${res.status}`); + error.message = await res.json(); throw error; } @@ -23,9 +23,11 @@ export function HistoryPage() { const token = useUserStore((state) => state.token); const [isLoadingEnd, setIsLoadingEnd] = useState(false); - const getKey = (pageIndex, previousPageData) => { + const getKey = (pageIndex: number, previousPageData: any) => { if (previousPageData && !previousPageData.content.length) return null; - return `/api/history?page=${pageIndex}&token=${token}`; + if (token) { + return `${ENDPOINTS.user.history}/${pageIndex}?token=${token}`; + } }; const { data, error, isLoading, size, setSize } = useSWRInfinite( diff --git a/app/pages/Index.jsx b/app/pages/Index.jsx deleted file mode 100644 index 5db6093..0000000 --- a/app/pages/Index.jsx +++ /dev/null @@ -1,64 +0,0 @@ -"use client"; -import useSWR from "swr"; -import { ReleaseCourusel } from "@/app/components/ReleaseCourusel/ReleaseCourusel"; -import { Spinner } from "@/app/components/Spinner/Spinner"; -const fetcher = (...args) => fetch(...args).then((res) => res.json()); -import { useUserStore } from "@/app/store/auth"; - -export function IndexPage() { - const userStore = useUserStore((state) => state); - const token = userStore.token; - - function useFetchReleases(status) { - let url; - - url = `/api/home?status=${status}`; - if (token) { - url += `&token=${token}`; - } - const { data } = useSWR(url, fetcher); - return [data]; - } - - const [lastReleasesData] = useFetchReleases("last"); - const [finishedReleasesData] = useFetchReleases("finished"); - const [ongoingReleasesData] = useFetchReleases("ongoing"); - const [announceReleasesData] = useFetchReleases("announce"); - - return ( -
- {lastReleasesData ? ( - - ) : ( -
- -
- )} - {finishedReleasesData && ( - - )} - {ongoingReleasesData && ( - - )} - {announceReleasesData && ( - - )} -
- ); -} diff --git a/app/pages/Index.tsx b/app/pages/Index.tsx new file mode 100644 index 0000000..4647ef0 --- /dev/null +++ b/app/pages/Index.tsx @@ -0,0 +1,94 @@ +"use client"; +import { ReleaseCourusel } from "#/components/ReleaseCourusel/ReleaseCourusel"; +import { Spinner } from "#/components/Spinner/Spinner"; +import { useUserStore } from "#/store/auth"; +import { useState, useEffect } from "react"; +import { _FetchHomePageReleases } from "#/api/utils"; + +export function IndexPage() { + const token = useUserStore((state) => state.token); + const [isLoading, setIsLoading] = useState(true); + const [lastReleasesData, setLastReleasesData] = useState(null); + const [ongoingReleasesData, setOngoingReleasesData] = useState(null); + const [finishedReleasesData, setFinishedReleasesData] = useState(null); + const [announceReleasesData, setAnnounceReleasesData] = useState(null); + const [filmsReleasesData, setFilmsReleasesData] = useState(null); + + useEffect(() => { + async function _loadReleases() { + setIsLoading(true); + setLastReleasesData(null); + setOngoingReleasesData(null); + setFinishedReleasesData(null); + setAnnounceReleasesData(null); + setFilmsReleasesData(null); + + const lastReleases = await _FetchHomePageReleases("last", token); + const ongoingReleases = await _FetchHomePageReleases("ongoing", token); + const finishedReleases = await _FetchHomePageReleases("finished", token); + const announceReleases = await _FetchHomePageReleases("announce", token); + const filmsReleases = await _FetchHomePageReleases("films", token); + + setLastReleasesData(lastReleases); + setOngoingReleasesData(ongoingReleases); + setFinishedReleasesData(finishedReleases); + setAnnounceReleasesData(announceReleases); + setFilmsReleasesData(filmsReleases); + setIsLoading(false); + } + _loadReleases(); + }, [token]); + + return ( +
+ {lastReleasesData ? ( + + ) : ( +
+ +
+ )} + {finishedReleasesData && ( + + )} + {ongoingReleasesData && ( + + )} + {announceReleasesData && ( + + )} + {filmsReleasesData && ( + + )} + {!isLoading && + !lastReleasesData && + !finishedReleasesData && + !ongoingReleasesData && + !announceReleasesData && ( +
+

Ошибка загрузки контента...

+
+ )} +
+ ); +} diff --git a/app/pages/IndexCategory.jsx b/app/pages/IndexCategory.jsx deleted file mode 100644 index 586dfc9..0000000 --- a/app/pages/IndexCategory.jsx +++ /dev/null @@ -1,91 +0,0 @@ -"use client"; -import useSWRInfinite from "swr/infinite"; -import { ReleaseSection } from "@/app/components/ReleaseSection/ReleaseSection"; -import { Spinner } from "@/app/components/Spinner/Spinner"; -import { useState, useEffect } from "react"; -import { useScrollPosition } from "@/app/hooks/useScrollPosition"; -import { useUserStore } from "../store/auth"; - -const fetcher = async (url) => { - const res = await fetch(url); - - if (!res.ok) { - const error = new Error("An error occurred while fetching the data."); - error.info = await res.json(); - error.status = res.status; - throw error; - } - - return res.json(); -}; - -export function IndexCategoryPage(props) { - const userStore = useUserStore((state) => state); - const [isLoadingEnd, setIsLoadingEnd] = useState(false); - const token = userStore.token; - const getKey = (pageIndex, previousPageData) => { - if (previousPageData && !previousPageData.content.length) return null; - if (token) { - return `/api/home?status=${props.slug}&page=${pageIndex}&token=${token}`; - } - return `/api/home?status=${props.slug}&page=${pageIndex}`; - }; - - const { data, error, isLoading, size, setSize } = useSWRInfinite( - getKey, - fetcher, - { initialSize: 2, revalidateFirstPage: false } - ); - - const [content, setContent] = useState(null); - useEffect(() => { - if (data) { - let allReleases = []; - for (let i = 0; i < data.length; i++) { - allReleases.push(...data[i].content); - } - setContent(allReleases); - setIsLoadingEnd(true); - } - }, [data]); - - const scrollPosition = useScrollPosition(); - useEffect(() => { - if (scrollPosition >= 98 && scrollPosition <= 99) { - setSize(size + 1); - } - }, [scrollPosition]); - - if (error) return
failed to load
; - - return ( -
- {content && content.length > 0 ? ( - - ) : !isLoadingEnd ? ( -
- -
- ) : ( -
- -

- В списке {props.SectionTitleMapping[props.slug]} пока ничего нет... -

-
- )} - {data && data[data.length - 1].content.length == 25 && ( - - )} -
- ); -} diff --git a/app/pages/IndexCategory.tsx b/app/pages/IndexCategory.tsx new file mode 100644 index 0000000..a9a9fbf --- /dev/null +++ b/app/pages/IndexCategory.tsx @@ -0,0 +1,77 @@ +"use client"; +import { ReleaseSection } from "#/components/ReleaseSection/ReleaseSection"; +import { Spinner } from "#/components/Spinner/Spinner"; +import { useState, useEffect } from "react"; +import { useScrollPosition } from "#/hooks/useScrollPosition"; +import { useUserStore } from "../store/auth"; +import { _FetchHomePageReleases } from "#/api/utils"; + +export function IndexCategoryPage(props) { + const token = useUserStore((state) => state.token); + const [isLoading, setIsLoading] = useState(true); + const [content, setContent] = useState(null); + const [page, setPage] = useState(0); + + useEffect(() => { + async function _loadInitialReleases() { + setIsLoading(true); + setContent(null); + + const data: any = await _FetchHomePageReleases(props.slug, token, page); + + setContent(data.content); + setIsLoading(false); + } + + _loadInitialReleases(); + }, [token]); + + useEffect(() => { + async function _loadNextReleasesPage() { + const data: any = await _FetchHomePageReleases(props.slug, token, page); + const newContent = [...content, ...data.content]; + setContent(newContent); + } + if (content) { + _loadNextReleasesPage(); + } + }, [page]); + + const scrollPosition = useScrollPosition(); + useEffect(() => { + if (scrollPosition == 98) { + setPage(page + 1); + } + }, [scrollPosition]); + + // if (error) return
failed to load
; + + return ( +
+ {content && content.length > 0 ? ( + + ) : isLoading ? ( +
+ +
+ ) : ( +
+ +

+ В списке {props.SectionTitleMapping[props.slug]} пока ничего нет... +

+
+ )} + +
+ ); +} diff --git a/app/pages/Login.jsx b/app/pages/Login.tsx similarity index 93% rename from app/pages/Login.jsx rename to app/pages/Login.tsx index d0c25c8..0f1f239 100644 --- a/app/pages/Login.jsx +++ b/app/pages/Login.tsx @@ -1,13 +1,13 @@ "use client"; import { useState, useEffect } from "react"; -import { useUserStore } from "@/app/store/auth"; -import { setJWT } from "@/app/api/utils"; +import { useUserStore } from "#/store/auth"; +import { setJWT } from "#/api/utils"; import { useRouter } from "next/navigation"; export function LoginPage() { const [login, setLogin] = useState(""); const [password, setPassword] = useState(""); - const [remember, setRemember] = useState(false); + const [remember, setRemember]: any = useState(false); const userStore = useUserStore(); const router = useRouter(); @@ -68,17 +68,17 @@ export function LoginPage() { htmlFor="email" className="block mb-2 text-sm font-medium text-gray-900 dark:text-white" > - Эл. почта + Логин или эл. почта setLogin(e.target.value)} - required="" + required={true} />
@@ -94,7 +94,7 @@ export function LoginPage() { id="password" placeholder="••••••••" className="bg-gray-50 border border-gray-300 text-gray-900 rounded-lg focus:ring-primary-600 focus:border-primary-600 block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500" - required="" + required={true} value={password} onChange={(e) => setPassword(e.target.value)} /> @@ -107,7 +107,7 @@ export function LoginPage() { aria-describedby="remember" type="checkbox" className="w-4 h-4 border border-gray-300 rounded bg-gray-50 focus:ring-3 focus:ring-primary-300 dark:bg-gray-700 dark:border-gray-600 dark:focus:ring-primary-600 dark:ring-offset-gray-800" - required="" + required={true} value={remember} onChange={(e) => setRemember(e.target.checked)} /> diff --git a/app/pages/Profile.jsx b/app/pages/Profile.tsx similarity index 90% rename from app/pages/Profile.jsx rename to app/pages/Profile.tsx index 4406d84..73a3dbd 100644 --- a/app/pages/Profile.jsx +++ b/app/pages/Profile.tsx @@ -1,14 +1,14 @@ "use client"; -import { useUserStore } from "@/app/store/auth"; +import { useUserStore } from "#/store/auth"; import { useEffect, useState } from "react"; import { fetchDataViaGet } from "../api/utils"; import { Spinner } from "../components/Spinner/Spinner"; import { Avatar, Card, Button, Table } from "flowbite-react"; import { Chip } from "../components/Chip/Chip"; import { unixToDate, minutesToTime } from "../api/utils"; -import { ReleaseLink } from "../components/ReleaseLink/ReleaseLink"; +import { ReleaseCourusel } from "#/components/ReleaseCourusel/ReleaseCourusel"; -export const ProfilePage = (props) => { +export const ProfilePage = (props: any) => { const authUser = useUserStore((state) => state); const [user, setUser] = useState(null); const [isMyProfile, setIsMyProfile] = useState(false); @@ -74,7 +74,7 @@ export const ProfilePage = (props) => { ]; return ( -
+
@@ -111,20 +111,17 @@ export const ProfilePage = (props) => { > {socials.map((social) => { if (!social.nickname) return null; - if (social.name == "discord") return ( - - ) + if (social.name == "discord") + return ( + + ); return (
-
- -

Недавно просмотренные

-
- {user.history.map((release) => { - return ; - })} -
-
+
+
); diff --git a/app/pages/Release.tsx b/app/pages/Release.tsx new file mode 100644 index 0000000..0cef66a --- /dev/null +++ b/app/pages/Release.tsx @@ -0,0 +1,359 @@ +"use client"; + +import useSWR from "swr"; +import { Spinner } from "#/components/Spinner/Spinner"; +const fetcher = (...args: any) => + fetch([...args] as any).then((res) => res.json()); +import { useUserStore } from "#/store/auth"; +import { Card, Dropdown, Button } from "flowbite-react"; +import { useEffect, useState } from "react"; +import { unixToDate, getSeasonFromUnix, minutesToTime } from "#/api/utils"; +import { ReleaseLink } from "#/components/ReleaseLink/ReleaseLink"; +import { ReleasePlayer } from "#/components/ReleasePlayer/ReleasePlayer"; +import { ENDPOINTS } from "#/api/config"; +import { Table } from "flowbite-react"; +import { ReleaseInfoSearchLink } from "#/components/ReleaseInfo/ReleaseInfo.SearchLink"; +import Link from "next/link"; + +const lists = [ + { list: 0, name: "Не смотрю" }, + { list: 1, name: "Смотрю" }, + { list: 2, name: "В планах" }, + { list: 3, name: "Просмотрено" }, + { list: 4, name: "Отложено" }, + { list: 5, name: "Брошено" }, +]; + +const weekDay = [ + "_", + "каждый понедельник", + "каждый вторник", + "каждую среду", + "каждый четверг", + "каждую пятницу", + "каждую субботу", + "каждое воскресенье", +]; + +const YearSeason = ["_", "Зима", "Весна", "Лето", "Осень"]; + +const DropdownTheme = { + floating: { + target: + "flex-1 bg-blue-600 enabled:hover:bg-blue-700 focus:ring-4 focus:outline-none focus:ring-blue-300 text-center dark:bg-blue-600 dark:hover:bg-blue-700 dark:focus:ring-blue-800", + }, +}; + +export const ReleasePage = (props: any) => { + const token = useUserStore((state) => state.token); + const [userList, setUserList] = useState(0); + const [userFavorite, setUserFavorite] = useState(false); + + function useFetch(id: number) { + let url: string; + + url = `/api/release/${id}`; + if (token) { + url += `?token=${token}`; + } + const { data, isLoading, error } = useSWR(url, fetcher); + return [data, isLoading, error]; + } + const [data, isLoading, error] = useFetch(props.id); + + useEffect(() => { + if (data) { + const el = document.getElementById("note"); + if (el) { + el.innerHTML = data.release.note; + } + setUserList(data.release.profile_list_status || 0); + setUserFavorite(data.release.is_favorite); + } + }, [data]); + + function _addToFavorite() { + if (data && token) { + setUserFavorite(!userFavorite); + if (userFavorite) { + fetch( + `${ENDPOINTS.user.favorite}/delete/${data.release.id}?token=${token}` + ); + } else { + fetch( + `${ENDPOINTS.user.favorite}/add/${data.release.id}?token=${token}` + ); + } + } + } + + function _addToList(list: number) { + if (data && token) { + setUserList(list); + fetch( + `${ENDPOINTS.user.bookmark}/add/${list}/${data.release.id}?token=${token}` + ); + } + } + + return data ? ( +
+
+
+ +
+ +
+
+ {data.release.title_ru && ( +

+ {data.release.title_ru} +

+ )} + {data.release.title_original && ( +

+ {data.release.title_original} +

+ )} +
+ {data.release.note && ( +
+
+
+ )} + {data.release.description &&

{data.release.description}

} +
+
+
+ {data.release.status.name.toLowerCase() != "анонс" && ( + + )} +
+
+ + + + + + {data.release.country ? ( + data.release.country.toLowerCase() == "япония" ? ( + + ) : ( + + ) + ) : ( + + )} + + + {data.release.country && data.release.country} + {(data.release.aired_on_date != 0 || data.release.year) && + ", "} + {data.release.aired_on_date != 0 && + `${getSeasonFromUnix(data.release.aired_on_date)} `} + {data.release.year && `${data.release.year} г.`} + + + + + + + + {data.release.episodes_released + ? data.release.episodes_released + : "?"} + {"/"} + {data.release.episodes_total + ? data.release.episodes_total + " эп. " + : "? эп. "} + {data.release.duration != 0 && + `по ${minutesToTime(data.release.duration)}`} + + + + + + + + {data.release.category.name} + {", "} + {data.release.broadcast == 0 + ? data.release.status.name.toLowerCase() + : `выходит ${weekDay[data.release.broadcast]}`} + + + + + + + + {data.release.studio && ( + <> + {"Студия: "} + {data.release.studio + .split(", ") + .map((studio: string, index: number) => { + return ( + <> + {index > 0 && ", "} + + + ); + })} + {(data.release.author || data.release.director) && ", "} + + )} + {data.release.author && ( + <> + {"Автор: "} + + {data.release.director && ", "} + + )} + {data.release.director && ( + <> + {"Режиссёр: "} + + + )} + + + + + + + + {data.release.genres && + data.release.genres + .split(", ") + .map((genre: string, index: number) => { + return ( + <> + {index > 0 && ", "} + + + ); + })} + + + {data.release.status.name.toLowerCase() == "анонс" && ( + + + + + + {data.release.aired_on_date != 0 ? ( + unixToDate(data.release.aired_on_date) + ) : data.release.year ? ( + <> + {data.release.season && data.release.season != 0 + ? `${YearSeason[data.release.season]} ` + : ""} + {data.release.year && `${data.release.year} г.`} + + ) : ( + "Скоро" + )} + + + )} + +
+
+ {token && ( + +
+ + {lists.map((list) => ( + _addToList(list.list)} + > + {list.name} + + ))} + + +
+
+ )} + {data.release.related_releases.length > 0 && ( + +
+
+

Связанные релизы

+ {data.release.related && ( + +
+

Показать все

+ +
+ + )} +
+
+ {data.release.related_releases.map((release) => { + if (release.id == data.release.id) return null; + return ; + })} +
+
+
+ )} +
+
+
+ ) : ( +
+ +
+ ); +}; + +{ + /* */ +} diff --git a/app/pages/Search.jsx b/app/pages/Search.tsx similarity index 85% rename from app/pages/Search.jsx rename to app/pages/Search.tsx index 28098f9..b1938bc 100644 --- a/app/pages/Search.jsx +++ b/app/pages/Search.tsx @@ -1,21 +1,20 @@ "use client"; import useSWRInfinite from "swr/infinite"; -import { ReleaseSection } from "@/app/components/ReleaseSection/ReleaseSection"; -import { RelatedSection } from "@/app/components/RelatedSection/RelatedSection"; -import { Spinner } from "@/app/components/Spinner/Spinner"; +import { ReleaseSection } from "#/components/ReleaseSection/ReleaseSection"; +import { RelatedSection } from "#/components/RelatedSection/RelatedSection"; +import { Spinner } from "#/components/Spinner/Spinner"; import { useState, useEffect } from "react"; -import { useScrollPosition } from "@/app/hooks/useScrollPosition"; +import { useScrollPosition } from "#/hooks/useScrollPosition"; import { useRouter } from "next/navigation"; import { useSearchParams } from "next/navigation"; import { useUserStore } from "../store/auth"; -const fetcher = async (url) => { +const fetcher = async (url: string) => { const res = await fetch(url); if (!res.ok) { - const error = new Error("An error occurred while fetching the data."); - error.info = await res.json(); - error.status = res.status; + const error = new Error(`An error occurred while fetching the data. status: ${res.status}`); + error.message = await res.json(); throw error; } @@ -26,18 +25,34 @@ export function SearchPage() { const router = useRouter(); const searchParams = useSearchParams(); const [query, setQuery] = useState(searchParams.get("q") || null); + const where = searchParams.get("where") || null + const searchBy = searchParams.get("searchBy") || null + const list = searchParams.get("list") || null + const token = useUserStore((state) => state.token); - const getKey = (pageIndex, previousPageData) => { + const getKey = (pageIndex: number, previousPageData: any) => { if (previousPageData && !previousPageData.releases.length) return null; const url = new URL("/api/search", window.location.origin); - url.searchParams.set("page", pageIndex); + url.searchParams.set("page", pageIndex.toString()); if (token) { url.searchParams.set("token", token); } + if (where) { + url.searchParams.set("where", where); + } + + if (searchBy) { + url.searchParams.set("searchBy", searchBy); + } + + if (list) { + url.searchParams.set("list", list); + } + if (query) { url.searchParams.set("q", query); return url.toString(); diff --git a/app/pages/common.js b/app/pages/common.ts similarity index 91% rename from app/pages/common.js rename to app/pages/common.ts index 01e777c..24e05d8 100644 --- a/app/pages/common.js +++ b/app/pages/common.ts @@ -5,26 +5,32 @@ export const sort = { { name: "По добавлению новых", value: "adding_descending", + id: 1 }, { name: "По добавлению старых", value: "adding_ascending", + id: 2 }, { name: "По дате выхода новых", value: "year_descending", + id: 3 }, { name: "По дате выхода старых", value: "year_ascending", + id: 4 }, { name: "По алфавиту А-Я", value: "alphabet_descending", + id: 5 }, { name: "По алфавиту Я-А", value: "alphabet_ascending", + id: 6 }, ], }; diff --git a/app/profile/[id]/page.js b/app/profile/[id]/page.js deleted file mode 100644 index 5117077..0000000 --- a/app/profile/[id]/page.js +++ /dev/null @@ -1,17 +0,0 @@ -import { ProfilePage } from "@/app/pages/Profile"; -import { fetchDataViaGet } from "@/app/api/utils"; -import { ENDPOINTS } from "@/app/api/config"; - -export async function generateMetadata({ params }) { - const id = params.id - const profile = await fetchDataViaGet(`${ENDPOINTS.user.profile}/${id}`); - - return { - title: "Профиль " + profile.profile.login, - }; -} - -export default async function Search({ params }) { - const id = params.id - return ; -} diff --git a/app/profile/[id]/page.tsx b/app/profile/[id]/page.tsx new file mode 100644 index 0000000..644beca --- /dev/null +++ b/app/profile/[id]/page.tsx @@ -0,0 +1,16 @@ +import { ProfilePage } from "#/pages/Profile"; +import { fetchDataViaGet } from "#/api/utils"; + +export async function generateMetadata({ params }) { + const id:string = params.id; + const profile: any = await fetchDataViaGet(`https://api.anixart.tv/profile/${id}`); + + return { + title: "Профиль " + profile.profile.login, + }; +} + +export default async function Profile({ params }) { + const id: string = params.id; + return ; +} diff --git a/app/profile/page.js b/app/profile/page.js deleted file mode 100644 index 4bfa736..0000000 --- a/app/profile/page.js +++ /dev/null @@ -1,14 +0,0 @@ -"use client" -import { useRouter } from "next/navigation"; -import { getJWT } from "../api/utils"; - -export default function myProfile() { - const user = getJWT() - const router = useRouter() - - if (!user) { - return router.push("/login") - } else { - return router.push(`/profile/${user.user_id}`) - } -} \ No newline at end of file diff --git a/app/release/[id]/page.tsx b/app/release/[id]/page.tsx new file mode 100644 index 0000000..6630a5c --- /dev/null +++ b/app/release/[id]/page.tsx @@ -0,0 +1,16 @@ +import { ReleasePage } from "#/pages/Release"; +import { fetchDataViaGet } from "#/api/utils"; + +export async function generateMetadata({ params }) { + const id = params.id + const release = await fetchDataViaGet(`https://api.anixart.tv/release/${id}`); + + return { + title: release.release.title_ru, + }; +} + +export default async function Search({ params }) { + const id = params.id + return ; +} diff --git a/app/search/page.js b/app/search/page.tsx similarity index 87% rename from app/search/page.js rename to app/search/page.tsx index 229a4d2..0bec3c3 100644 --- a/app/search/page.js +++ b/app/search/page.tsx @@ -1,5 +1,5 @@ import dynamic from "next/dynamic"; -import { SearchPage } from "@/app/pages/Search"; +import { SearchPage } from "#/pages/Search"; export async function generateMetadata({ searchParams }) { const query = searchParams.q; diff --git a/app/store/auth.js b/app/store/auth.ts similarity index 63% rename from app/store/auth.js rename to app/store/auth.ts index a46c655..45c4364 100644 --- a/app/store/auth.js +++ b/app/store/auth.ts @@ -1,13 +1,22 @@ "use client"; import { create } from "zustand"; -import { getJWT, setJWT, removeJWT, fetchDataViaGet } from "@/app/api/utils"; +import { getJWT, removeJWT, fetchDataViaGet } from "#/api/utils"; -export const useUserStore = create((set, get) => ({ +interface userState { + isAuth: boolean + user: Object | null + token: string | null + login: (user: Object, token: string) => void + logout: () => void + checkAuth: () => void +} + +export const useUserStore = create((set, get) => ({ isAuth: false, user: null, token: null, - login: (user, token) => { + login: (user: Object, token: string) => { set({ isAuth: true, user: user, token: token }); }, logout: () => { diff --git a/app/store/preferences.js b/app/store/preferences.ts similarity index 100% rename from app/store/preferences.js rename to app/store/preferences.ts diff --git a/jsconfig.json b/jsconfig.json deleted file mode 100644 index 2a2e4b3..0000000 --- a/jsconfig.json +++ /dev/null @@ -1,7 +0,0 @@ -{ - "compilerOptions": { - "paths": { - "@/*": ["./*"] - } - } -} diff --git a/package-lock.json b/package-lock.json index 07a7fef..555b662 100644 --- a/package-lock.json +++ b/package-lock.json @@ -23,6 +23,8 @@ "@iconify-json/mdi": "^1.1.67", "@iconify-json/twemoji": "^1.1.15", "@iconify/tailwind": "^1.1.1", + "@types/node": "20.14.12", + "@types/react": "18.3.3", "eslint": "^8", "eslint-config-next": "14.2.5", "postcss": "^8", @@ -604,6 +606,31 @@ "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", "dev": true }, + "node_modules/@types/node": { + "version": "20.14.12", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.12.tgz", + "integrity": "sha512-r7wNXakLeSsGT0H1AU863vS2wa5wBOK4bWMjZz2wj+8nBx+m5PeIn0k8AloSLpRuiwdRQZwarZqHE4FNArPuJQ==", + "dev": true, + "dependencies": { + "undici-types": "~5.26.4" + } + }, + "node_modules/@types/prop-types": { + "version": "15.7.12", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.12.tgz", + "integrity": "sha512-5zvhXYtRNRluoE/jAp4GVsSduVUzNWKkOZrCDBWYtE7biZywwdC2AcEzg+cSMLFRfVgeAFqpfNabiPjxFddV1Q==", + "devOptional": true + }, + "node_modules/@types/react": { + "version": "18.3.3", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.3.tgz", + "integrity": "sha512-hti/R0pS0q1/xx+TsI73XIqk26eBsISZ2R0wUijXIngRK9R/e7Xw/cXVxQK7R5JjW+SV4zGcn5hXjudkN/pLIw==", + "devOptional": true, + "dependencies": { + "@types/prop-types": "*", + "csstype": "^3.0.2" + } + }, "node_modules/@types/resolve": { "version": "1.20.2", "resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-1.20.2.tgz", @@ -1276,6 +1303,12 @@ "node": ">=4" } }, + "node_modules/csstype": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", + "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", + "devOptional": true + }, "node_modules/damerau-levenshtein": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.8.tgz", @@ -4875,6 +4908,12 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/undici-types": { + "version": "5.26.5", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", + "dev": true + }, "node_modules/uri-js": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", diff --git a/package.json b/package.json index fa0caa7..e7d4423 100644 --- a/package.json +++ b/package.json @@ -24,6 +24,8 @@ "@iconify-json/mdi": "^1.1.67", "@iconify-json/twemoji": "^1.1.15", "@iconify/tailwind": "^1.1.1", + "@types/node": "20.14.12", + "@types/react": "18.3.3", "eslint": "^8", "eslint-config-next": "14.2.5", "postcss": "^8", diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..69ca516 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,42 @@ +{ + "compilerOptions": { + "lib": [ + "dom", + "dom.iterable", + "esnext" + ], + "allowJs": true, + "skipLibCheck": true, + "strict": false, + "noEmit": true, + "incremental": true, + "module": "esnext", + "esModuleInterop": true, + "moduleResolution": "node", + "resolveJsonModule": true, + "isolatedModules": true, + "jsx": "preserve", + "plugins": [ + { + "name": "next" + } + ], + "baseUrl": "app", + "paths": { + "#/components/*": ["components/*"], + "#/api/*": ["api/*"], + "#/store/*": ["store/*"], + "#/hooks/*": ["hooks/*"], + "#/pages/*": ["pages/*"], + }, + }, + "include": [ + "next-env.d.ts", + ".next/types/**/*.ts", + "**/*.ts", + "**/*.tsx" + ], + "exclude": [ + "node_modules" + ] +}