diff --git a/app/api/utils.ts b/app/api/utils.ts index e464009..9fa41c7 100644 --- a/app/api/utils.ts +++ b/app/api/utils.ts @@ -119,12 +119,11 @@ const months = [ "дек.", ]; -export function unixToDate(unix: number, type: string = "short") { +export function unixToDate( + unix: number, + type: "full" | "dayMonth" | "dayMonthYear" +) { const date = new Date(unix * 1000); - if (type === "short") - return ( - date.getDate() + " " + months[date.getMonth()] + " " + date.getFullYear() - ); if (type === "full") return ( date.getDate() + @@ -137,6 +136,12 @@ export function unixToDate(unix: number, type: string = "short") { ":" + date.getMinutes() ); + if (type === "dayMonth") + return date.getDate() + " " + months[date.getMonth()]; + if (type === "dayMonthYear") + return ( + date.getDate() + " " + months[date.getMonth()] + " " + date.getFullYear() + ); } export const getSeasonFromUnix = (unix: number) => { diff --git a/app/components/Profile/Profile.PrivacyBanner.tsx b/app/components/Profile/Profile.PrivacyBanner.tsx index 8164992..eb6e807 100644 --- a/app/components/Profile/Profile.PrivacyBanner.tsx +++ b/app/components/Profile/Profile.PrivacyBanner.tsx @@ -1,5 +1,3 @@ -import { unixToDate } from "#/api/utils"; - export const ProfilePrivacyBanner = (props: { is_privacy: boolean }) => { return ( <> diff --git a/app/components/Profile/Profile.WatchDynamic.tsx b/app/components/Profile/Profile.WatchDynamic.tsx new file mode 100644 index 0000000..66c4b0a --- /dev/null +++ b/app/components/Profile/Profile.WatchDynamic.tsx @@ -0,0 +1,102 @@ +import { Card } from "flowbite-react"; +import ApexCharts, { ApexOptions } from "apexcharts"; +import { useEffect } from "react"; +import { unixToDate } from "#/api/utils"; +export const ProfileWatchDynamic = (props: { watchDynamic: Array }) => { + const lastTenDays = props.watchDynamic.slice( + Math.max(props.watchDynamic.length - 10, 0) + ); + const data = { + ids: lastTenDays.map((item) => item.id), + counts: lastTenDays.map((item) => item.count), + timestamps: lastTenDays.map((item) => + unixToDate(item.timestamp, "dayMonth") + ), + }; + + const options: ApexOptions = { + chart: { + height: "100%", + type: "area", + fontFamily: "Inter, sans-serif", + dropShadow: { + enabled: false, + }, + toolbar: { + show: false, + }, + }, + tooltip: { + enabled: true, + x: { + show: false, + }, + }, + fill: { + type: "gradient", + gradient: { + opacityFrom: 0.55, + opacityTo: 0, + shade: "#1C64F2", + gradientToColors: ["#1C64F2"], + }, + }, + dataLabels: { + enabled: false, + }, + stroke: { + width: 6, + }, + grid: { + show: true, + strokeDashArray: 4, + padding: { + left: 2, + right: 2, + top: 0, + }, + }, + series: [ + { + name: "Серий", + data: data.counts, + color: "#1C64F2", + }, + ], + xaxis: { + categories: data.timestamps, + labels: { + show: false, + }, + axisBorder: { + show: false, + }, + axisTicks: { + show: false, + }, + }, + yaxis: { + show: false, + }, + }; + + useEffect(() => { + if ( + document.getElementById("area-chart") && + typeof ApexCharts !== "undefined" + ) { + const chart = new ApexCharts( + document.getElementById("area-chart"), + options + ); + chart.render(); + } + }, []); + + return ( + +

Динамика просмотра серий

+
+
+ ); +}; diff --git a/app/components/Profile/ProfileBannedBanner.tsx b/app/components/Profile/ProfileBannedBanner.tsx index 6781f19..c7625f7 100644 --- a/app/components/Profile/ProfileBannedBanner.tsx +++ b/app/components/Profile/ProfileBannedBanner.tsx @@ -15,7 +15,7 @@ export const ProfileBannedBanner = (props: { {props.is_perm_banned ? "Пользователь был заблокирован администрацией навсегда" : `Пользователь был заблокирован администрацией до - ${unixToDate(props.ban_expires)}`} + ${unixToDate(props.ban_expires, "full")}`}

{props.ban_reason} diff --git a/app/components/ReleaseInfo/ReleaseInfo.Info.tsx b/app/components/ReleaseInfo/ReleaseInfo.Info.tsx index 5856351..0f6e93e 100644 --- a/app/components/ReleaseInfo/ReleaseInfo.Info.tsx +++ b/app/components/ReleaseInfo/ReleaseInfo.Info.tsx @@ -133,7 +133,7 @@ export const ReleaseInfoInfo = (props: { {props.aired_on_date != 0 ? ( - unixToDate(props.aired_on_date) + unixToDate(props.aired_on_date, "full") ) : props.year ? ( <> {props.season && props.season != 0 diff --git a/app/pages/Profile copy.tsx b/app/pages/Profile copy.tsx deleted file mode 100644 index def785a..0000000 --- a/app/pages/Profile copy.tsx +++ /dev/null @@ -1,312 +0,0 @@ -"use client"; -import { useUserStore } from "#/store/auth"; -import { useEffect, useState } from "react"; -import { Spinner } from "../components/Spinner/Spinner"; -import { Avatar, Card, Button, Table } from "flowbite-react"; -import { Chip } from "../components/Chip/Chip"; -import { fetchDataViaGet, unixToDate, minutesToTime } from "../api/utils"; -import { ReleaseCourusel } from "#/components/ReleaseCourusel/ReleaseCourusel"; -import { ENDPOINTS } from "#/api/config"; - -export const ProfilePage = (props: any) => { - const authUser = useUserStore((state) => state); - const [user, setUser] = useState(null); - const [isMyProfile, setIsMyProfile] = useState(false); - - useEffect(() => { - async function _getData() { - let url = `${ENDPOINTS.user.profile}/${props.id}`; - if (authUser.token) { - url += `?token=${authUser.token}`; - } - const data = await fetchDataViaGet(url); - setUser(data.profile); - setIsMyProfile(data.is_my_profile); - } - _getData(); - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [authUser]); - - if (!user) { - return ( -

- -
- ); - } - - const hasSocials = - user.vk_page != "" || - user.tg_page != "" || - user.tt_page != "" || - user.inst_page != "" || - user.discord_page != "" || - false; - const socials = [ - { - name: "vk", - nickname: user.vk_page, - icon: "fa6-brands--vk", - urlPrefix: "https://vk.com", - }, - { - name: "telegram", - nickname: user.tg_page, - icon: "fa6-brands--telegram", - urlPrefix: "https://t.me", - }, - { - name: "discord", - nickname: user.discord_page, - icon: "fa6-brands--discord", - }, - { - name: "tiktok", - nickname: user.tt_page, - icon: "fa6-brands--tiktok", - urlPrefix: "https://tiktok.com", - }, - { - name: "instagram", - nickname: user.inst_page, - icon: "fa6-brands--instagram", - urlPrefix: "https://instagram.com", - }, - ]; - - const hasChips = user.is_verified || user.is_blocked || isMyProfile; - - return ( -
- {(user.is_banned || user.is_perm_banned) && ( -
-
-

- {user.is_perm_banned - ? "Пользователь был заблокирован администрацией навсегда" - : `Пользователь был заблокирован администрацией до - ${unixToDate(user.ban_expires)}`} -

-

- {user.ban_reason} -

-
-
- )} - -
- - {hasChips && ( -
- {isMyProfile && ( - - )} - {user.is_blocked && ( - - )} - {user.is_verified && ( - - )} -
- )} - -
-
{user.login}
-

- {user.status} -

-
-
- {hasSocials && ( -
- {socials - .filter((social: any) => { - if (social.nickname == "") { - return false; - } - return true; - }) - .map((social: any) => { - if (social.name == "discord" && social.nickname != "") - return ( - - ); - return ( - - ); - })} -
- )} -
-
- -

Активность

- - - - - Регистрация - - - {unixToDate(user.register_date)} - - - - - Был(а) в сети - - - {unixToDate(user.last_activity_time)} - - - - - Комментарий - - - {user.comment_count} - - - - - друзей - - - {user.friend_count} - - - - - видео - - - {user.video_count} - - - - - коллекций - - - {user.collection_count} - - - -
-
- -

Статистика

- - - - - - Просмотрено серий - - - {user.watched_episode_count} - - - - - - Время просмотра - - - {minutesToTime(user.watched_time) || - "Нет просмотренных серий."} - - - - - - {minutesToTime(user.watched_time) || - "Нет просмотренных серий."} - - - - - - Смотрю - - - {user.watching_count} - - - - - - В Планах - - - {user.plan_count} - - - - - - Просмотрено - - - {user.completed_count} - - - - - - Отложено - - - {user.hold_on_count} - - - - - - Брошено - - - {user.dropped_count} - - - -
-
-
-
- {user.history.length > 0 && ( -
- -
- )} -
- ); -}; diff --git a/app/pages/Profile.tsx b/app/pages/Profile.tsx index 8da4749..d78fc8c 100644 --- a/app/pages/Profile.tsx +++ b/app/pages/Profile.tsx @@ -10,6 +10,7 @@ import { ProfileBannedBanner } from "#/components/Profile/ProfileBannedBanner"; import { ProfilePrivacyBanner } from "#/components/Profile/Profile.PrivacyBanner"; import { ProfileActivity } from "#/components/Profile/Profile.Activity"; import { ProfileStats } from "#/components/Profile/Profile.Stats"; +import { ProfileWatchDynamic } from "#/components/Profile/Profile.WatchDynamic"; export const ProfilePage = (props: any) => { const authUser = useUserStore((state) => state); @@ -135,19 +136,24 @@ export const ProfilePage = (props: any) => { )} {!user.is_stats_hidden && ( -
- -
+ <> +
+ +
+
+ +
+ )}