frontend: add user profile page

This commit is contained in:
Kentai Radiquum 2024-04-24 14:40:53 +05:00
parent 32a4da1a31
commit 320a36d27e
Signed by: Radiquum
GPG key ID: 858E8EE696525EED
5 changed files with 194 additions and 3 deletions

View file

@ -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" }}>

View 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>
) : (
""
)}
</>
);
};

View file

@ -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}>

View 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>
)}
</>
);
}

View file

@ -1,5 +1,6 @@
/** @type {import('next').NextConfig} */
const nextConfig = {
reactStrictMode: false,
images: {
remotePatterns: [
{