refactor: start redesign of user page

This commit is contained in:
Kentai Radiquum 2025-04-03 21:07:44 +05:00
parent 68a7117a97
commit 8cf1bb534d
Signed by: Radiquum
GPG key ID: 858E8EE696525EED
5 changed files with 211 additions and 185 deletions

2
.gitignore vendored
View file

@ -59,3 +59,5 @@ videos/*
!videos/*.js
!videos/*.ts
public/_next-video
API-Trace/*

View file

@ -0,0 +1,15 @@
interface UserRoleProps {
name: string;
color: string;
}
export const UserRole = ({ name, color }: UserRoleProps) => {
return (
<div
className={`text-[var(--color)] border border-[var(--color)] rounded-md`}
style={{ "--color": `#${color}` } as React.CSSProperties}
>
<p className="px-1.5 py-0.5">{name}</p>
</div>
);
};

View file

@ -0,0 +1,20 @@
interface UserSocialProps {
icon: string;
url?: string;
nickname: string;
color: string;
}
export const UserSocial = ({ nickname, icon, color }: UserSocialProps) => {
return (
<div
className={`border border-[var(--color)] rounded-md`}
style={{ "--color": `#${color}` } as React.CSSProperties}
>
<div className="flex gap-1 items-center px-1.5 py-1">
<span className={`iconify w-6 h-6 bg-[var(--color)] ${icon}`}></span>
<p>{nickname}</p>
</div>
</div>
);
};

View file

@ -1,143 +1,150 @@
"use client";
import { Avatar, Card, Button } from "flowbite-react";
import Link from "next/link";
import { useRouter } from "next/navigation";
import { Chip } from "../Chip/Chip";
export const ProfileUser = (props: {
isOnline: boolean;
import { Avatar, Card, useThemeMode } from "flowbite-react";
import { UserRole } from "./Profile.Role";
import { UserSocial } from "./Profile.Social";
import Link from "next/link";
interface ProfileUserProps {
avatar: string;
login: string;
status: string;
socials: {
isPrivate: boolean;
hasSocials: boolean;
socials: {
name: string;
nickname: any;
icon: string;
urlPrefix?: string | undefined;
}[];
};
chips: {
hasChips: boolean;
isMyProfile: boolean;
isVerified: boolean;
isSponsor: boolean;
isBlocked: boolean;
roles?: {
rating: number;
roles: {
id: number;
name: string;
color: string;
}[];
isMyProfile: boolean;
isSponsor: boolean;
isBlocked: boolean;
isVerified: boolean;
isOnline: boolean;
socials: {
vk: string;
tg: string;
tt: string;
inst: string;
discord: string;
};
rating: number;
}) => {
const router = useRouter();
}
export const ProfileUser = ({
avatar,
login,
status,
rating,
roles,
isMyProfile,
isVerified,
isOnline,
isSponsor,
isBlocked,
socials,
}: ProfileUserProps) => {
const theme = useThemeMode().mode;
return (
<Card className="h-fit">
{props.chips.hasChips && (
<div className="flex gap-1 overflow-x-auto scrollbar-thin">
{props.chips.isMyProfile && (
<Chip bg_color="bg-blue-500" name="Мой профиль" />
)}
{props.chips.isVerified && (
<Chip bg_color="bg-green-500" name="Верифицирован" />
)}
{props.chips.isSponsor && (
<Chip bg_color="bg-yellow-500" name="Спонсор Anixart" />
)}
{props.chips.isBlocked && (
<Chip bg_color="bg-red-500" name="Заблокирован" />
)}
{props.chips.roles &&
props.chips.roles.length > 0 &&
props.chips.roles.map((role: any) => (
<Chip
key={role.id}
bg_color={`bg-[var(--role-color)]`}
name={role.name}
style={
{
"--role-color": `#${role.color}`,
} as React.CSSProperties
}
/>
<Card>
{(isMyProfile ||
isVerified ||
isSponsor ||
isBlocked ||
roles.length > 0) && (
<div className="flex flex-wrap gap-2">
{isMyProfile && <UserRole name="Мой профиль" color="3f83f8" />}
{isBlocked && <UserRole name="Заблокирован" color="f56565" />}
{isVerified && <UserRole name="Верифицирован" color="0e9f6e" />}
{isSponsor && <UserRole name="Спонсор Anixart" color="ecc94b" />}
{roles.map((role) => (
<UserRole key={role.id} name={role.name} color={role.color} />
))}
</div>
)}
<div className="flex flex-col items-center gap-4 sm:items-start sm:flex-row">
<Avatar
alt=""
img={props.avatar}
img={avatar}
rounded={true}
size={"lg"}
className="relative flex-col items-center justify-center sm:justify-start sm:flex-row"
bordered={true}
color={props.isOnline ? "success" : "light"}
>
<div className="space-y-1 text-2xl font-medium whitespace-pre-wrap dark:text-white">
<div className="text-center sm:text-left">
{props.login}{" "}
color={isOnline ? "success" : "light"}
className="flex-shrink-0"
/>
<div className="flex flex-col gap-2">
<p className="flex items-center gap-2 text-2xl font-semibold">
{login}
<span
className={`border rounded-md px-2 py-1 text-sm ${
props.rating > 0
? "border-green-500 text-green-500"
className={`border rounded-md px-2 py-1 min-w-8 text-sm flex items-center justify-center ${
rating > 0 ?
"border-green-500 text-green-500"
: "border-red-500 text-red-500"
}`}
>
{props.rating}
{rating}
</span>
</div>
<div className="text-sm text-gray-500 whitespace-pre-wrap sm:text-md dark:text-gray-400 ">
{props.status}
</p>
<p className="text-sm whitespace-pre-wrap sm:text-md">{status}</p>
</div>
</div>
</Avatar>
{props.socials.hasSocials && !props.socials.isPrivate && (
<div className="flex items-center gap-1 overflow-x-auto scrollbar-thin">
{props.socials.socials
.filter((social: any) => {
if (social.nickname == "") {
return false;
}
return true;
})
.map((social: any) => {
if (social.name == "discord" && social.nickname != "")
return (
<Button
color="light"
key={social.name}
onClick={() => {
window.navigator.clipboard.writeText(social.nickname);
alert("Скопировано!");
}}
>
<div className="flex items-center justify-center gap-2">
<span
className={`iconify h-4 w-4 sm:h-6 sm:w-6 ${social.icon} dark:fill-white`}
></span>
{social.nickname}
</div>
</Button>
);
return (
{(socials.vk ||
socials.tg ||
socials.discord ||
socials.tt ||
socials.inst) && (
<div className="flex flex-wrap gap-2">
{socials.vk && (
<Link href={`https://vk.com/${socials.vk}`} target="_blank">
<UserSocial
nickname={socials.vk}
icon="fa6-brands--vk"
url={`https://vk.com/${socials.vk}`}
color="4a76a8"
/>
</Link>
)}
{socials.tg && (
<Link href={`https://t.me/${socials.tg}`} target="_blank">
<UserSocial
nickname={socials.tg}
icon="fa6-brands--telegram"
url={`https://t.me/${socials.tg}`}
color="2aabee"
/>
</Link>
)}
{socials.tt && (
<Link href={`https://tiktok.com/@${socials.tt}`} target="_blank">
<UserSocial
nickname={socials.tt}
icon="fa6-brands--tiktok"
url={`https://tiktok.com/@${socials.tt}`}
color={theme == "light" ? "000000" : "ffffff"}
/>
</Link>
)}
{socials.inst && (
<Link
key={social.name}
href={`${social.urlPrefix}${social.nickname}`}
href={`https://instagram.com/${socials.inst}`}
target="_blank"
>
<Button color="light">
<div className="flex items-center justify-center gap-2">
<span
className={`iconify h-4 w-4 sm:h-6 sm:w-6 ${social.icon} dark:fill-white`}
></span>
{social.nickname}
</div>
</Button>
<UserSocial
nickname={socials.inst}
icon="fa6-brands--instagram"
url={`https://instagram.com/${socials.inst}`}
color="c32aa3"
/>
</Link>
);
})}
)}
{socials.discord && (
<UserSocial
nickname={socials.discord}
icon="fa6-brands--discord"
url={`https://discord.com/${socials.discord}`}
color="5865f2"
/>
)}
</div>
)}
</Card>

View file

@ -59,50 +59,7 @@ export const ProfilePage = (props: any) => {
);
}
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 ||
(user.roles && user.roles.length > 0) ||
isMyProfile;
const isPrivacy =
user.is_stats_hidden || user.is_counts_hidden || user.is_social_hidden;
@ -120,7 +77,45 @@ export const ProfilePage = (props: any) => {
is_me_blocked={user.is_me_blocked}
/>
</div>
<div
<div className="grid grid-cols-1 gap-2 lg:grid-cols-2">
<div className="flex flex-col gap-2">
<ProfileUser
avatar={user.avatar}
login={user.login}
status={user.status}
roles={user.roles}
rating={user.rating_score}
isMyProfile={isMyProfile}
isVerified={user.is_verified}
isOnline={user.is_online}
isSponsor={user.is_sponsor}
isBlocked={user.is_blocked}
socials={{
vk: user.vk_page || null,
tg: user.tg_page || null,
tt: user.tt_page || null,
inst: user.inst_page || null,
discord: user.discord_page || null,
}}
/>
{authUser.token && (
<ProfileActions
isMyProfile={isMyProfile}
profile_id={user.id}
isFriendRequestsDisallowed={user.is_friend_requests_disallowed}
friendStatus={user.friend_status}
my_profile_id={authUser.user.id}
token={authUser.token}
is_me_blocked={user.is_me_blocked}
is_blocked={user.is_blocked}
edit_isOpen={isOpen}
edit_setIsOpen={setIsOpen}
/>
)}
</div>
<div className="flex flex-col gap-2"></div>
</div>
{/* <div
className={`flex flex-wrap gap-2 ${
isPrivacy || user.is_banned || user.is_perm_banned ? "mt-4" : ""
}`}
@ -171,20 +166,7 @@ export const ProfilePage = (props: any) => {
)}
</div>
<div className="flex flex-col w-full gap-2 xl:flex-1 xl:w-auto ">
{authUser.token && (
<ProfileActions
isMyProfile={isMyProfile}
profile_id={user.id}
isFriendRequestsDisallowed={user.is_friend_requests_disallowed}
friendStatus={user.friend_status}
my_profile_id={authUser.user.id}
token={authUser.token}
is_me_blocked={user.is_me_blocked}
is_blocked={user.is_blocked}
edit_isOpen={isOpen}
edit_setIsOpen={setIsOpen}
/>
)}
{!user.is_stats_hidden && (
<>
<ProfileStats
@ -215,7 +197,7 @@ export const ProfilePage = (props: any) => {
</>
)}
</div>
</div>
</div> */}
<ProfileEditModal
isOpen={isOpen && isMyProfile}
setIsOpen={setIsOpen}