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