impl friends system
All checks were successful
Deploy Backend / deploy (push) Successful in 2m43s
Deploy Website / deploy (push) Successful in 4m55s

This commit is contained in:
Lee
2024-10-16 11:23:28 +01:00
parent 2e2c03241e
commit 8ab81b1b27
9 changed files with 375 additions and 12 deletions

View File

@ -2,8 +2,12 @@ import Dexie, { EntityTable } from "dexie";
import BeatSaverMap from "./types/beatsaver-map";
import Settings from "./types/settings";
import { setCookieValue } from "@/common/cookie-utils";
import { Friend } from "@/common/database/types/friends";
import ScoreSaberPlayerToken from "@ssr/common/types/token/scoresaber/score-saber-player-token";
import { scoresaberService } from "@ssr/common/service/impl/scoresaber";
const SETTINGS_ID = "SSR"; // DO NOT CHANGE
const FRIENDS_ID = "FRIENDS"; // DO NOT CHANGE
export default class Database extends Dexie {
/**
@ -16,6 +20,11 @@ export default class Database extends Dexie {
*/
beatSaverMaps!: EntityTable<BeatSaverMap, "hash">;
/**
* The added friends
*/
friends!: EntityTable<Friend, "id">;
constructor() {
super("ScoreSaberReloaded");
@ -23,6 +32,7 @@ export default class Database extends Dexie {
this.version(1).stores({
settings: "id",
beatSaverMaps: "hash",
friends: "id",
});
// Mapped tables
@ -68,6 +78,49 @@ export default class Database extends Dexie {
return this.settings.update(SETTINGS_ID, settings);
}
/**
* Adds a friend
*
* @param id the id of the friend
*/
public async addFriend(id: string) {
await this.friends.add({ id });
}
/**
* Removes a friend
*
* @param id the id of the friend
*/
public async removeFriend(id: string) {
await this.friends.delete(id);
}
/**
* Checks if this player is a friend
*
* @param id the id of the player
*/
public async isFriend(id: string): Promise<boolean> {
const friend = await this.friends.get(id);
return friend != undefined;
}
/**
* Gets all friends as {@link ScoreSaberPlayerToken}'s
*
* @returns the friends
*/
public async getFriends(): Promise<ScoreSaberPlayerToken[]> {
const friends = await this.friends.toArray();
const players = await Promise.all(
friends.map(async ({ id }) => {
return await scoresaberService.lookupPlayer(id, true);
})
);
return players.filter(player => player !== undefined) as ScoreSaberPlayerToken[];
}
/**
* Resets the settings in the database
*/

View File

@ -0,0 +1,6 @@
export interface Friend {
/**
* The id of the friend
*/
id: string;
}

View File

@ -0,0 +1,49 @@
"use client";
import { useLiveQuery } from "dexie-react-hooks";
import useDatabase from "../../hooks/use-database";
import { useToast } from "@/hooks/use-toast";
import Tooltip from "../tooltip";
import { Button } from "../ui/button";
import { PersonIcon } from "@radix-ui/react-icons";
import ScoreSaberPlayer from "@ssr/common/types/player/impl/scoresaber-player";
type Props = {
/**
* The ID of the players profile to claim.
*/
player: ScoreSaberPlayer;
};
export default function AddFriend({ player }: Props) {
const { id, name } = player;
const database = useDatabase();
const isFriend = useLiveQuery(() => database.isFriend(id));
const settings = useLiveQuery(() => database.getSettings());
const { toast } = useToast();
/**
* Adds this player as a friend
*/
async function addFriend() {
await database.addFriend(id);
toast({
title: "Friend Added",
description: `You have added ${name} as a friend.`,
});
}
// If the player is already a friend, don't show the button
if (isFriend || settings?.playerId == id) {
return null;
}
return (
<Tooltip display={<p>Add {name} as a friend!</p>} side={"bottom"}>
<Button variant={"outline"} onClick={addFriend}>
<PersonIcon className="size-6 text-green-500" />
</Button>
</Tooltip>
);
}

View File

@ -0,0 +1,56 @@
import ScoreSaberPlayerToken from "@ssr/common/types/token/scoresaber/score-saber-player-token";
import { Avatar, AvatarImage } from "@/components/ui/avatar";
import { formatNumberWithCommas } from "@ssr/common/utils/number-utils";
import Link from "next/link";
import { XIcon } from "lucide-react";
import useDatabase from "@/hooks/use-database";
import { useToast } from "@/hooks/use-toast";
import Tooltip from "@/components/tooltip";
type FriendProps = {
/**
* The friend to display.
*/
player: ScoreSaberPlayerToken;
/**
* When the friend is clicked
*/
onClick?: () => void;
};
export default function Friend({ player, onClick }: FriendProps) {
const { id, name } = player;
const database = useDatabase();
const { toast } = useToast();
/**
* Adds this player as a friend
*/
async function removeFriend() {
await database.removeFriend(id);
toast({
title: "Friend Removed",
description: `You have removed ${name} as a friend.`,
});
}
return (
<li className="flex items-center justify-between gap-2 hover:bg-accent transition-all transform-gpu p-2 rounded-md select-none">
<Link href={`/player/${player.id}`} onClick={() => onClick?.()} className="flex items-center gap-2 w-full">
<Avatar>
<AvatarImage src={player.profilePicture} alt={player.name} />
</Avatar>
<div className="flex flex-col">
<p className="text-lg font-semibold">{player.name}</p>
<p className="text-gray-400">#{formatNumberWithCommas(player.rank)}</p>
</div>
</Link>
<Tooltip display={<p className="cursor-default pointer-events-none">Remove {name} from your friends</p>}>
<button onClick={() => removeFriend()}>
<XIcon className="w-5 h-5" />
</button>
</Tooltip>
</li>
);
}

View File

@ -0,0 +1,47 @@
"use client";
import useDatabase from "@/hooks/use-database";
import { useLiveQuery } from "dexie-react-hooks";
import NavbarButton from "./navbar-button";
import { PersonIcon } from "@radix-ui/react-icons";
import { Popover, PopoverContent, PopoverTrigger } from "@/components/ui/popover";
import Friend from "@/components/friend/friend";
import { useState } from "react";
import { Button } from "@/components/ui/button";
import Link from "next/link";
export default function FriendsButton() {
const [open, setOpen] = useState(false);
const database = useDatabase();
const friends = useLiveQuery(() => database.getFriends());
if (friends == undefined) {
return; // Friends haven't loaded yet
}
return (
<Popover open={open} onOpenChange={setOpen}>
<PopoverTrigger className="h-full">
<NavbarButton>
<PersonIcon className="w-6 h-6" />
<p className="hidden lg:block">Friends</p>
</NavbarButton>
</PopoverTrigger>
<PopoverContent className="p-2">
{friends.length > 0 ? (
friends.map(friend => <Friend player={friend} key={friend.id} onClick={() => setOpen(false)} />)
) : (
<div className="text-sm flex flex-col gap-2 justify-center items-center">
<p>You don&#39;t have any friends :(</p>
<Link href="/search">
<Button size="sm" onClick={() => setOpen(false)}>
Search Player
</Button>
</Link>
</div>
)}
</PopoverContent>
</Popover>
);
}

View File

@ -1,10 +1,11 @@
import { CogIcon, HomeIcon } from "@heroicons/react/24/solid";
import { CogIcon } from "@heroicons/react/24/solid";
import { MagnifyingGlassIcon } from "@radix-ui/react-icons";
import Link from "next/link";
import React from "react";
import NavbarButton from "./navbar-button";
import ProfileButton from "./profile-button";
import { SwordIcon, TrendingUpIcon } from "lucide-react";
import FriendsButton from "@/components/navbar/friends-button";
type NavbarItem = {
name: string;
@ -14,12 +15,6 @@ type NavbarItem = {
};
const items: NavbarItem[] = [
{
name: "Home",
link: "/",
align: "left",
icon: <HomeIcon className="h-5 w-5" />,
},
{
name: "Ranking",
link: "/ranking",
@ -61,11 +56,25 @@ export default function Navbar() {
return (
<div className="w-full sticky top-0 z-[999] h-10 items-center flex justify-between bg-secondary/95 px-1">
<div className="md:max-w-[1600px] w-full h-full flex justify-between m-auto">
{/* Left-aligned items */}
<div className="flex items-center h-full">
<ProfileButton />
{/* Home Button */}
<Link href="/" className="h-full">
<NavbarButton>
{renderNavbarItem({
name: "Home",
link: "/",
align: "left",
icon: <img src="/assets/logos/scoresaber.png" className="h-5 w-5" alt="Website Logo" />,
})}
</NavbarButton>
</Link>
{leftItems.map((item, index) => (
{/* Player Buttons */}
<ProfileButton />
<FriendsButton />
{/* Left-aligned items */}
{leftItems.splice(0, leftItems.length).map((item, index) => (
<Link href={item.link} key={index} className="h-full">
<NavbarButton key={index}>{renderNavbarItem(item)}</NavbarButton>
</Link>

View File

@ -11,6 +11,7 @@ import PlayerTrackedStatus from "@/components/player/player-tracked-status";
import ScoreSaberPlayer from "@ssr/common/types/player/impl/scoresaber-player";
import Link from "next/link";
import { capitalizeFirstLetter } from "@/common/string-utils";
import AddFriend from "@/components/friend/add-friend";
/**
* Renders the change for a stat.
@ -197,7 +198,8 @@ export default function PlayerHeader({ player }: Props) {
<PlayerStats player={player} />
<div className="absolute top-0 right-0">
<div className="absolute top-0 right-0 gap-2 flex">
<AddFriend player={player} />
<ClaimProfile playerId={player.id} />
</div>
</div>