mirror of
https://github.com/Radiquum/AniX.git
synced 2025-04-06 00:04:39 +00:00
frontend: add user profile page
This commit is contained in:
parent
32a4da1a31
commit
320a36d27e
5 changed files with 194 additions and 3 deletions
|
@ -3,10 +3,13 @@ import Image from "next/image";
|
|||
|
||||
export const ReleaseCard = (props) => {
|
||||
return (
|
||||
<Link href={`/release/${props.id}`} className="s3">
|
||||
<Link
|
||||
href={`/release/${props.id}`}
|
||||
className={props.className ? props.className : "s3"}
|
||||
>
|
||||
<article
|
||||
className="no-padding round fill"
|
||||
style={{ width: 284, height: 508 }}
|
||||
style={{ width: 284, height: props.height ? props.height : 508 }}
|
||||
>
|
||||
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||
<div style={{ aspectRatio: "1/1" }}>
|
||||
|
|
170
frontend/app/components/UserProfile/UserProfile.jsx
Normal file
170
frontend/app/components/UserProfile/UserProfile.jsx
Normal file
|
@ -0,0 +1,170 @@
|
|||
import Image from "next/image";
|
||||
import { ReleaseCard } from "../ReleaseCard/ReleaseCard";
|
||||
import { getData } from "@/app/api/api-utils";
|
||||
import { endpoints } from "@/app/api/config";
|
||||
import { useEffect, useState } from "react";
|
||||
|
||||
function getNoun(number, one, two, five) {
|
||||
let n = Math.abs(number);
|
||||
n %= 100;
|
||||
if (n >= 5 && n <= 20) {
|
||||
return five;
|
||||
}
|
||||
n %= 10;
|
||||
if (n === 1) {
|
||||
return one;
|
||||
}
|
||||
if (n >= 2 && n <= 4) {
|
||||
return two;
|
||||
}
|
||||
return five;
|
||||
}
|
||||
|
||||
function convertMinutes(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} ${getNoun(d, "день", "дня", "дней")}, ` : "";
|
||||
var hDisplay = h > 0 ? `${h} ${getNoun(h, "час", "часа", "часов")}, ` : "";
|
||||
var mDisplay = m > 0 ? `${m} ${getNoun(m, "минута", "минуты", "минут")}` : "";
|
||||
return dDisplay + hDisplay + mDisplay;
|
||||
}
|
||||
|
||||
export const UserProfile = (props) => {
|
||||
const [lastWatched, setLastWatched] = useState();
|
||||
|
||||
useEffect(() => {
|
||||
async function _getData() {
|
||||
const data = await getData(
|
||||
`${endpoints.user.profile}/${props.profile.id}`,
|
||||
);
|
||||
setLastWatched(data.profile.history);
|
||||
}
|
||||
_getData();
|
||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<>
|
||||
<div className="grid">
|
||||
<div className="s4">
|
||||
<article className="primary-container">
|
||||
<i className="extra">account_circle</i>
|
||||
<div className="row">
|
||||
<Image
|
||||
className="circle"
|
||||
src={props.profile.avatar}
|
||||
alt="Ваш профиль"
|
||||
width="512"
|
||||
height="512"
|
||||
style={{ blockSize: "7rem", inlineSize: "7rem" }}
|
||||
/>
|
||||
<div className="max">
|
||||
<h5>{props.profile.login}</h5>
|
||||
<p>{props.profile.status}</p>
|
||||
</div>
|
||||
</div>
|
||||
</article>
|
||||
</div>
|
||||
<div className="s4">
|
||||
<article className="secondary-container">
|
||||
<i className="extra">avg_pace</i>
|
||||
<h5>Активность</h5>
|
||||
<div className="row">
|
||||
<div className="center-align">
|
||||
<h5 className="small">{props.profile.comment_count}</h5>
|
||||
<p>коммент</p>
|
||||
</div>
|
||||
<div className="center-align">
|
||||
<h5 className="small">{props.profile.video_count}</h5>
|
||||
<p>видео</p>
|
||||
</div>
|
||||
<div className="center-align">
|
||||
<h5 className="small">{props.profile.collection_count}</h5>
|
||||
<p>коллекций</p>
|
||||
</div>
|
||||
<div className="center-align">
|
||||
<h5 className="small">{props.profile.friend_count}</h5>
|
||||
<p>друзей</p>
|
||||
</div>
|
||||
</div>
|
||||
</article>
|
||||
</div>
|
||||
<div className="s4">
|
||||
<article className="tertiary-container">
|
||||
<i className="extra">show_chart</i>
|
||||
<div>
|
||||
<h5>Статистика</h5>
|
||||
<div>
|
||||
<p className="small">
|
||||
Просмотрено серий:{" "}
|
||||
<span class="bold">
|
||||
{props.profile.watched_episode_count}
|
||||
</span>
|
||||
</p>
|
||||
<p className="small">
|
||||
Время просмотра:{" "}
|
||||
<span class="bold">
|
||||
{convertMinutes(props.profile.watched_time)}
|
||||
</span>
|
||||
</p>
|
||||
</div>
|
||||
<div>
|
||||
<div>
|
||||
<p>
|
||||
<i>play_arrow</i> Смотрю:{" "}
|
||||
<span className="bold">{props.profile.watching_count}</span>
|
||||
</p>
|
||||
<p>
|
||||
<i>note_stack</i> В Планах:{" "}
|
||||
<span className="bold">{props.profile.plan_count}</span>
|
||||
</p>
|
||||
<p>
|
||||
<i>done</i> Просмотрено:{" "}
|
||||
<span className="bold">
|
||||
{props.profile.completed_count}
|
||||
</span>
|
||||
</p>
|
||||
<p>
|
||||
<i>schedule</i> Отложено:{" "}
|
||||
<span className="bold">{props.profile.hold_on_count}</span>
|
||||
</p>
|
||||
<p>
|
||||
<i>backspace</i> Брошено:{" "}
|
||||
<span className="bold">{props.profile.dropped_count}</span>
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</article>
|
||||
</div>
|
||||
</div>
|
||||
{lastWatched ? (
|
||||
<article className="grid">
|
||||
<div className="row s12">
|
||||
<i>tab_recent</i>
|
||||
<h5>Недавно просмотрено</h5>
|
||||
</div>
|
||||
<nav className="s12 scroll">
|
||||
{lastWatched.map((item) => {
|
||||
return (
|
||||
<ReleaseCard
|
||||
className={"s2"}
|
||||
key={item.id}
|
||||
id={item.id}
|
||||
title={item.title_ru}
|
||||
poster={item.image}
|
||||
description={item.last_view_episode.name}
|
||||
height={440}
|
||||
/>
|
||||
);
|
||||
})}
|
||||
</nav>
|
||||
</article>
|
||||
) : (
|
||||
""
|
||||
)}
|
||||
</>
|
||||
);
|
||||
};
|
|
@ -38,7 +38,7 @@ export default function LoginPage() {
|
|||
}, [userStore.user]);
|
||||
|
||||
return (
|
||||
<div className="absolute padding tertiary center middle round">
|
||||
<div className="absolute padding secondary-container center middle round">
|
||||
<i className="extra">login</i>
|
||||
<h5>Вход в аккаунт anixart.</h5>
|
||||
<form onSubmit={handleSubmit}>
|
||||
|
|
17
frontend/app/profile/page.js
Normal file
17
frontend/app/profile/page.js
Normal file
|
@ -0,0 +1,17 @@
|
|||
"use client";
|
||||
import { useUserStore } from "@/app/store/user-store";
|
||||
import { UserProfile } from "@/app/components/UserProfile/UserProfile";
|
||||
|
||||
export default function Profile() {
|
||||
const userStore = useUserStore();
|
||||
|
||||
return (
|
||||
<>
|
||||
{userStore.user ? (
|
||||
<UserProfile profile={userStore.user.profile} />
|
||||
) : (
|
||||
<progress></progress>
|
||||
)}
|
||||
</>
|
||||
);
|
||||
}
|
|
@ -1,5 +1,6 @@
|
|||
/** @type {import('next').NextConfig} */
|
||||
const nextConfig = {
|
||||
reactStrictMode: false,
|
||||
images: {
|
||||
remotePatterns: [
|
||||
{
|
||||
|
|
Loading…
Add table
Reference in a new issue