diff --git a/.cz.yaml b/.cz.yaml deleted file mode 100644 index baecdf3..0000000 --- a/.cz.yaml +++ /dev/null @@ -1,8 +0,0 @@ ---- -commitizen: - major_version_zero: true - name: cz_conventional_commits - tag_format: $version - update_changelog_on_bump: true - version: 0.11.0 - version_scheme: semver diff --git a/.dockerignore b/.dockerignore deleted file mode 100644 index 3321c42..0000000 --- a/.dockerignore +++ /dev/null @@ -1,56 +0,0 @@ -# Python -__pycache__ -venv -.mypy_cache - -# VSCode -.VSCode -*.code-workspace - -# NextJS -## dependencies -node_modules -.pnp -.pnp.js -.yarn/install-state.gz - -## testing -coverage - -## next.js -.next -out - -## production -build - -## misc -.DS_Store -*.pem - -## debug -npm-debug.log* -yarn-debug.log* -yarn-error.log* - -## local env files -.env*.local - -## vercel -.vercel - -## typescript -*.tsbuildinfo -next-env.d.ts - -# traefik -traefik - -# OtherFiles -CHANGELOG.md -LICENSE -README.md -TODO.md -.cz.yaml -.pre-commit-config.yaml -docs diff --git a/.env b/.env deleted file mode 100644 index 4c53f0b..0000000 --- a/.env +++ /dev/null @@ -1,5 +0,0 @@ -EMAIL = your_email@example.com # EMAIL for certificate resolver. -DOMAIN = 127.0.0.1 # DOMAIN on which is app is hosted. - -HTTP_PORT = 80 -HTTPS_PORT = 443 diff --git a/.gitignore b/.gitignore index 1199e56..c295f68 100644 --- a/.gitignore +++ b/.gitignore @@ -49,3 +49,5 @@ next-env.d.ts # traefik traefik/traefik + +old/ \ No newline at end of file diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml deleted file mode 100644 index dd01dfd..0000000 --- a/.pre-commit-config.yaml +++ /dev/null @@ -1,67 +0,0 @@ -# See https://pre-commit.com for more information -# See https://pre-commit.com/hooks.html for more hooks -repos: - - repo: https://github.com/pre-commit/mirrors-mypy - rev: v1.9.0 - hooks: - - id: mypy - args: [--no-strict-optional, --ignore-missing-imports] - additional_dependencies: [tokenize-rt==3.2.0, types-requests] - files: ^(backend/) - - - repo: https://github.com/psf/black - rev: 24.4.0 - hooks: - - id: black - args: [--safe] - files: ^(backend/) - - - repo: https://github.com/pre-commit/pre-commit-hooks - rev: v4.5.0 - hooks: - - id: trailing-whitespace - - id: end-of-file-fixer - - id: check-yaml - - id: check-added-large-files - - id: debug-statements - language_version: python3 - - - repo: https://github.com/PyCQA/flake8 - rev: 7.0.0 - hooks: - - id: flake8 - language_version: python3 - args: [--ignore=E501] - files: ^(backend/) - - - repo: https://github.com/asottile/reorder_python_imports - rev: v3.12.0 - hooks: - - id: reorder-python-imports - args: [--py39-plus] - files: ^(backend/) - - - repo: https://github.com/asottile/pyupgrade - rev: v3.15.2 - hooks: - - id: pyupgrade - args: [--py39-plus] - files: ^(backend/) - - - repo: local - hooks: - - id: next-lint - name: next-lint - entry: bash -c 'cd ./frontend && npm run lint' - language: system - files: ^(frontend/) - stages: - - manual - - - repo: https://github.com/pre-commit/mirrors-prettier - rev: v3.1.0 # Use the sha or tag you want to point at - hooks: - - id: prettier - additional_dependencies: - - prettier@3.1.0 - files: ^(frontend/) diff --git a/.spaceignore b/.spaceignore deleted file mode 100644 index 38b511b..0000000 --- a/.spaceignore +++ /dev/null @@ -1,57 +0,0 @@ -# Python -__pycache__ -venv -.mypy_cache - -# VSCode -.VSCode -*.code-workspace - -# NextJS -## dependencies -node_modules -.pnp -.pnp.js -.yarn/install-state.gz - -## testing -coverage - -## next.js -.next -out - -## production -build - -## misc -.DS_Store -*.pem - -## debug -npm-debug.log* -yarn-debug.log* -yarn-error.log* - -## local env files -.env*.local - -## vercel -.vercel - -## typescript -*.tsbuildinfo -next-env.d.ts - -# traefik -traefik - -# OtherFiles -CHANGELOG.md -docker* -LICENSE -README.md -TODO.md -.cz.yaml -.pre-commit-config.yaml -docs diff --git a/CHANGELOG.md b/CHANGELOG.md deleted file mode 100644 index b53f16a..0000000 --- a/CHANGELOG.md +++ /dev/null @@ -1,116 +0,0 @@ -## 0.11.0 (2024-05-15) - -### Feat - -- **deployment**: allow deploy to vercel - -### Fix - -- **vercel/backend**: fix vercel backend deploy -- **vercel/backend**: fix vercel.json headers schema -- **vercel/backend**: fix missing CORS headers -- **frontend**: fix API_URL environment variable being undefined - -## 0.10.0 (2024-05-13) - -### Feat - -- **deploy**: allow deploying on deta.space - -## 0.9.0 (2024-05-11) - -### Feat - -- **ReleaseOverview**: add ability to change view modes for releases pages - -### Refactor - -- **Design/Content**: redising of the main content view - -## 0.8.0 (2024-05-11) - -### Feat - -- **Design/Navbar**: move logout button to tooltip on user avatar hover -- **navigation**: add a copy current url button to navbar for ease of sharing -- **Design**: navigation rail redisign - -### Fix - -- **ReleasePlayer**: fixed fullscreen view - -## 0.7.0 (2024-05-01) - -### Feat - -- **frontend/release**: add a favorite button to release page -- **frontend**: add user favorites page -- **backend**: add a user favorites route - -### Fix - -- **frontend/release**: fix voiceover selection overflow in player - -### Refactor - -- **backend-&-frontend**: change bookmarks api urls to /api/bookmarks/{path} from /api/favorites/{path} - -## 0.6.0 (2024-05-01) - -### Feat - -- **frontend/release**: add ability to change boormarks list on releases page for authorized users - -## 0.5.0 (2024-04-30) - -### Feat - -- **frontend/releases**: add release info and related releases cards - -### Fix - -- **frontend/releases**: fix null episode name if API returning null in episode.name - -## 0.4.0 (2024-04-30) - -### Feat - -- **frontend**: add a release page - -### Refactor - -- **frontend/release**: add a loading circle for release player when fetching episodes data -- **frontend/release**: move release player in to the component -- **frontend/release-page**: less layout shift on player - -## 0.3.0 (2024-04-30) - -### Feat - -- **frontend**: add a release page -- **docker**: add docker compose file for development environment - -## 0.2.1 (2024-04-29) - -### Fix - -- **frontend**: fix text overflow on releases cards -- **frontend**: fix overflow of releases cards on pages with overview - -## 0.2.0 (2024-04-29) - -### Feat - -- **next-page-loading**: disable load more button when length of fetched page is less than 25 on all pages - -### Refactor - -- **history-page**: use ReleaseOverview component on search page - -## 0.1.0 (2024-04-29) - -### Feat - -- Update TODO.md -- update TODO.md -- disable favorites nav button diff --git a/LICENSE b/LICENSE deleted file mode 100644 index 99454a8..0000000 --- a/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -MIT License - -Copyright (c) 2024 Kentai Radiquum - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. diff --git a/README.md b/README.md deleted file mode 100644 index 77d9dc2..0000000 --- a/README.md +++ /dev/null @@ -1,184 +0,0 @@ -# AniX - Unofficial Web Client for Anixart - -AniX is an unofficial web client for the Android application Anixart. It allows you to access and manage your Anixart account from a web browser on your desktop or laptop computer. - -[Changelog](./CHANGELOG.md) - -[Backend Readme](./backend/README.md) - -## Screenshots - -**User profile page** - -
- -| Dark | Light | -| ------------------------------------------------------------- | --------------------------------------------------------------- | -| ![user profile page image dark mode](./docs/profile_dark.jpg) | ![user profile page image light mode](./docs/profile_light.jpg) | - -
- -**Pages with releases lists** - -
- -| Dark | Light | -| ---------------------------------------------------------------------- | ------------------------------------------------------------------------ | -| ![index page image dark mode cards view ](./docs/index_cards_dark.jpg) | ![index page image light mode cards view ](./docs/index_cards_light.jpg) | -| ![index page image dark mode rows view ](./docs/index_rows_dark.jpg) | ![index page image light mode rows view ](./docs/index_rows_light.jpg) | - -
- -**Release page and Player** - -
- -| Dark | Light | -| ------------------------------------------------------ | -------------------------------------------------------- | -| ![player page image dark mode](./docs/player_dark.jpg) | ![player page image light mode](./docs/player_light.jpg) | - -
- -## Features - -1. Dynamic theming (light/dark mode, app colors) -2. login/logout to AnixArt account -3. sync watch history, bookmarks and favorites -4. row or card layout for releases view - -## Project Structure - -This project consists of two main parts: - -1. **Backend (Python with FastAPI):** This handles communication with the Anixart API and provides data to the frontend. -2. **Frontend (Next.js):** This is the user interface that you interact with in your web browser. It fetches data from the backend and displays it in a user-friendly way. - -## Disclaimer - -Please note that AniX is an unofficial project and is not affiliated with the developers of Anixart. It is recommended to use the official Anixart app for the most up-to-date features and functionality. - -## Development - -To maintain code formatting it's recommended to use pre-commit hooks - -0. Install global pre-commit `pip install pre-commit` -1. Install pre-commit hooks via `pre-commit install` inside repository folder - -To maintain readable git commit messages it's recommended to use [commitizen](https://commitizen-tools.github.io/commitizen/) - -Feel free to make changes and experiment with the project. - -## Getting Started - -### Docker - -#### Docker Requirements - -- docker (>=26) -- docker compose (>= 2.27) - -#### Running development project via Docker - -Execute `docker compose -f docker-compose.dev.yml up` command in the root of the folder. - -You can access the AniX web client in your browser at `http://127.0.0.1`. And API docs at `http://127.0.0.1/api/v1/docs` - -#### Running development project via Docker with realtime changes - -To run development environment with realtime changes via docker you can execute the docker watch command, it will watch and copy local changes in to the running container. you will need to use the dev compose file and docker files. - -Execute the `docker compose -f docker-compose.dev.yml watch` command to set up the docker development environment with realtime changes. - -To access the docker logs you can use `docker compose -f docker-compose.dev.yml logs -f` command. - -## Deployment - -### first steps - -1. clone this repository via `git clone https://github.com/Radiquum/AniX.git` -2. cd into the repository folder. -3. deploy app with below instructions. - -### Docker Deployment - -#### Docker Prerequisites - -1. **Docker:** Make sure Docker is installed on your system. You can download it from [https://www.docker.com/](https://www.docker.com/). -2. **Docker Compose:** Docker Compose should also be installed. It's usually included with Docker installations. -3. **Domain Name:** (optional) You need a registered domain name (e.g., `example.com`) and have it pointing to your server's IP address. -4. **SSL Certificate:** (optional) For HTTPS, you'll need an SSL certificate. Let's Encrypt is a free and popular option. - -#### Docker steps - -1. Edit the .env file -2. Edit the docker-compose.yml file to match your needs. -3. run `docker compose up -d` to build and run production images. -4. your app will be available at `http(s)://{DOMAIN}`. -5. run `docker compose down` to stop the containers. - -*notes*: - -- application is deployed as http by default, to enable https you need to edit docker compose and .env files. - - if https is enabled, traefik will automatically issue HTTPS certificate from lets encrypt and redirect to HTTPS. - -- you can also run `docker compose up --build` to build and run images without re-deploying the containers. - -### Deta Space - -1. **Install the Space CLI:** - - ```bash - curl -fsSL https://deta.space/assets/space-cli.sh | sh - ``` - -2. **Login to Deta Space:** - - ```bash - space login - ``` - - (You'll need to grab an access token from your Space dashboard) - -3. **Create a new Space Project:** - - ```bash - space new - ``` - - (This will guide you through setting up your project) - -4. **Push your app to Deta Space:** - - ```bash - space push --runner-version experimental - ``` - - (This will build and deploy your application) - -You can find more details in the Deta Space documentation: [Deta Space Docs](https://docs.deta.space/) - -### Vercel - -*note* you don't need to complete the first steps for vercel deploy - -1. [Deploy the backend](./backend/README.md#vercel) and copy the production url, including the prefix -2. create a new project on vercel and set the root directory to frontend -3. set API_URL env variable to copied url -4. click deploy - -#### Available environment variables - -- API_URL - sets the api url to use for frontend to connect to the backend - -#### One-Click Deploy - -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/clone?repository-url=https://github.com/Radiquum/AniX/tree/main/frontend) - -*note*: this will create a new repository. - -*note*: don't forget to set API_URL to backend url. - -## Contributing - -We welcome contributions to this project! If you have any bug fixes, improvements, or new features, please feel free to create a pull request. diff --git a/Spacefile b/Spacefile deleted file mode 100644 index 652a117..0000000 --- a/Spacefile +++ /dev/null @@ -1,14 +0,0 @@ -v: 0 -# icon: ./icon.png -micros: - - name: anix-api - src: ./backend/ - engine: python3.11 - path: api - run: uvicorn main:app --root-path /api - dev: uvicorn main:app --reload - - - name: anix-app - src: ./frontend/ - engine: next - primary: true diff --git a/TODO.md b/TODO.md deleted file mode 100644 index a5ba96e..0000000 --- a/TODO.md +++ /dev/null @@ -1,14 +0,0 @@ -# TODO - -- [ ] Add docker deployment - -## Релизы - -- [ ] Авто-ген ссылка с именем -- [ ] Комментарии? -- [ ] ... - -## Идеи - -- уведомления -- метатеги diff --git a/backend/.python-version b/backend/.python-version deleted file mode 100644 index 21af950..0000000 --- a/backend/.python-version +++ /dev/null @@ -1 +0,0 @@ -3.9.13 diff --git a/backend/README.md b/backend/README.md deleted file mode 100644 index ae3fc4d..0000000 --- a/backend/README.md +++ /dev/null @@ -1,28 +0,0 @@ -# Anix Backend - -This is the backend for AniX, an unofficial WEB client for the Android app Anixart. - -It is using FastAPI and server as a proxy between app API and Web Client. - -## Deployment - -### Environment variables - -- API_PREFIX - sets the api prefix - -### Docker - - [Refer the docker deployment from root README file](../README.md#docker-deployment) - -## Vercel - -1. fork the repository -2. create a new project on vercel and set the root directory to backend -3. (optionally) set API_PREFIX env variable -4. click deploy - -### One-Click Deploy - -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/clone?repository-url=https://github.com/Radiquum/AniX/tree/main/backend) - -*note*: this will create a new repository diff --git a/backend/main.py b/backend/main.py deleted file mode 100644 index b33bf54..0000000 --- a/backend/main.py +++ /dev/null @@ -1,77 +0,0 @@ -import os - -import uvicorn -from fastapi import FastAPI -from fastapi.middleware.cors import CORSMiddleware -from modules import proxy -from modules.pages import bookmarks -from modules.pages import favorites -from modules.pages import index -from modules.pages import search -from modules.release import release -from modules.user import auth -from modules.user import profile - -TAGS = [ - { - "name": "Index", - "description": "Main page API requests", - }, - { - "name": "Profile", - "description": "Profile API requests", - }, - { - "name": "Releases", - "description": "Releases API requests", - }, - { - "name": "Bookmarks", - "description": "Bookmarks API requests", - }, - { - "name": "Favorites", - "description": "Favorites API requests", - }, - { - "name": "Search", - "description": "Search API requests", - }, -] - -PREFIX = "/v1" - -if os.getenv("API_PREFIX"): - PREFIX = os.getenv("API_PREFIX") - -app = FastAPI( - openapi_tags=TAGS, - title="AniX API", - description="unofficial API proxy for Anixart android application.", - openapi_url=f"{PREFIX}/openapi.json", - docs_url=f"{PREFIX}/docs", - redoc_url=None, -) - -app.add_middleware( - CORSMiddleware, - allow_origins=["*"], - allow_credentials=True, - allow_methods=["GET", "POST"], - allow_headers=["*"], -) - -app.include_router(profile.router, prefix=f"{PREFIX}/profile", tags=["Profile"]) -app.include_router(auth.router, prefix=f"{PREFIX}/auth", tags=["Profile"]) - -app.include_router(release.router, prefix=f"{PREFIX}/release", tags=["Releases"]) - -app.include_router(index.router, prefix=f"{PREFIX}/index", tags=["Index"]) -app.include_router(bookmarks.router, prefix=f"{PREFIX}/bookmarks", tags=["Bookmarks"]) -app.include_router(favorites.router, prefix=f"{PREFIX}/favorites", tags=["Favorites"]) -app.include_router(search.router, prefix=f"{PREFIX}/search", tags=["Search"]) - -app.include_router(proxy.router, prefix=f"{PREFIX}/proxy") - -if __name__ == "__main__": - uvicorn.run("main:app", host="0.0.0.0", port=8000) diff --git a/backend/modules/__init__.py b/backend/modules/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/modules/pages/__init__.py b/backend/modules/pages/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/modules/pages/bookmarks.py b/backend/modules/pages/bookmarks.py deleted file mode 100644 index cfbdde2..0000000 --- a/backend/modules/pages/bookmarks.py +++ /dev/null @@ -1,75 +0,0 @@ -from fastapi import APIRouter -from fastapi import Request -from modules.proxy import apiRequest -from modules.proxy import ENDPOINTS - -router = APIRouter() - - -@router.get("/history", summary="Get user watch history") -async def GetUserHistory(request: Request, token: str, page: int = 0): - return await apiRequest( - request, ENDPOINTS["user"]["history"], page, query=f"?token={token}" - ) - - -@router.get("/watching", summary="Get user watch list") -async def GetUserWatching(request: Request, token: str, page: int = 0): - return await apiRequest( - request, ENDPOINTS["user"]["watching"], page, query=f"?token={token}" - ) - - -@router.get("/planned", summary="Get user planned list") -async def GetUserPlanned(request: Request, token: str, page: int = 0): - return await apiRequest( - request, ENDPOINTS["user"]["planned"], page, query=f"?token={token}" - ) - - -@router.get("/watched", summary="Get user watched list") -async def GetUserWatched(request: Request, token: str, page: int = 0): - return await apiRequest( - request, ENDPOINTS["user"]["watched"], page, query=f"?token={token}" - ) - - -@router.get("/delayed", summary="Get user delayed list") -async def GetUserDelayed(request: Request, token: str, page: int = 0): - return await apiRequest( - request, ENDPOINTS["user"]["delayed"], page, query=f"?token={token}" - ) - - -@router.get("/abandoned", summary="Get user abandoned list") -async def GetUserAbandoned(request: Request, token: str, page: int = 0): - return await apiRequest( - request, ENDPOINTS["user"]["abandoned"], page, query=f"?token={token}" - ) - - -@router.get( - "/list/{bookmark_list_id}/{release_id}/add", summary="Add release to bookmarks list" -) -async def addReleaseToBookmarks( - request: Request, release_id: int, bookmark_list_id: int, token: str -): - return await apiRequest( - request, - f"{ENDPOINTS['profile']}/list/add/{bookmark_list_id}/{release_id}", - query=f"?token={token}", - ) - - -@router.get( - "/list/{bookmark_list_id}/{release_id}/delete", - summary="Remove release from bookmarks list", -) -async def deleteReleaseFromBookmarks( - request: Request, release_id: int, bookmark_list_id: int, token: str -): - return await apiRequest( - request, - f"{ENDPOINTS['profile']}/list/delete/{bookmark_list_id}/{release_id}", - query=f"?token={token}", - ) diff --git a/backend/modules/pages/favorites.py b/backend/modules/pages/favorites.py deleted file mode 100644 index e90e852..0000000 --- a/backend/modules/pages/favorites.py +++ /dev/null @@ -1,33 +0,0 @@ -from fastapi import APIRouter -from fastapi import Request -from modules.proxy import apiRequest -from modules.proxy import ENDPOINTS - -router = APIRouter() - - -@router.get("", summary="Get user favorites list") -async def GetUserFavorites(request: Request, token: str, page: int = 0): - return await apiRequest( - request, ENDPOINTS["user"]["favorite"], f"all/{page}", query=f"?token={token}" - ) - - -@router.get("/list/{release_id}/add", summary="Add release to user favorites") -async def addReleaseToFavorites(request: Request, release_id: int, token: str): - return await apiRequest( - request, - ENDPOINTS["user"]["favorite"], - f"add/{release_id}", - query=f"?token={token}", - ) - - -@router.get("/list/{release_id}/delete", summary="Remove release from user favorites") -async def deleteReleaseFromFavorites(request: Request, release_id: int, token: str): - return await apiRequest( - request, - ENDPOINTS["user"]["favorite"], - f"delete/{release_id}", - query=f"?token={token}", - ) diff --git a/backend/modules/pages/index.py b/backend/modules/pages/index.py deleted file mode 100644 index 378294a..0000000 --- a/backend/modules/pages/index.py +++ /dev/null @@ -1,56 +0,0 @@ -import json -from typing import Union - -from fastapi import APIRouter -from fastapi import Request -from modules.proxy import apiRequest -from modules.proxy import ENDPOINTS - -router = APIRouter() - - -async def GetMainPageFilter( - request: Request, page: int = 0, status_id: Union[None, int] = None -): - data = json.dumps( - { - "country": None, - "season": None, - "sort": 0, - "studio": None, - "age_ratings": [], - "category_id": None, - "end_year": None, - "episode_duration_from": None, - "episode_duration_to": None, - "episodes_from": None, - "episodes_to": None, - "genres": [], - "profile_list_exclusions": [], - "start_year": None, - "status_id": status_id, - "types": [], - "is_genres_exclude_mode_enabled": False, - } - ) - return await apiRequest(request, ENDPOINTS["filter"], page, data=data) - - -@router.get("/last", summary="Get new releases") -async def GetMainPage(request: Request, page: int = 0): - return await GetMainPageFilter(request, page, None) - - -@router.get("/ongoing", summary="Get ongoing releases") -async def GetOngoingPage(request: Request, page: int = 0): - return await GetMainPageFilter(request, page, 2) - - -@router.get("/announce", summary="Get announced releases") -async def GetAnnouncePage(request: Request, page: int = 0): - return await GetMainPageFilter(request, page, 3) - - -@router.get("/finished", summary="Get finished releases") -async def GetFinishedPage(request: Request, page: int = 0): - return await GetMainPageFilter(request, page, 1) diff --git a/backend/modules/pages/search.py b/backend/modules/pages/search.py deleted file mode 100644 index 21dad6c..0000000 --- a/backend/modules/pages/search.py +++ /dev/null @@ -1,14 +0,0 @@ -import json - -from fastapi import APIRouter -from fastapi import Request -from modules.proxy import apiRequest -from modules.proxy import ENDPOINTS - -router = APIRouter() - - -@router.get("", summary="Search for a release") -async def Search(request: Request, query: str, page: int = 0): - data = json.dumps({"query": query, "searchBy": 0}) - return await apiRequest(request, ENDPOINTS["search"], page, data=data) diff --git a/backend/modules/proxy.py b/backend/modules/proxy.py deleted file mode 100644 index bccf61e..0000000 --- a/backend/modules/proxy.py +++ /dev/null @@ -1,86 +0,0 @@ -from typing import TypedDict -from typing import Union - -import requests -from fastapi import APIRouter -from fastapi import Request -from fastapi import Response - - -class Endpoints(TypedDict): - release: dict[str, str] - profile: str - filter: str - auth: str - user: dict[str, str] - search: str - statistic: dict[str, str] - - -API_URL = "https://api.anixart.tv" -ENDPOINTS: Endpoints = { - "release": { - "info": f"{API_URL}/release", - "episode": f"{API_URL}/episode", - }, - "profile": f"{API_URL}/profile", - "filter": f"{API_URL}/filter", - "auth": f"{API_URL}/auth/signIn", - "user": { - "history": f"{API_URL}/history", - "watching": f"{API_URL}/profile/list/all/1", - "planned": f"{API_URL}/profile/list/all/2", - "watched": f"{API_URL}/profile/list/all/3", - "delayed": f"{API_URL}/profile/list/all/4", - "abandoned": f"{API_URL}/profile/list/all/5", - "favorite": f"{API_URL}/favorite", - }, - "search": f"{API_URL}/search/releases", - "statistic": { - "addHistory": f"{API_URL}/history/add", - "markWatched": f"{API_URL}/episode/watch", - }, -} -USER_AGENT = "AnixartApp/8.2.1-23121216 (Android 11; SDK 30; arm64-v8a;)" - - -async def apiRequest( - request: Request = None, - endpoint: Union[str, Endpoints] = "", - path: Union[str, int] = "", - query: str = "", - data: Union[None, str, dict] = None, -): - - headers = { - "User-Agent": USER_AGENT, - "Content-Type": "application/json; charset=UTF-8", - } - - if data is not None or request.method == "POST": - r = requests.post( - # noqa: E501 - f"{endpoint}/{path}{query}", - headers=headers, - data=data, - ) - else: - r = requests.get(f"{endpoint}/{path}{query}", headers=headers) - - if r.status_code != 200: - return {"error": r.text} - return r.json() - - -router = APIRouter() - - -@router.get( - "/image", - responses={200: {"content": {"image/jpg": {}, "image/png": {}}}}, - response_class=Response, -) -async def imageProxy(url: str): - type = url.split(".")[-1] - response: bytes = requests.get(url).content - return Response(content=response, media_type=f"image/{type}") diff --git a/backend/modules/release/__init__.py b/backend/modules/release/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/modules/release/release.py b/backend/modules/release/release.py deleted file mode 100644 index 3f76457..0000000 --- a/backend/modules/release/release.py +++ /dev/null @@ -1,75 +0,0 @@ -import requests -from fastapi import APIRouter -from fastapi import Request -from modules.proxy import apiRequest -from modules.proxy import ENDPOINTS -from modules.proxy import USER_AGENT - -router = APIRouter() - - -@router.get("/{release_id}", summary="Get release info") -async def GetReleaseById(request: Request, release_id: str, token: str = ""): - return await apiRequest( - request, ENDPOINTS["release"]["info"], release_id, query=f"?token={token}" - ) - - -@router.get("/{release_id}/voiceover", summary="Get release voiceover info") -async def GetReleaseVoiceover(request: Request, release_id: str): - return await apiRequest(request, ENDPOINTS["release"]["episode"], release_id) - - -@router.get( - "/{release_id}/{voiceover_id}", - summary="Get available players for selected voiceover of a release", -) -async def GetReleaseVoiceoverPlayer( - request: Request, release_id: str, voiceover_id: str -): - return await apiRequest( - request, ENDPOINTS["release"]["episode"], f"{release_id}/{voiceover_id}" - ) - - -@router.get( - "/{release_id}/{voiceover_id}/{source_id}", - summary="Get available episodes for selected voiceover and a player of a release", -) -async def GetReleaseEpisodes( - request: Request, - release_id: str, - voiceover_id: str, - source_id: str, - token: str = "", -): - return await apiRequest( - request, - ENDPOINTS["release"]["episode"], - f"{release_id}/{voiceover_id}/{source_id}", - query=f"?token={token}", - ) - - -@router.get( - "/{release_id}/{source_id}/{episode}/saveToHistory", - summary="mark episode of a selected voiceover as watched and save it to watch history", -) -async def MarkEpisodeAsWatched( - request: Request, release_id: str, source_id: str, episode: str, token: str -): - headers = { - "User-Agent": USER_AGENT, - "Content-Type": "application/json; charset=UTF-8", - } - - requests.get( - f"{ENDPOINTS['statistic']['markWatched']}/{release_id}/{source_id}/{episode}?token={token}", - headers=headers, - ) - requests.get( - f"{ENDPOINTS['statistic']['addHistory']}/{release_id}/{source_id}/{episode}?token={token}", - headers=headers, - ) - - return {"success"} diff --git a/backend/modules/user/__init__.py b/backend/modules/user/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/modules/user/auth.py b/backend/modules/user/auth.py deleted file mode 100644 index 1f70eca..0000000 --- a/backend/modules/user/auth.py +++ /dev/null @@ -1,48 +0,0 @@ -import requests -from fastapi import APIRouter -from modules.proxy import ENDPOINTS -from modules.proxy import USER_AGENT -from pydantic import BaseModel - - -class User(BaseModel): - email: str - password: str - - -router = APIRouter() - - -@router.post("", summary="logging in") -async def userSignIn( - user: User, - short: bool = False, -): - headers = { - "User-Agent": USER_AGENT, - "Sign": "9aa5c7af74e8cd70c86f7f9587bde23d", - "Content-Type": "application/x-www-form-urlencoded", - } - r = requests.post( - # noqa: E501 - f"{ENDPOINTS['auth']}", - headers=headers, - data={"login": user.email, "password": user.password}, - ) - if r.status_code != 200: - return {"error": r.text} - res = r.json() - if short is True: - return { - "code": res["code"], - "profile": { - "id": res["profile"]["id"], - "login": res["profile"]["login"], - "avatar": res["profile"]["avatar"], - }, - "profileToken": { - "id": res["profileToken"]["id"], - "token": res["profileToken"]["token"], - }, - } - return res diff --git a/backend/modules/user/profile.py b/backend/modules/user/profile.py deleted file mode 100644 index 7829f30..0000000 --- a/backend/modules/user/profile.py +++ /dev/null @@ -1,30 +0,0 @@ -from typing import Union - -from fastapi import APIRouter -from fastapi import Request -from modules.proxy import apiRequest -from modules.proxy import ENDPOINTS - -router = APIRouter() - - -@router.get("/{user_id}", summary="Get user profile by user ID") -async def getUserById( - request: Request, user_id: str, short: bool = False, token: Union[None, str] = None -): - query = "" - if token: - query = f"?token={token}" - - res = await apiRequest(request, ENDPOINTS["profile"], user_id, query=query) - if short is False: - return res - return { - "code": res["code"], - "profile": { - "id": res["profile"]["id"], - "login": res["profile"]["login"], - "avatar": res["profile"]["avatar"], - }, - "is_my_profile": res["is_my_profile"], - } diff --git a/backend/requirements.txt b/backend/requirements.txt deleted file mode 100644 index 2c640fa..0000000 --- a/backend/requirements.txt +++ /dev/null @@ -1,4 +0,0 @@ -requests == 2.31.0 -fastAPI == 0.110.1 -uvicorn[standard] -python-multipart diff --git a/backend/vercel.json b/backend/vercel.json deleted file mode 100644 index 65295de..0000000 --- a/backend/vercel.json +++ /dev/null @@ -1,21 +0,0 @@ -{ - "$schema": "https://openapi.vercel.sh/vercel.json", - "builds": [ - { - "src": "main.py", - "use": "@vercel/python" - } - ], - "routes": [ - { - "src": "/(.*)", - "dest": "main.py", - "headers": { - "Access-Control-Allow-Credentials": "true", - "Access-Control-Allow-Origin": "*", - "Access-Control-Allow-Methods": "GET,POST", - "Access-Control-Allow-Headers": "X-CSRF-Token, X-Requested-With, Accept, Accept-Version, Content-Length, Content-MD5, Content-Type, Date, X-Api-Version" - } - } - ] -} diff --git a/docker-compose.dev.yml b/docker-compose.dev.yml deleted file mode 100644 index 1b8a49d..0000000 --- a/docker-compose.dev.yml +++ /dev/null @@ -1,57 +0,0 @@ -services: - frontend: - container_name: "AniX-frontend" - build: - context: ./frontend - dockerfile: ../docker/frontend.dev.Dockerfile - labels: - - "traefik.enable=true" - - "traefik.http.routers.anix-frontend.rule=Host(`127.0.0.1`)" - - "traefik.http.routers.anix-frontend.entrypoints=web" - expose: - - 3000 - develop: - watch: - - action: sync - path: ./frontend - target: /app - ignore: - - node_modules/ - - .next/ - - action: rebuild - path: ./frontend/package.json - backend: - container_name: "AniX-backend" - build: - context: ./backend - dockerfile: ../docker/backend.dev.Dockerfile - labels: - - "traefik.enable=true" - - "traefik.http.routers.anix-backend.rule=Host(`127.0.0.1`) && PathPrefix(`/api/`)" - - traefik.http.middlewares.anix-backend_stripprefix.stripprefix.prefixes=/api - - "traefik.http.routers.anix-backend.middlewares=anix-backend_stripprefix@docker" - - "traefik.http.routers.anix-backend.entrypoints=web" - expose: - - 8000 - develop: - watch: - - action: sync - path: ./backend - target: /app - - action: rebuild - path: ./backend/requirements.txt - traefik: - image: "traefik:v3.0" - container_name: "AniX-traefik" - command: - #- "--log.level=DEBUG" - - "--api.dashboard=true" - - "--api.insecure=true" - - "--providers.docker=true" - - "--providers.docker.exposedbydefault=false" - - "--entryPoints.web.address=:80" - ports: - - "80:80" - - "8080:8080" - volumes: - - "/var/run/docker.sock:/var/run/docker.sock:ro" diff --git a/docker-compose.yml b/docker-compose.yml deleted file mode 100644 index 25b4f92..0000000 --- a/docker-compose.yml +++ /dev/null @@ -1,77 +0,0 @@ -services: - frontend: - env_file: .env - container_name: "AniX-frontend" - build: - context: ./frontend - dockerfile: ../docker/frontend.Dockerfile - labels: - # --- ${DOMAIN} will be replaced with DOMAIN from .env --- - - "traefik.enable=true" - - "traefik.http.routers.anix-frontend.rule=Host(`${DOMAIN}`)" - - # --- !COMMENT THIS FOR HTTPS! --- - - "traefik.http.routers.anix-frontend.entrypoints=web" - - # --- !UNCOMMENT THIS FOR HTTPS! --- - #- "traefik.http.routers.anix-frontend.entrypoints=websecure" - #- "traefik.http.routers.anix-frontend.tls=true" - #- "traefik.http.routers.user-api-backend.tls.certresolver=letsencrypt" - expose: - - 3000 - backend: - env_file: .env - container_name: "AniX-backend" - build: - context: ./backend - dockerfile: ../docker/backend.Dockerfile - labels: - # --- ${DOMAIN} will be replaced with DOMAIN from .env --- - - "traefik.enable=true" - - "traefik.http.routers.anix-backend.rule=Host(`${DOMAIN}`) && PathPrefix(`/api`)" - - traefik.http.middlewares.anix-backend_stripprefix.stripprefix.prefixes=/api - - "traefik.http.routers.anix-backend.middlewares=anix-backend_stripprefix@docker" - - # --- !COMMENT THIS FOR HTTPS! --- - - "traefik.http.routers.anix-backend.entrypoints=web" - - # --- !UNCOMMENT THIS FOR HTTPS! --- - #- "traefik.http.routers.anix-backend.tls=true" - #- "traefik.http.routers.anix-backend.entrypoints=websecure" - #- "traefik.http.routers.user-api-backend.tls.certresolver=letsencrypt" - expose: - - 8000 - traefik: - env_file: .env - image: "traefik:v3.0" - container_name: "AniX-traefik" - command: - - "--api.dashboard=false" - - "--api.insecure=false" - - "--providers.docker=true" - - "--providers.docker.exposedbydefault=false" - - # --- !COMMENT THIS FOR HTTPS! --- - - "--entryPoints.web.address=:${HTTP_PORT:-80}" - - # --- !UNCOMMENT THIS FOR HTTPS! --- - #- "--entrypoints.websecure.address=:${HTTPS_PORT:-443}" - #- "--entrypoints.web.http.redirections.entryPoint.to=websecure" - #- "--entrypoints.web.http.redirections.entryPoint.scheme=https" - #- "--certificatesresolvers.letsencrypt.acme.tlschallenge=true" - #- "--certificatesresolvers.letsencrypt.acme.email=${EMAIL}" - #- "--certificatesresolvers.letsencrypt.acme.storage=/letsencrypt/acme.json" - ports: - # --- !COMMENT THIS FOR HTTPS! --- - - ${HTTP_PORT:-80}:${HTTP_PORT:-80} - # --- !UNCOMMENT THIS FOR HTTPS! --- - #- ${HTTPS_PORT:-443}:${HTTPS_PORT:-443} - volumes: - - "/var/run/docker.sock:/var/run/docker.sock:ro" - depends_on: - - frontend - - backend - - # --- !UNCOMMENT THIS FOR HTTPS! --- - # acme.json should be created on host instance! - #- "./acme.json:/letsencrypt/acme.json" diff --git a/docker/backend.Dockerfile b/docker/backend.Dockerfile deleted file mode 100644 index 0d943ab..0000000 --- a/docker/backend.Dockerfile +++ /dev/null @@ -1,18 +0,0 @@ -# The first instruction is what image we want to base our container on -# We Use an official Python runtime as a parent image -FROM python:3.9 - -# The enviroment variable ensures that the python output is set straight -# to the terminal with out buffering it first -ENV PYTHONUNBUFFERED 1 - -# Set the working directory to /app -WORKDIR /app - -# Copy the current directory contents into the container at /app -COPY . . - -# Install any needed packages specified in requirements.txt -RUN pip install -r requirements.txt - -CMD ["uvicorn", "main:app", "--host", "0.0.0.0", "--root-path", "/api"] diff --git a/docker/backend.dev.Dockerfile b/docker/backend.dev.Dockerfile deleted file mode 100644 index df60500..0000000 --- a/docker/backend.dev.Dockerfile +++ /dev/null @@ -1,18 +0,0 @@ -# The first instruction is what image we want to base our container on -# We Use an official Python runtime as a parent image -FROM python:3.9 - -# The enviroment variable ensures that the python output is set straight -# to the terminal with out buffering it first -ENV PYTHONUNBUFFERED 1 - -# Set the working directory to /app -WORKDIR /app - -# Copy the current directory contents into the container at /app -COPY . . - -# Install any needed packages specified in requirements.txt -RUN pip install -r requirements.txt - -CMD ["uvicorn", "main:app", "--reload", "--host", "0.0.0.0", "--root-path", "/api"] diff --git a/docker/frontend.Dockerfile b/docker/frontend.Dockerfile deleted file mode 100644 index 1ce2346..0000000 --- a/docker/frontend.Dockerfile +++ /dev/null @@ -1,22 +0,0 @@ -FROM node:21-alpine - -RUN mkdir /app -RUN mkdir /prepare - -WORKDIR /prepare - -ADD package.json . - -RUN npm update -g npm --loglevel verbose -COPY . . -# RUN npm install --loglevel verbose -RUN npm ci --omit=dev --no-audit --maxsockets 1 --loglevel verbose -RUN npm run docker --loglevel verbose - -RUN cp -a ./build/. /app/ - -WORKDIR /app -RUN rm -rf /prepare -RUN npm i sharp --loglevel verbose - -CMD ["node", "./server.js"] diff --git a/docker/frontend.dev.Dockerfile b/docker/frontend.dev.Dockerfile deleted file mode 100644 index 3ed60bc..0000000 --- a/docker/frontend.dev.Dockerfile +++ /dev/null @@ -1,12 +0,0 @@ -FROM node:21-alpine - -WORKDIR /app - -ADD package.json . - -RUN npm update -g npm --loglevel verbose -COPY . . -RUN npm install --loglevel verbose -# RUN npm ci --no-audit --maxsockets 1 - -CMD ["npm", "run", "dev"] diff --git a/docs/index_cards_dark.jpg b/docs/index_cards_dark.jpg deleted file mode 100644 index 86a8573..0000000 Binary files a/docs/index_cards_dark.jpg and /dev/null differ diff --git a/docs/index_cards_light.jpg b/docs/index_cards_light.jpg deleted file mode 100644 index 0cc964d..0000000 Binary files a/docs/index_cards_light.jpg and /dev/null differ diff --git a/docs/index_rows_dark.jpg b/docs/index_rows_dark.jpg deleted file mode 100644 index 66ec2e6..0000000 Binary files a/docs/index_rows_dark.jpg and /dev/null differ diff --git a/docs/index_rows_light.jpg b/docs/index_rows_light.jpg deleted file mode 100644 index 6f90240..0000000 Binary files a/docs/index_rows_light.jpg and /dev/null differ diff --git a/docs/player_dark.jpg b/docs/player_dark.jpg deleted file mode 100644 index d05f0a7..0000000 Binary files a/docs/player_dark.jpg and /dev/null differ diff --git a/docs/player_light.jpg b/docs/player_light.jpg deleted file mode 100644 index 96b1ab9..0000000 Binary files a/docs/player_light.jpg and /dev/null differ diff --git a/docs/profile_dark.jpg b/docs/profile_dark.jpg deleted file mode 100644 index b7a66e5..0000000 Binary files a/docs/profile_dark.jpg and /dev/null differ diff --git a/docs/profile_light.jpg b/docs/profile_light.jpg deleted file mode 100644 index 885e2cc..0000000 Binary files a/docs/profile_light.jpg and /dev/null differ diff --git a/frontend/.eslintrc.json b/frontend/.eslintrc.json deleted file mode 100644 index 957cd15..0000000 --- a/frontend/.eslintrc.json +++ /dev/null @@ -1,3 +0,0 @@ -{ - "extends": ["next/core-web-vitals"] -} diff --git a/frontend/.gitignore b/frontend/.gitignore deleted file mode 100644 index fd3dbb5..0000000 --- a/frontend/.gitignore +++ /dev/null @@ -1,36 +0,0 @@ -# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. - -# dependencies -/node_modules -/.pnp -.pnp.js -.yarn/install-state.gz - -# testing -/coverage - -# next.js -/.next/ -/out/ - -# production -/build - -# misc -.DS_Store -*.pem - -# debug -npm-debug.log* -yarn-debug.log* -yarn-error.log* - -# local env files -.env*.local - -# vercel -.vercel - -# typescript -*.tsbuildinfo -next-env.d.ts diff --git a/frontend/app/App.jsx b/frontend/app/App.jsx deleted file mode 100644 index 41e9ba0..0000000 --- a/frontend/app/App.jsx +++ /dev/null @@ -1,100 +0,0 @@ -"use client"; - -import "beercss"; -import "material-dynamic-colors"; -import { NavigationRail } from "@/app/components/NavigationRail/NavigationRail"; -import { useEffect, useState } from "react"; -import { ColorPicker } from "@/app/components/ColorPicker/ColorPicker"; -import { useUserStore } from "./store/user-store"; -import Settings from "./components/Settings/Settings"; - -function setMode(mode) { - localStorage.setItem("mode", mode); -} -function getMode() { - return localStorage.getItem("mode"); -} - -function setTheme(theme) { - localStorage.setItem("theme", theme); -} -function getTheme() { - return localStorage.getItem("theme"); -} - -export const App = (props) => { - const [colorPicker, setColorPicker] = useState(false); - const [settingsPopup, setSettingsPopup] = useState(false); - const userStore = useUserStore(); - - const theme = async (from) => { - setTheme(from); - await ui("theme", from); - }; - - const mode = () => { - let newMode = ui("mode") == "dark" ? "light" : "dark"; - setMode(newMode); - ui("mode", getMode()); - }; - - useEffect(() => { - const mode = getMode(); - const theme = getTheme(); - if (mode != ui("mode")) { - ui("mode", getMode()); - } - if (theme != ui("theme")) { - ui("theme", theme); - } - }, []); - - useEffect(() => { - userStore.checkAuth(); - // eslint-disable-next-line react-hooks/exhaustive-deps - }, []); - - return ( - -
-
- - {colorPicker && ( - - )} - {settingsPopup && ( - - )} -
-
-
- {props.children} -
-
-
- - ); -}; diff --git a/frontend/app/api/api-utils.js b/frontend/app/api/api-utils.js deleted file mode 100644 index 65f9d14..0000000 --- a/frontend/app/api/api-utils.js +++ /dev/null @@ -1,57 +0,0 @@ -export const isResponseOk = (response) => { - return !(response instanceof Error); -}; - -export const getData = async (url) => { - try { - const response = await fetch(url); - if (response.status !== 200) { - throw new Error("Ошибка получения данных"); - } - return await response.json(); - } catch (error) { - return error; - } -}; - -export const authorize = async (url, data) => { - try { - const response = await fetch(url, { - method: "POST", - headers: { "Content-Type": "application/json" }, - body: JSON.stringify(data), - }); - if (response.status !== 200) { - throw new Error("Ошибка получения данных"); - } - return await response.json(); - } catch (error) { - return error; - } -}; - -export const getMe = async (url, jwt) => { - try { - const response = await fetch(`${url}?token=${jwt}`, { - method: "GET", - }); - if (response.status !== 200) { - throw new Error("Ошибка получения данных"); - } - return await response.json(); - } catch (error) { - return error; - } -}; - -export function setJWT(jwt, user_id) { - const data = { jwt: jwt, user_id: user_id }; - localStorage.setItem("data", JSON.stringify(data)); -} -export function getJWT() { - const data = localStorage.getItem("data"); - return JSON.parse(data); -} -export function removeJWT() { - localStorage.removeItem("data"); -} diff --git a/frontend/app/api/config.js b/frontend/app/api/config.js deleted file mode 100644 index 7eb8ada..0000000 --- a/frontend/app/api/config.js +++ /dev/null @@ -1,30 +0,0 @@ -export let API_URL = "/api/v1"; - -if (process.env.API_URL) { - API_URL = process.env.API_URL; -} - -export const endpoints = { - index: { - last: `${API_URL}/index/last`, - ongoing: `${API_URL}/index/ongoing`, - announce: `${API_URL}/index/announce`, - finished: `${API_URL}/index/finished`, - }, - search: `${API_URL}/search`, - user: { - profile: `${API_URL}/profile`, - auth: `${API_URL}/auth`, - bookmarks: { - list: `${API_URL}/bookmarks/list`, - history: `${API_URL}/bookmarks/history`, - watching: `${API_URL}/bookmarks/watching`, - planned: `${API_URL}/bookmarks/planned`, - watched: `${API_URL}/bookmarks/watched`, - delayed: `${API_URL}/bookmarks/delayed`, - abandoned: `${API_URL}/bookmarks/abandoned`, - }, - favorites: `${API_URL}/favorites`, - }, - release: `${API_URL}/release`, -}; diff --git a/frontend/app/bookmarks/page.js b/frontend/app/bookmarks/page.js deleted file mode 100644 index 2f49b1e..0000000 --- a/frontend/app/bookmarks/page.js +++ /dev/null @@ -1,11 +0,0 @@ -"use client"; - -import { useUserStore } from "@/app/store/user-store"; -import { LogInNeeded } from "@/app/components/LogInNeeded/LogInNeeded"; -import BookmarksPage from "../components/Pages/BookmarksPage"; - -export default function Bookmarks() { - const userStore = useUserStore(); - - return <>{!userStore.isAuth ? : }; -} diff --git a/frontend/app/components/CardList/CardList.jsx b/frontend/app/components/CardList/CardList.jsx deleted file mode 100644 index 2e51ca1..0000000 --- a/frontend/app/components/CardList/CardList.jsx +++ /dev/null @@ -1,29 +0,0 @@ -import { ReleaseCard } from "@/app/components/ReleaseCard/ReleaseCard"; -import { ReleaseList } from "@/app/components/ReleaseList/ReleaseList"; - -export const CardList = (props) => { - return props.data.map((item) => { - if (props.view == "grid") { - return ( - - ); - } - if (props.view == "list") { - return ( - - ); - } - }); -}; diff --git a/frontend/app/components/ColorPicker/ColorPicker.jsx b/frontend/app/components/ColorPicker/ColorPicker.jsx deleted file mode 100644 index 360bde5..0000000 --- a/frontend/app/components/ColorPicker/ColorPicker.jsx +++ /dev/null @@ -1,77 +0,0 @@ -"use client"; - -import { useState } from "react"; -import Styles from "./ColorPicker.module.css"; -import { useUserStore } from "@/app/store/user-store"; -import { API_URL } from "@/app/api/config"; - -export const ColorPicker = (props) => { - const userStore = useUserStore(); - const colors = [ - { hex: "#ffffff", color: "white" }, - { hex: "#e91e63", color: "pink" }, - { hex: "#ff9800", color: "orange" }, - { hex: "#4caf50", color: "green" }, - { hex: "#009688", color: "teal" }, - { hex: "#9c27b0", color: "purple" }, - { hex: "#673ab7", color: "deep-purple" }, - { hex: "#ffeb3b", color: "yellow" }, - { hex: "#ffc8ff", color: Styles["radiquum-pink"] }, - { hex: "#0087c7", color: Styles["fuxigen-blue"] }, - { hex: "#e54040", color: Styles["anixart-red"] }, - ]; - const [mode, setMode] = useState(ui("mode")); - - return ( - -
Выбор темы
-
- {colors.map((item) => { - return ( - - ); - })} - {userStore.user ? ( - - ) : ( - "" - )} -
-
- -
- ); -}; diff --git a/frontend/app/components/ColorPicker/ColorPicker.module.css b/frontend/app/components/ColorPicker/ColorPicker.module.css deleted file mode 100644 index c845d42..0000000 --- a/frontend/app/components/ColorPicker/ColorPicker.module.css +++ /dev/null @@ -1,9 +0,0 @@ -.radiquum-pink { - background-color: #ffc8ff !important; -} -.fuxigen-blue { - background-color: #0087c7 !important; -} -.anixart-red { - background-color: #e54040 !important; -} diff --git a/frontend/app/components/LogInNeeded/LogInNeeded.jsx b/frontend/app/components/LogInNeeded/LogInNeeded.jsx deleted file mode 100644 index 7ca2e32..0000000 --- a/frontend/app/components/LogInNeeded/LogInNeeded.jsx +++ /dev/null @@ -1,11 +0,0 @@ -"use client"; - -export const LogInNeeded = (props) => { - return ( -
- no_accounts -
Требуется авторизация
-

Для доступа к этой вкладке требуется авторизация в аккаунте anixart

-
- ); -}; diff --git a/frontend/app/components/NavigationRail/NavigationRail.jsx b/frontend/app/components/NavigationRail/NavigationRail.jsx deleted file mode 100644 index 2de5726..0000000 --- a/frontend/app/components/NavigationRail/NavigationRail.jsx +++ /dev/null @@ -1,128 +0,0 @@ -"use client"; - -import { usePathname } from "next/navigation"; -import Link from "next/link"; -import Image from "next/image"; -import { useUserStore } from "@/app/store/user-store"; -import { useRouter } from "next/navigation"; -import useCopyToClipboard from "@/app/hooks/useCopyToClipboard"; - -export const NavigationRail = (props) => { - const [isCopied, copyToClipboard] = useCopyToClipboard(); - const pathname = usePathname(); - const userStore = useUserStore(); - const router = useRouter(); - - const items = [ - { - title: "Домашняя", - icon: "home", - path: "/", - }, - { - title: "Поиск", - icon: "search", - path: "/search", - }, - { - title: "Закладки", - icon: "bookmark", - path: "/bookmarks", - }, - - { - title: "Избранное", - icon: "favorite", - path: "/favorites", - }, - - { - title: "История", - icon: "history", - path: "/history", - }, - ]; - - return ( - - ); -}; diff --git a/frontend/app/components/Pages/BookmarksPage.jsx b/frontend/app/components/Pages/BookmarksPage.jsx deleted file mode 100644 index 80069a1..0000000 --- a/frontend/app/components/Pages/BookmarksPage.jsx +++ /dev/null @@ -1,117 +0,0 @@ -"use client"; - -import { getData } from "@/app/api/api-utils"; -import { endpoints } from "@/app/api/config"; -import { useEffect, useState, useCallback } from "react"; -import { usePathname, useRouter } from "next/navigation"; -import { useSearchParams } from "next/navigation"; -import ReleasesOverview from "@/app/components/ReleasesOverview/ReleasesOverview"; -import { useUserStore } from "@/app/store/user-store"; - -export default function BookmarksPage() { - const router = useRouter(); - const pathname = usePathname(); - const userStore = useUserStore(); - - const [list, setList] = useState(); - const [releases, setReleases] = useState(); - const [page, setPage] = useState(0); - - const [isNextPage, setIsNextPage] = useState(true); - - const searchParams = useSearchParams(); - const createQueryString = useCallback( - (name, value) => { - const params = new URLSearchParams(searchParams.toString()); - params.set(name, value); - - return params.toString(); - }, - [searchParams], - ); - - // set list on initial page load - useEffect(() => { - const query = searchParams.get("list"); - if (query) { - setList(query); - } else { - setList("watching"); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, []); - - async function fetchData(list, page = 0) { - if (userStore.token) { - const url = `${endpoints.user.bookmarks[list]}?page=${page}&token=${userStore.token}`; - const data = await getData(url); - - if (data.content.length < 25) { - setIsNextPage(false); - } else { - setIsNextPage(true); - } - - // Handle initial load (page 0) or subsequent pagination - if (page === 0) { - setReleases(data.content); - } else { - setReleases([...releases, ...data.content]); - } - } - } - - useEffect(() => { - if (list) { - router.push(pathname + "?" + createQueryString("list", list)); - setReleases(null); - setPage(0); - fetchData(list); // Call fetchData here - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [list, userStore.token]); - - useEffect(() => { - if (list && releases) { - fetchData(list, page); // Use fetchData for pagination - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [page]); - - const chips = [ - { - title: "Смотрю", - list: "watching", - }, - { - title: "В планах", - list: "planned", - }, - { - title: "Просмотрено", - list: "watched", - }, - { - title: "Отложено", - list: "delayed", - }, - { - title: "Заброшено", - list: "abandoned", - }, - ]; - - return ( - <> - - - ); -} diff --git a/frontend/app/components/Pages/IndexPage.jsx b/frontend/app/components/Pages/IndexPage.jsx deleted file mode 100644 index 86141c9..0000000 --- a/frontend/app/components/Pages/IndexPage.jsx +++ /dev/null @@ -1,107 +0,0 @@ -"use client"; - -import { getData } from "@/app/api/api-utils"; -import { endpoints } from "@/app/api/config"; -import { useEffect, useState, useCallback } from "react"; -import { usePathname, useRouter } from "next/navigation"; -import { useSearchParams } from "next/navigation"; -import ReleasesOverview from "@/app/components/ReleasesOverview/ReleasesOverview"; - -export default function IndexPage() { - const router = useRouter(); - const pathname = usePathname(); - - const [list, setList] = useState(); - const [releases, setReleases] = useState(); - const [page, setPage] = useState(0); - - const [isNextPage, setIsNextPage] = useState(true); - - const searchParams = useSearchParams(); - const createQueryString = useCallback( - (name, value) => { - const params = new URLSearchParams(searchParams.toString()); - params.set(name, value); - - return params.toString(); - }, - [searchParams], - ); - - // set list on initial page load - useEffect(() => { - const query = searchParams.get("list"); - if (query) { - setList(query); - } else { - setList("last"); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, []); - - async function fetchData(list, page = 0) { - const url = `${endpoints.index[list]}?page=${page}`; - const data = await getData(url); - - if (data.content.length < 25) { - setIsNextPage(false); - } else { - setIsNextPage(true); - } - - // Handle initial load (page 0) or subsequent pagination - if (page === 0) { - setReleases(data.content); - } else { - setReleases([...releases, ...data.content]); - } - } - - useEffect(() => { - if (list) { - router.push(pathname + "?" + createQueryString("list", list)); - setReleases(null); - setPage(0); - fetchData(list); // Call fetchData here - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [list]); - - useEffect(() => { - if (list && releases) { - fetchData(list, page); // Use fetchData for pagination - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [page]); - - const chips = [ - { - title: "последнее", - list: "last", - }, - { - title: "в эфире", - list: "ongoing", - }, - { - title: "анонсировано", - list: "announce", - }, - { - title: "завершено", - list: "finished", - }, - ]; - - return ( - - ); -} diff --git a/frontend/app/components/Pages/SearchPage.jsx b/frontend/app/components/Pages/SearchPage.jsx deleted file mode 100644 index e2acc46..0000000 --- a/frontend/app/components/Pages/SearchPage.jsx +++ /dev/null @@ -1,150 +0,0 @@ -"use client"; - -import { getData } from "@/app/api/api-utils"; -import { endpoints } from "@/app/api/config"; -import { useEffect, useState, useCallback } from "react"; -import { usePathname, useRouter } from "next/navigation"; -import ReleasesOverview from "@/app/components/ReleasesOverview/ReleasesOverview"; -import { useSearchParams } from "next/navigation"; - -function saveSearches(search) { - localStorage.setItem("searches", search); -} -function getSearches() { - return localStorage.getItem("searches"); -} - -export default function SearchPage() { - const router = useRouter(); - const pathname = usePathname(); - - const [releases, setReleases] = useState(); - const [page, setPage] = useState(0); - const [query, setQuery] = useState(""); - const [isNextPage, setIsNextPage] = useState(true); - - const [searches, setSearches] = useState(JSON.parse(getSearches())); - - const searchParams = useSearchParams(); - const createQueryString = useCallback( - (name, value) => { - const params = new URLSearchParams(searchParams.toString()); - params.set(name, value); - - return params.toString(); - }, - [searchParams], - ); - - async function fetchData(query, page = 0) { - const url = `${endpoints.search}?query=${query}&page=${page}`; - const data = await getData(url); - - if (data.content.length < 25) { - setIsNextPage(false); - } else { - setIsNextPage(true); - } - - // Handle initial load (page 0) or subsequent pagination - if (page === 0) { - setReleases(data.content); - } else { - setReleases([...releases, ...data.content]); - } - } - - useEffect(() => { - const query = searchParams.get("query"); - if (query) { - setQuery(query); - fetchData(query, 0); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, []); - - useEffect(() => { - if (releases) { - fetchData(query, page); // Use fetchData for pagination - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [page]); - - const handleInput = (e) => { - setQuery(e.target.value); - }; - - const handleSubmit = async (e) => { - e.preventDefault(); - if (query != "") { - router.push(pathname + "?" + createQueryString("query", query)); - setReleases(null); - setPage(0); - fetchData(query); - - // save searches and update search history - if (!searches) { - setSearches([query]); - saveSearches(JSON.stringify([query])); - } else { - console.log(searches); - if (!searches.find((element) => element == query)) { - setSearches([query, ...searches.slice(0, 5)]); - saveSearches(JSON.stringify([query, ...searches.slice(0, 5)])); - } - } - } - }; - - return ( - <> -
-
- search - - - {searches - ? searches.map((item) => { - return ( - { - setQuery(item); - }} - className="row" - > - history -
{item}
-
- ); - }) - : ""} -
-
-
- - {releases ? ( - releases.length > 0 ? ( - - ) : ( -
- search -
Ничего не найдено.
-

Введите другой поисковой запрос.

-
- ) - ) : ( -
- search -
Здесь пока ничего нет.
-

Введите поисковой запрос для начала поиска.

-
- )} - - ); -} diff --git a/frontend/app/components/Release/ReleaseInfo.jsx b/frontend/app/components/Release/ReleaseInfo.jsx deleted file mode 100644 index 6e87592..0000000 --- a/frontend/app/components/Release/ReleaseInfo.jsx +++ /dev/null @@ -1,181 +0,0 @@ -"use client"; - -import { useEffect, useState } from "react"; -import { getData } from "@/app/api/api-utils"; -import { endpoints } from "@/app/api/config"; -import { ReleaseCard } from "../ReleaseCard/ReleaseCard"; -import { useUserStore } from "@/app/store/user-store"; - -export const ReleaseInfo = (props) => { - const userStore = useUserStore(); - const [releaseInfo, setReleaseInfo] = useState(); - const [list, setList] = useState(); - const [isFavorite, setIsFavorite] = useState(false); - const [timer, seTimer] = useState(); - - useEffect(() => { - async function _fetchInfo() { - let url = `${endpoints.release}/${props.id}`; - - if (userStore.token) { - url = `${endpoints.release}/${props.id}?token=${userStore.token}`; - } - - const release = await getData(url); - setReleaseInfo(release); - if (userStore.token) { - setList(release.release.profile_list_status || 0); - setIsFavorite(release.release.is_favorite); - } - } - - // I really think it's not the way it is should be done - // but it works - // FIX: double requests, 1st without token, and second with it. - // now it's only 1 request with or w/o token, if page is reloaded. - if (userStore.token) { - clearTimeout(timer); - } - if (props.id) { - seTimer( - setTimeout(() => { - _fetchInfo(); - }, 1000), - ); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [userStore.token]); - - useEffect(() => { - async function _setList() { - const url = `${endpoints.user.bookmarks.list}/${list}/${props.id}/add?token=${userStore.token}`; - await getData(url); - } - if ( - userStore.token && - releaseInfo && - list != releaseInfo.release.profile_list_status - ) { - _setList(); - releaseInfo.release.profile_list_status = list; - } - }, [userStore.token, list]); - - function _setFav() { - async function __updateFavorite() { - const add_url = `${endpoints.user.favorites}/list/${props.id}/add?token=${userStore.token}`; - const delete_url = `${endpoints.user.favorites}/list/${props.id}/delete?token=${userStore.token}`; - await getData(!isFavorite ? add_url : delete_url); - } - __updateFavorite(); - } - - const lists = [ - { list: 0, name: "Не смотрю" }, - { list: 1, name: "Смотрю" }, - { list: 2, name: "В планах" }, - { list: 3, name: "Просмотрено" }, - { list: 4, name: "Отложено" }, - { list: 5, name: "Брошено" }, - ]; - - return ( - <> - {releaseInfo ? ( - <> -
-
-
- -
-
-
-
-
-
{releaseInfo.release.title_ru}
-
- {releaseInfo.release.title_original} -
-
-
- {userStore.token && list >= 0 && ( - - )} - {userStore.token && releaseInfo && ( - - )} -
-
-

- {releaseInfo.release.country} •{" "} - {releaseInfo.release.status.name} •{" "} - {releaseInfo.release.episodes_released}/ - {releaseInfo.release.episodes_total - ? releaseInfo.release.episodes_total - : "?"} -

-

{releaseInfo.release.description}

-
-
-
-
- {releaseInfo.release.related_releases.length > 0 && ( -
-
- hub -
Связанные релизы
-
- -
- )} - - ) : ( -
- -
- )} - - ); -}; diff --git a/frontend/app/components/Release/ReleasePlayer.jsx b/frontend/app/components/Release/ReleasePlayer.jsx deleted file mode 100644 index e9155d9..0000000 --- a/frontend/app/components/Release/ReleasePlayer.jsx +++ /dev/null @@ -1,172 +0,0 @@ -"use client"; - -import { useEffect, useState } from "react"; -import { getData } from "@/app/api/api-utils"; -import { endpoints } from "@/app/api/config"; -import { useUserStore } from "@/app/store/user-store"; -import { useSettingsStore } from "@/app/store/settings-store"; - -export const ReleasePlayer = (props) => { - const userStore = useUserStore(); - const settingsStore = useSettingsStore(); - - const [voiceoverInfo, setVoiceoverInfo] = useState(); - const [selectedVoiceover, setSelectedVoiceover] = useState(); - const [sourcesInfo, setSourcesInfo] = useState(); - const [selectedSources, setSelectedSources] = useState(); - const [episodeInfo, setEpisodeInfo] = useState(); - const [selectedEpisode, setSelectedEpisode] = useState(); - const [episodeURL, setEpisodeURL] = useState(); - - useEffect(() => { - async function _fetchInfo() { - const voiceover = await getData( - `${endpoints.release}/${props.id}/voiceover`, - ); - setVoiceoverInfo(voiceover); - setSelectedVoiceover(voiceover.types[0].id); - } - if (props.id) { - _fetchInfo(); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, []); - - useEffect(() => { - async function _fetchInfo() { - const sources = await getData( - `${endpoints.release}/${props.id}/${selectedVoiceover}`, - ); - setSourcesInfo(sources); - setSelectedSources(sources.sources[0].id); - } - if (selectedVoiceover) { - _fetchInfo(); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [selectedVoiceover]); - - useEffect(() => { - async function _fetchInfo() { - let url = `${endpoints.release}/${props.id}/${selectedVoiceover}/${selectedSources}`; - if (userStore.token) { - url = `${endpoints.release}/${props.id}/${selectedVoiceover}/${selectedSources}?token=${userStore.token}`; - } - - const episodes = await getData(url); - - setEpisodeInfo(episodes); - setSelectedEpisode(episodes.episodes[0].position); - setEpisodeURL(episodes.episodes[0].url); - } - if (selectedSources) { - _fetchInfo(); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [selectedSources, userStore.token]); - - useEffect(() => { - async function _markAsWatched() { - const url = `${endpoints.release}/${props.id}/${selectedSources}/${selectedEpisode}`; - await getData(`${url}/saveToHistory?token=${userStore.token}`); - } - if (userStore.token && settingsStore.saveToHistory) { - _markAsWatched(); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [selectedEpisode]); - - return ( - <> - {voiceoverInfo && sourcesInfo && episodeInfo ? ( -
-