added lobby membership dialog case of already being in a lobby

This commit is contained in:
shrt 2025-03-30 13:15:36 +02:00
parent 4009c44841
commit abc698998e
16 changed files with 323 additions and 154 deletions

View File

@ -58,7 +58,8 @@
"tailwind-merge": "^3.0.2", "tailwind-merge": "^3.0.2",
"tsx": "^4.19.3", "tsx": "^4.19.3",
"tw-animate-css": "^1.2.4", "tw-animate-css": "^1.2.4",
"zod": "^3.24.2" "zod": "^3.24.2",
"zustand": "^5.0.3"
}, },
"devDependencies": { "devDependencies": {
"@eslint/eslintrc": "^3.3.1", "@eslint/eslintrc": "^3.3.1",

26
pnpm-lock.yaml generated
View File

@ -125,6 +125,9 @@ importers:
zod: zod:
specifier: ^3.24.2 specifier: ^3.24.2
version: 3.24.2 version: 3.24.2
zustand:
specifier: ^5.0.3
version: 5.0.3(@types/react@19.0.12)(react@19.0.0)
devDependencies: devDependencies:
'@eslint/eslintrc': '@eslint/eslintrc':
specifier: ^3.3.1 specifier: ^3.3.1
@ -3382,6 +3385,24 @@ packages:
zod@3.24.2: zod@3.24.2:
resolution: {integrity: sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==} resolution: {integrity: sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==}
zustand@5.0.3:
resolution: {integrity: sha512-14fwWQtU3pH4dE0dOpdMiWjddcH+QzKIgk1cl8epwSE7yag43k/AD/m4L6+K7DytAOr9gGBe3/EXj9g7cdostg==}
engines: {node: '>=12.20.0'}
peerDependencies:
'@types/react': '>=18.0.0'
immer: '>=9.0.6'
react: '>=18.0.0'
use-sync-external-store: '>=1.2.0'
peerDependenciesMeta:
'@types/react':
optional: true
immer:
optional: true
react:
optional: true
use-sync-external-store:
optional: true
snapshots: snapshots:
'@alloc/quick-lru@5.2.0': {} '@alloc/quick-lru@5.2.0': {}
@ -6498,3 +6519,8 @@ snapshots:
yocto-queue@0.1.0: {} yocto-queue@0.1.0: {}
zod@3.24.2: {} zod@3.24.2: {}
zustand@5.0.3(@types/react@19.0.12)(react@19.0.0):
optionalDependencies:
'@types/react': 19.0.12
react: 19.0.0

View File

@ -1,16 +1,22 @@
import React from "react"; import React from "react";
import Navbar from "@/components/navbar"; import Navbar from "@/components/navbar";
import Footer from "@/components/footer"; import Footer from "@/components/footer";
import { api } from "@/trpc/server";
import AppProvider from "../_components/app-provider";
async function Layout({ children }: { children: React.ReactNode }) {
const sessionPlayer = await api.player.getBySession();
function Layout({ children }: { children: React.ReactNode }) {
return ( return (
<div className="gradient-bg min-h-screen"> <AppProvider initialSessionPlayer={sessionPlayer}>
<Navbar /> <div className="gradient-bg min-h-screen">
<div className="container mb-8 flex size-full min-h-screen flex-col items-center space-y-6"> <Navbar />
{children} <div className="container mb-8 flex size-full min-h-screen flex-col items-center space-y-6">
{children}
</div>
<Footer />
</div> </div>
<Footer /> </AppProvider>
</div>
); );
} }

View File

@ -0,0 +1,22 @@
"use client";
import { useSessionPlayerStore } from "@/lib/store/current-player-store";
import type { Player } from "@/server/db/schema";
import React from "react";
function AppProvider({
initialSessionPlayer,
children,
}: {
initialSessionPlayer?: Player | null;
children: React.ReactNode;
}) {
const setSessionPlayer = useSessionPlayerStore(
(state) => state.setSessionPlayer,
);
React.useEffect(() => {
setSessionPlayer(initialSessionPlayer);
}, [initialSessionPlayer]);
return children;
}
export default AppProvider;

View File

@ -37,7 +37,7 @@ function LobbyForm({
const form = useForm<z.infer<typeof lobbyPatchSchema>>({ const form = useForm<z.infer<typeof lobbyPatchSchema>>({
resolver: zodResolver(lobbyPatchSchema), resolver: zodResolver(lobbyPatchSchema),
defaultValues: { defaultValues: {
name: server_lobby?.name ?? "", name: server_lobby?.name ?? "Random Name",
maxPlayers: server_lobby?.maxPlayers ?? 2, maxPlayers: server_lobby?.maxPlayers ?? 2,
}, },
}); });
@ -107,7 +107,10 @@ function LobbyForm({
<div className="flex items-center justify-end"> <div className="flex items-center justify-end">
<Button <Button
type="submit" type="submit"
disabled={loading || !form.formState.isDirty} disabled={
loading ||
(Boolean(server_lobby?.id?.length) && !form.formState.isDirty)
}
className="ml-auto" className="ml-auto"
> >
{server_lobby?.id?.length ? "Update" : "Create"} Lobby {server_lobby?.id?.length ? "Update" : "Create"} Lobby

View File

@ -26,19 +26,39 @@ function LobbyMembershipDialog({
join: boolean; join: boolean;
}) { }) {
const [loading, setLoading] = React.useState(false); const [loading, setLoading] = React.useState(false);
const membership = api.lobby.membership.useMutation();
const labelText = join ? "join" : "leave"; const labelText = join ? "join" : "leave";
const router = useRouter(); const router = useRouter();
const membership = api.lobby.membership.useMutation({
onSuccess(data) {
if (data?.success) {
toast.success(`Successfully ${labelText} the lobby.`);
if (join) router.push(appRoutes.currentlobby);
else router.push(appRoutes.lobby(lobbyId));
router.refresh();
} else {
if (data?.knownError) {
toast.message(data.error, {
action: {
label: "Jump to Lobby",
onClick: () => router.push(appRoutes.currentlobby),
},
});
}
}
},
});
const handleConfirm = async () => { const handleConfirm = async () => {
setLoading(true); setLoading(true);
const result = await membership.mutateAsync({ lobbyId, join }); try {
if (result) { await membership.mutateAsync({
toast.success(`Successfully ${labelText} the lobby.`); lobbyId,
if (join) router.push(appRoutes.currentlobby); join,
else router.push(appRoutes.lobby(lobbyId)); });
} catch (e) {
// Catching any errors here will prevent them from reaching Next.js's error boundary
console.error("Handled mutation error:", e);
}
router.refresh();
} else toast.error("Something went wrong");
setLoading(false); setLoading(false);
}; };

View File

@ -1,90 +1,104 @@
"use client"; "use client";
import React from "react"; import React from "react";
import type { Lobby, Player } from "@/server/db/schema"; import type { Lobby } from "@/server/db/schema";
import LobbyPlayerCard from "@/app/_components/lobby/lobby-player/lobby-player-card"; import LobbyPlayerCard from "@/app/_components/lobby/lobby-player/player-card";
import LobbyMembershipDialog from "@/app/_components/lobby/lobby-membership-dialog"; import LobbyMembershipDialog from "@/app/_components/lobby/lobby-membership-dialog";
import { Badge } from "@/components/ui/badge"; import { Badge } from "@/components/ui/badge";
import CopyToClip from "@/components/copy-to-clip"; import CopyToClip from "@/components/copy-to-clip";
import { appRoutes } from "@/config/app.routes"; import { appRoutes } from "@/config/app.routes";
import LobbySettingsDialog from "./lobby-settings-dialog"; import LobbySettingsDialog from "./lobby-settings-dialog";
import { useRealtimeLobby } from "@/hooks/use-lobby";
import { getBaseUrl } from "@/lib/utils"; import { getBaseUrl } from "@/lib/utils";
import GameSelector from "./game-selector"; import GameSelector from "./game-selector";
import LobbyProvider from "./lobby-provider";
import { useSessionPlayerStore } from "@/lib/store/current-player-store";
import { useLobbyStore } from "@/lib/store/lobby-store";
import { Share2, UserPlus } from "lucide-react";
function LobbyPage({ function LobbyPage({ initialLobby }: { initialLobby: Lobby }) {
sessionPlayer, const sessionPlayer = useSessionPlayerStore((state) => state.sessionPlayer);
initialLobby, const lobby = useLobbyStore((state) => state.lobby);
}: { const members = useLobbyStore((state) => state.members);
sessionPlayer?: Player | null; const isJoined = sessionPlayer
initialLobby: Lobby; ? members.find((m) => m.playerId === sessionPlayer.id)
}) { : null;
const { members, lobby } = useRealtimeLobby({ initialLobby, sessionPlayer });
const isJoined = members.find((m) => m.playerId === sessionPlayer?.id);
const isAdmin = isJoined?.role === "admin"; const isAdmin = isJoined?.role === "admin";
return ( return (
<div className="grid max-w-4xl grid-cols-3 grid-rows-2 gap-4"> <LobbyProvider initialLobby={initialLobby}>
<div className="container-bg col-span-2 w-full space-y-4 p-6"> <div className="grid max-w-4xl grid-cols-3 grid-rows-2 gap-4">
<div className="flex items-center gap-2"> <div className="container-bg col-span-2 w-full space-y-4 p-6">
{isJoined && ( <div className="flex items-center gap-2">
<CopyToClip <CopyToClip
target="invite link" target="invite link"
text={getBaseUrl() + appRoutes.lobby(lobby.id)} text={getBaseUrl() + appRoutes.lobby(lobby.id)}
buttonProps={{ variant: "outline" }} buttonProps={{ variant: "outline" }}
> >
Invite Players {isJoined ? (
<>
<UserPlus className="size-4" />
<span>Invite Players</span>
</>
) : (
<>
<Share2 className="size-4" />
<span>Share Lobby</span>
</>
)}
</CopyToClip> </CopyToClip>
)}
<div className="ml-auto">
{sessionPlayer && (
<LobbyMembershipDialog lobbyId={lobby.id} join={!isJoined} />
)}
</div>
</div>
<div className="flex w-full items-center justify-between text-nowrap">
<h1 className="w-full overflow-hidden text-2xl font-bold text-ellipsis">
{lobby.name}
</h1>
{isAdmin && <LobbySettingsDialog lobby={lobby} />}
</div>
<div className="border-border/20 flex items-center justify-between border-b pb-4">
<h2 className="text-sm font-medium">
Players ({members?.length || 0}/{lobby.maxPlayers || 8})
</h2>
<span className="text-xs">
{members?.length === (lobby.maxPlayers || 8)
? "Lobby Full"
: "Waiting..."}
</span>
</div>
<ul className="space-y-2"> <div className="ml-auto">
{members?.map((member, idx) => ( {sessionPlayer && (
<li key={idx}> <LobbyMembershipDialog lobbyId={lobby.id} join={!isJoined} />
<LobbyPlayerCard )}
lobbyId={lobby.id} </div>
highlight={member?.player?.id === sessionPlayer?.id} </div>
player={member?.player!} <div className="flex w-full items-center justify-between text-nowrap">
className="relative" <h1 className="w-full overflow-hidden text-2xl font-bold text-ellipsis">
showOptions={isAdmin && member?.playerId !== sessionPlayer?.id} {lobby.name}
> </h1>
{member?.role === "admin" && ( {isAdmin && <LobbySettingsDialog lobby={lobby} />}
<Badge className="absolute -top-2 right-2 z-20 p-px px-2 text-white"> </div>
Admin <div className="border-border/20 flex items-center justify-between border-b pb-4">
</Badge> <h2 className="text-sm font-medium">
)} Players ({members?.length || 0}/{lobby.maxPlayers || 8})
</LobbyPlayerCard> </h2>
</li> <span className="text-xs">
))} {members?.length === (lobby.maxPlayers || 8)
</ul> ? "Lobby Full"
: "Waiting..."}
</span>
</div>
<ul className="space-y-2">
{members?.map((member, idx) => (
<li key={idx}>
<LobbyPlayerCard
lobbyId={lobby.id}
highlight={member?.player?.id === sessionPlayer?.id}
player={member?.player!}
className="relative"
showOptions={
isAdmin && member?.playerId !== sessionPlayer?.id
}
>
{member?.role === "admin" && (
<Badge className="absolute -top-2 right-2 z-20 p-px px-2 text-white">
Admin
</Badge>
)}
</LobbyPlayerCard>
</li>
))}
</ul>
</div>
<div className="container-bg col-span-1 size-full"></div>
<div className="col-span-3">
<GameSelector />
</div>
</div> </div>
<div className="container-bg col-span-1 size-full"></div> </LobbyProvider>
<div className="col-span-3">
<GameSelector />
</div>
</div>
); );
} }

View File

@ -2,7 +2,7 @@ import React from "react";
import Avatar from "@/components/avatar"; import Avatar from "@/components/avatar";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import type { Player } from "@/server/db/schema"; import type { Player } from "@/server/db/schema";
import LobbyPlayerOptions from "./lobby-player-options"; import LobbyPlayerOptions from "./player-options";
function LobbyPlayerCard({ function LobbyPlayerCard({
lobbyId, lobbyId,

View File

@ -23,7 +23,7 @@ import type { Player } from "@/server/db/schema";
import Link from "next/link"; import Link from "next/link";
import { appRoutes } from "@/config/app.routes"; import { appRoutes } from "@/config/app.routes";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import KickPlayerDialog from "./kick-player-dialog"; import KickPlayerDialog from "./player-kick-dialog";
import { api } from "@/trpc/react"; import { api } from "@/trpc/react";
function LobbyPlayerOptions({ function LobbyPlayerOptions({
player, player,

View File

@ -0,0 +1,57 @@
import { appRoutes } from "@/config/app.routes";
import { useSessionPlayerStore } from "@/lib/store/current-player-store";
import { useLobbyStore } from "@/lib/store/lobby-store";
import type { Lobby, LobbyMember } from "@/server/db/schema";
import type { LobbyMemberLeaveEventData } from "@/server/redis/events";
import { api } from "@/trpc/react";
import { useRouter } from "next/navigation";
import React from "react";
function LobbyProvider({
children,
initialLobby,
}: {
children: React.ReactNode;
initialLobby: Lobby;
}) {
const sessionPlayer = useSessionPlayerStore((state) => state.sessionPlayer);
const lobby = useLobbyStore((state) => state.lobby);
const updateLobby = useLobbyStore((state) => state.updateLobby);
const resetLobby = useLobbyStore((state) => state.resetLobby);
const setMembers = useLobbyStore((state) => state.setMembers);
const addMember = useLobbyStore((state) => state.addMember);
const removeMember = useLobbyStore((state) => state.removeMember);
const router = useRouter();
api.lobby.onMemberUpdate.useSubscription(undefined, {
onData({ data: _data }) {
const joined = _data.joined;
if (joined) {
const data = _data.membership as LobbyMember;
addMember(data);
} else {
const data = _data.membership as LobbyMemberLeaveEventData;
removeMember(data.playerId);
if (data?.kicked && data?.playerId === sessionPlayer?.id) {
router.push(appRoutes.lobby(lobby.id));
router.refresh();
}
}
},
});
api.lobby.onUpdate.useSubscription(undefined, {
onData({ data }) {
if (!data?.lobby) return;
updateLobby(data.lobby);
},
});
React.useEffect(() => {
resetLobby(initialLobby);
setMembers(initialLobby?.members ?? []);
}, [initialLobby]);
return children;
}
export default LobbyProvider;

View File

@ -1,50 +0,0 @@
"use client";
import { appRoutes } from "@/config/app.routes";
import type { Lobby, LobbyMember, Player } from "@/server/db/schema";
import type { LobbyMemberLeaveEventData } from "@/server/redis/events";
import { api } from "@/trpc/react";
import { useRouter } from "next/navigation";
import React from "react";
export function useRealtimeLobby({
initialLobby,
sessionPlayer,
}: {
initialLobby: Lobby;
sessionPlayer?: Player | null;
}) {
const [lobby, setLobby] = React.useState(initialLobby);
const [members, setMembers] = React.useState<Array<LobbyMember>>(
initialLobby?.members ?? [],
);
const router = useRouter();
api.lobby.onMemberUpdate.useSubscription(undefined, {
onData({ data: _data }) {
const joined = _data.joined;
if (joined) {
const data = _data.membership as LobbyMember;
setMembers((prev) => {
if (prev.find((m) => m.playerId === data.playerId)) return prev;
return [...prev, data];
});
} else {
const data = _data.membership as LobbyMemberLeaveEventData;
setMembers((prev) => prev.filter((m) => m.playerId !== data.playerId));
if (data?.kicked && data?.playerId === sessionPlayer?.id) {
router.push(appRoutes.lobby(lobby.id));
router.refresh();
}
}
},
});
api.lobby.onUpdate.useSubscription(undefined, {
onData({ data }) {
if (!data?.lobby) return;
setLobby((prev) => ({ ...data.lobby, members: prev.members }));
},
});
return { lobby, members };
}

View File

@ -0,0 +1,12 @@
import { create } from "zustand";
import type { Player } from "@/server/db/schema";
type SessionPlayerStore = {
sessionPlayer?: Player | null;
setSessionPlayer: (player?: Player | null) => void;
};
export const useSessionPlayerStore = create<SessionPlayerStore>((set) => ({
sessionPlayer: undefined,
setSessionPlayer: (player) => set({ sessionPlayer: player }),
}));

View File

@ -0,0 +1,37 @@
import { create } from "zustand";
import type { Lobby, LobbyMember } from "@/server/db/schema";
type LobbyStore = {
lobby: Lobby;
updateLobby: (lobby: Lobby) => void;
resetLobby: (lobby?: Lobby) => void;
members: Array<LobbyMember>;
setMembers: (members: Array<LobbyMember>) => void;
findMember: (playerId: string) => LobbyMember | null | undefined;
addMember: (member: LobbyMember) => void;
removeMember: (playerId: string, kicked?: boolean) => void;
selectedGame: number;
setSelectedGame: (gameId: number) => void;
};
export const useLobbyStore = create<LobbyStore>((set, get) => ({
lobby: {} as Lobby,
members: [],
selectedGame: 0,
setSelectedGame: (gameId) => set({ selectedGame: gameId }),
updateLobby: (lobby) =>
set((state) => ({ ...state, ...lobby, members: state.members })),
setMembers: (members) => set({ members }),
resetLobby: (lobby) => set({ lobby: lobby ?? ({} as Lobby) }),
findMember: (playerId) => get().members.find((m) => m.playerId === playerId),
addMember: (member) =>
set((state) => {
if (state.members.find((m) => m.playerId === member.playerId))
return state;
return { members: [...state.members, member] };
}),
removeMember: (playerId) =>
set((state) => ({
members: state.members.filter((m) => m.playerId !== playerId),
})),
}));

View File

@ -16,7 +16,7 @@ import {
redisAsyncIterator, redisAsyncIterator,
redisPublish, redisPublish,
} from "@/server/redis/sse-redis"; } from "@/server/redis/sse-redis";
import { tracked } from "@trpc/server"; import { tracked, TRPCError } from "@trpc/server";
import { trcpSubscriptionInput } from "@/lib/validations/trcp"; import { trcpSubscriptionInput } from "@/lib/validations/trcp";
import type { LobbyMemberLeaveEventData } from "@/server/redis/events"; import type { LobbyMemberLeaveEventData } from "@/server/redis/events";
@ -140,22 +140,43 @@ export const lobbyRouter = createTRPCRouter({
) )
.mutation(async ({ ctx, input }) => { .mutation(async ({ ctx, input }) => {
if (input.join) { if (input.join) {
const [member] = await ctx.db try {
.insert(lobbyMembers) const [member] = await ctx.db
.values({ .insert(lobbyMembers)
lobbyId: input.lobbyId, .values({
playerId: ctx.session.user.id, lobbyId: input.lobbyId,
isReady: false, playerId: ctx.session.user.id,
role: "player", isReady: false,
}) role: "player",
.returning();
const player = member
? await ctx.db.query.players.findFirst({
where: eq(players.id, member.playerId),
}) })
: undefined; .returning();
if (member) redisPublish("lobby:member:join", { ...member, player }); const player = member
return member; ? await ctx.db.query.players.findFirst({
where: eq(players.id, member.playerId),
})
: undefined;
if (member) redisPublish("lobby:member:join", { ...member, player });
return { success: true, member };
} catch (e: unknown) {
if (e instanceof Error) {
if (
e.message.includes(
"duplicate key value violates unique constraint",
)
)
return {
knownError: true,
succes: false,
error: "You can only be in one lobby at a time.",
};
} else {
return {
knownError: true,
succes: false,
error: "Error joining lobby",
};
}
}
} else { } else {
const [member] = await ctx.db const [member] = await ctx.db
.delete(lobbyMembers) .delete(lobbyMembers)
@ -168,7 +189,7 @@ export const lobbyRouter = createTRPCRouter({
.returning(); .returning();
if (member) if (member)
redisPublish("lobby:member:leave", { playerId: member.playerId }); redisPublish("lobby:member:leave", { playerId: member.playerId });
return member; return { success: true, member };
} }
}), }),
// admin mutaions // admin mutaions

View File

@ -1,13 +1,13 @@
import { eq } from "drizzle-orm"; import { eq } from "drizzle-orm";
import { createTRPCRouter, protectedProcedure, publicProcedure } from "../trpc"; import { createTRPCRouter, protectedProcedure, publicProcedure } from "../trpc";
import { players } from "@/server/db/schema"; import { players, type Player } from "@/server/db/schema";
export const playerRouter = createTRPCRouter({ export const playerRouter = createTRPCRouter({
getBySession: publicProcedure.query(async ({ ctx }) => { getBySession: publicProcedure.query(async ({ ctx }) => {
return ctx?.session?.user return ctx?.session?.user
? await ctx.db.query.players.findFirst({ ? ((await ctx.db.query.players.findFirst({
where: eq(players.id, ctx.session.user.id), where: eq(players.id, ctx.session.user.id),
}) })) as Player)
: null; : null;
}), }),
}); });