import { formatNumberWithCommas, formatPp } from "@ssr/common/utils/number-utils"; import { GlobeAmericasIcon } from "@heroicons/react/24/solid"; import Card from "../card"; import CountryFlag from "../country-flag"; import { Avatar, AvatarImage } from "../ui/avatar"; import ClaimProfile from "./claim-profile"; import PlayerStats from "./player-stats"; import Tooltip from "@/components/tooltip"; import { ReactElement } from "react"; import PlayerTrackedStatus from "@/components/player/player-tracked-status"; import ScoreSaberPlayer from "@ssr/common/player/impl/scoresaber-player"; import Link from "next/link"; import { capitalizeFirstLetter } from "@/common/string-utils"; import AddFriend from "@/components/friend/add-friend"; import PlayerSteamProfile from "@/components/player/player-steam-profile"; import { getScoreSaberRole } from "@ssr/common/scoresaber.util"; /** * Renders the change for a stat. * * @param change the amount of change * @param tooltip the tooltip to display * @param format the function to format the value */ const renderDailyChange = (change: number, tooltip: ReactElement, format?: (value: number) => string) => { format = format ?? formatNumberWithCommas; return (

0 ? "text-green-400" : "text-red-400"}`}> {change > 0 ? "+" : ""} {format(change)}

); }; /** * Renders the change over time a stat eg: rank, country rank * * @param player the player to get the stats for * @param children the children to render * @param type the type of stat to get the change for */ const renderChange = (player: ScoreSaberPlayer, type: "rank" | "countryRank" | "pp", children: ReactElement) => { const todayStats = player.statisticChange?.daily; const weeklyStats = player.statisticChange?.weekly; const monthlyStats = player.statisticChange?.monthly; const todayStat = todayStats?.[type]; const weeklyStat = weeklyStats?.[type]; const monthlyStat = monthlyStats?.[type]; const renderChange = (value: number | undefined, timeFrame: "daily" | "weekly" | "monthly") => { const format = (value: number | undefined) => { if (value == 0) { return 0; } if (value == undefined) { return "No Data"; } return type == "pp" ? formatPp(value) + "pp" : formatNumberWithCommas(value); }; return (

{capitalizeFirstLetter(timeFrame)} Change:{" "} = 0 ? (value == 0 ? "" : "text-green-500") : "text-red-500"}`} > {format(value)}

); }; // Don't show change if the player is banned or inactive if (player.banned || player.inactive) { return children; } return ( {renderChange(todayStat, "daily")} {renderChange(weeklyStat, "weekly")} {renderChange(monthlyStat, "monthly")} } > {children} ); }; const playerData = [ { showWhenInactiveOrBanned: false, icon: () => { return ; }, render: (player: ScoreSaberPlayer) => { const statisticChange = player.statisticChange; const rankChange = statisticChange?.daily?.rank ?? 0; return (
{renderChange( player, "rank",

#{formatNumberWithCommas(player.rank)}

)} {rankChange != 0 && renderDailyChange(rankChange,

The change in rank compared to yesterday

)}
); }, }, { showWhenInactiveOrBanned: false, icon: (player: ScoreSaberPlayer) => { return ; }, render: (player: ScoreSaberPlayer) => { const statisticChange = player.statisticChange; const rankChange = statisticChange?.daily?.countryRank ?? 0; return (
{renderChange( player, "countryRank",

#{formatNumberWithCommas(player.countryRank)}

)} {rankChange != 0 && renderDailyChange(rankChange,

The change in country rank compared to yesterday

)}
); }, }, { showWhenInactiveOrBanned: true, render: (player: ScoreSaberPlayer) => { const statisticChange = player.statisticChange; const ppChange = statisticChange?.daily?.pp ?? 0; return (
{renderChange( player, "pp",

{formatPp(player.pp)}pp

)} {ppChange != 0 && renderDailyChange(ppChange,

The change in pp compared to yesterday

)}
); }, }, ]; type Props = { player: ScoreSaberPlayer; }; export default function PlayerHeader({ player }: Props) { return (

{player.name}

{player.inactive &&

Inactive Account

} {player.banned &&

Banned Account

}
{playerData.map((subName, index) => { // Check if the player is inactive or banned and if the data should be shown if (!subName.showWhenInactiveOrBanned && (player.inactive || player.banned)) { return null; } return (
{subName.icon && subName.icon(player)} {subName.render && subName.render(player)}
); })}
); }