redone player/user structure. player can only be in one lobby; realtime with trpc subscriptions and redis
This commit is contained in:
		
							parent
							
								
									a00116a61d
								
							
						
					
					
						commit
						2db3af2dfc
					
				| @ -41,6 +41,7 @@ | |||||||
|     "dotenv": "^16.4.7", |     "dotenv": "^16.4.7", | ||||||
|     "drizzle-orm": "^0.41.0", |     "drizzle-orm": "^0.41.0", | ||||||
|     "express": "^4.21.2", |     "express": "^4.21.2", | ||||||
|  |     "ioredis": "^5.6.0", | ||||||
|     "lucide-react": "^0.483.0", |     "lucide-react": "^0.483.0", | ||||||
|     "mysql2": "^3.11.0", |     "mysql2": "^3.11.0", | ||||||
|     "next": "^15.2.3", |     "next": "^15.2.3", | ||||||
|  | |||||||
							
								
								
									
										61
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										61
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							| @ -71,6 +71,9 @@ importers: | |||||||
|       express: |       express: | ||||||
|         specifier: ^4.21.2 |         specifier: ^4.21.2 | ||||||
|         version: 4.21.2 |         version: 4.21.2 | ||||||
|  |       ioredis: | ||||||
|  |         specifier: ^5.6.0 | ||||||
|  |         version: 5.6.0 | ||||||
|       lucide-react: |       lucide-react: | ||||||
|         specifier: ^0.483.0 |         specifier: ^0.483.0 | ||||||
|         version: 0.483.0(react@19.0.0) |         version: 0.483.0(react@19.0.0) | ||||||
| @ -826,6 +829,9 @@ packages: | |||||||
|     cpu: [x64] |     cpu: [x64] | ||||||
|     os: [win32] |     os: [win32] | ||||||
| 
 | 
 | ||||||
|  |   '@ioredis/commands@1.2.0': | ||||||
|  |     resolution: {integrity: sha512-Sx1pU8EM64o2BrqNpEO1CNLtKQwyhuXuqyfH7oGKCk+1a33d2r5saW8zNwm3j6BTExtjrv2BxTgzzkMwts6vGg==} | ||||||
|  | 
 | ||||||
|   '@napi-rs/wasm-runtime@0.2.7': |   '@napi-rs/wasm-runtime@0.2.7': | ||||||
|     resolution: {integrity: sha512-5yximcFK5FNompXfJFoWanu5l8v1hNGqNHh9du1xETp9HWk/B/PzvchX55WYOPaIeNglG8++68AAiauBAtbnzw==} |     resolution: {integrity: sha512-5yximcFK5FNompXfJFoWanu5l8v1hNGqNHh9du1xETp9HWk/B/PzvchX55WYOPaIeNglG8++68AAiauBAtbnzw==} | ||||||
| 
 | 
 | ||||||
| @ -1691,6 +1697,10 @@ packages: | |||||||
|     resolution: {integrity: sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==} |     resolution: {integrity: sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==} | ||||||
|     engines: {node: '>=6'} |     engines: {node: '>=6'} | ||||||
| 
 | 
 | ||||||
|  |   cluster-key-slot@1.1.2: | ||||||
|  |     resolution: {integrity: sha512-RMr0FhtfXemyinomL4hrWcYJxmX6deFdCxpJzhDttxgO1+bcCnkk+9drydLVDmAMG7NE6aN/fl4F7ucU/90gAA==} | ||||||
|  |     engines: {node: '>=0.10.0'} | ||||||
|  | 
 | ||||||
|   color-convert@2.0.1: |   color-convert@2.0.1: | ||||||
|     resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} |     resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} | ||||||
|     engines: {node: '>=7.0.0'} |     engines: {node: '>=7.0.0'} | ||||||
| @ -2308,6 +2318,10 @@ packages: | |||||||
|     resolution: {integrity: sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==} |     resolution: {integrity: sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==} | ||||||
|     engines: {node: '>= 0.4'} |     engines: {node: '>= 0.4'} | ||||||
| 
 | 
 | ||||||
|  |   ioredis@5.6.0: | ||||||
|  |     resolution: {integrity: sha512-tBZlIIWbndeWBWCXWZiqtOF/yxf6yZX3tAlTJ7nfo5jhd6dctNxF7QnYlZLZ1a0o0pDoen7CgZqO+zjNaFbJAg==} | ||||||
|  |     engines: {node: '>=12.22.0'} | ||||||
|  | 
 | ||||||
|   ipaddr.js@1.9.1: |   ipaddr.js@1.9.1: | ||||||
|     resolution: {integrity: sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==} |     resolution: {integrity: sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==} | ||||||
|     engines: {node: '>= 0.10'} |     engines: {node: '>= 0.10'} | ||||||
| @ -2551,6 +2565,12 @@ packages: | |||||||
|     resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} |     resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} | ||||||
|     engines: {node: '>=10'} |     engines: {node: '>=10'} | ||||||
| 
 | 
 | ||||||
|  |   lodash.defaults@4.2.0: | ||||||
|  |     resolution: {integrity: sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==} | ||||||
|  | 
 | ||||||
|  |   lodash.isarguments@3.1.0: | ||||||
|  |     resolution: {integrity: sha512-chi4NHZlZqZD18a0imDHnZPrDeBbTtVN7GXMwuGdRH9qotxAjYs3aVLKc7zNOG9eddR5Ksd8rvFEBc9SsggPpg==} | ||||||
|  | 
 | ||||||
|   lodash.merge@4.6.2: |   lodash.merge@4.6.2: | ||||||
|     resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} |     resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} | ||||||
| 
 | 
 | ||||||
| @ -2950,6 +2970,14 @@ packages: | |||||||
|     resolution: {integrity: sha512-V8AVnmPIICiWpGfm6GLzCR/W5FXLchHop40W4nXBmdlEceh16rCN8O8LNWm5bh5XUX91fh7KpA+W0TgMKmgTpQ==} |     resolution: {integrity: sha512-V8AVnmPIICiWpGfm6GLzCR/W5FXLchHop40W4nXBmdlEceh16rCN8O8LNWm5bh5XUX91fh7KpA+W0TgMKmgTpQ==} | ||||||
|     engines: {node: '>=0.10.0'} |     engines: {node: '>=0.10.0'} | ||||||
| 
 | 
 | ||||||
|  |   redis-errors@1.2.0: | ||||||
|  |     resolution: {integrity: sha512-1qny3OExCf0UvUV/5wpYKf2YwPcOqXzkwKKSmKHiE6ZMQs5heeE/c8eXK+PNllPvmjgAbfnsbpkGZWy8cBpn9w==} | ||||||
|  |     engines: {node: '>=4'} | ||||||
|  | 
 | ||||||
|  |   redis-parser@3.0.0: | ||||||
|  |     resolution: {integrity: sha512-DJnGAeenTdpMEH6uAJRK/uiyEIH9WVsUmoLwzudwGJUwZPp80PDBWPHXSAGNPwNvIXAbe7MSUB1zQFugFml66A==} | ||||||
|  |     engines: {node: '>=4'} | ||||||
|  | 
 | ||||||
|   reflect.getprototypeof@1.0.10: |   reflect.getprototypeof@1.0.10: | ||||||
|     resolution: {integrity: sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==} |     resolution: {integrity: sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==} | ||||||
|     engines: {node: '>= 0.4'} |     engines: {node: '>= 0.4'} | ||||||
| @ -3102,6 +3130,9 @@ packages: | |||||||
|   stable-hash@0.0.5: |   stable-hash@0.0.5: | ||||||
|     resolution: {integrity: sha512-+L3ccpzibovGXFK+Ap/f8LOS0ahMrHTf3xu7mMLSpEGU0EO9ucaysSylKo9eRDFNhWve/y275iPmIZ4z39a9iA==} |     resolution: {integrity: sha512-+L3ccpzibovGXFK+Ap/f8LOS0ahMrHTf3xu7mMLSpEGU0EO9ucaysSylKo9eRDFNhWve/y275iPmIZ4z39a9iA==} | ||||||
| 
 | 
 | ||||||
|  |   standard-as-callback@2.1.0: | ||||||
|  |     resolution: {integrity: sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A==} | ||||||
|  | 
 | ||||||
|   statuses@2.0.1: |   statuses@2.0.1: | ||||||
|     resolution: {integrity: sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==} |     resolution: {integrity: sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==} | ||||||
|     engines: {node: '>= 0.8'} |     engines: {node: '>= 0.8'} | ||||||
| @ -3745,6 +3776,8 @@ snapshots: | |||||||
|   '@img/sharp-win32-x64@0.33.5': |   '@img/sharp-win32-x64@0.33.5': | ||||||
|     optional: true |     optional: true | ||||||
| 
 | 
 | ||||||
|  |   '@ioredis/commands@1.2.0': {} | ||||||
|  | 
 | ||||||
|   '@napi-rs/wasm-runtime@0.2.7': |   '@napi-rs/wasm-runtime@0.2.7': | ||||||
|     dependencies: |     dependencies: | ||||||
|       '@emnapi/core': 1.3.1 |       '@emnapi/core': 1.3.1 | ||||||
| @ -4592,6 +4625,8 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   clsx@2.1.1: {} |   clsx@2.1.1: {} | ||||||
| 
 | 
 | ||||||
|  |   cluster-key-slot@1.1.2: {} | ||||||
|  | 
 | ||||||
|   color-convert@2.0.1: |   color-convert@2.0.1: | ||||||
|     dependencies: |     dependencies: | ||||||
|       color-name: 1.1.4 |       color-name: 1.1.4 | ||||||
| @ -5367,6 +5402,20 @@ snapshots: | |||||||
|       hasown: 2.0.2 |       hasown: 2.0.2 | ||||||
|       side-channel: 1.1.0 |       side-channel: 1.1.0 | ||||||
| 
 | 
 | ||||||
|  |   ioredis@5.6.0: | ||||||
|  |     dependencies: | ||||||
|  |       '@ioredis/commands': 1.2.0 | ||||||
|  |       cluster-key-slot: 1.1.2 | ||||||
|  |       debug: 4.4.0 | ||||||
|  |       denque: 2.1.0 | ||||||
|  |       lodash.defaults: 4.2.0 | ||||||
|  |       lodash.isarguments: 3.1.0 | ||||||
|  |       redis-errors: 1.2.0 | ||||||
|  |       redis-parser: 3.0.0 | ||||||
|  |       standard-as-callback: 2.1.0 | ||||||
|  |     transitivePeerDependencies: | ||||||
|  |       - supports-color | ||||||
|  | 
 | ||||||
|   ipaddr.js@1.9.1: {} |   ipaddr.js@1.9.1: {} | ||||||
| 
 | 
 | ||||||
|   is-array-buffer@3.0.5: |   is-array-buffer@3.0.5: | ||||||
| @ -5593,6 +5642,10 @@ snapshots: | |||||||
|     dependencies: |     dependencies: | ||||||
|       p-locate: 5.0.0 |       p-locate: 5.0.0 | ||||||
| 
 | 
 | ||||||
|  |   lodash.defaults@4.2.0: {} | ||||||
|  | 
 | ||||||
|  |   lodash.isarguments@3.1.0: {} | ||||||
|  | 
 | ||||||
|   lodash.merge@4.6.2: {} |   lodash.merge@4.6.2: {} | ||||||
| 
 | 
 | ||||||
|   long@5.3.1: {} |   long@5.3.1: {} | ||||||
| @ -5902,6 +5955,12 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   react@19.0.0: {} |   react@19.0.0: {} | ||||||
| 
 | 
 | ||||||
|  |   redis-errors@1.2.0: {} | ||||||
|  | 
 | ||||||
|  |   redis-parser@3.0.0: | ||||||
|  |     dependencies: | ||||||
|  |       redis-errors: 1.2.0 | ||||||
|  | 
 | ||||||
|   reflect.getprototypeof@1.0.10: |   reflect.getprototypeof@1.0.10: | ||||||
|     dependencies: |     dependencies: | ||||||
|       call-bind: 1.0.8 |       call-bind: 1.0.8 | ||||||
| @ -6128,6 +6187,8 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   stable-hash@0.0.5: {} |   stable-hash@0.0.5: {} | ||||||
| 
 | 
 | ||||||
|  |   standard-as-callback@2.1.0: {} | ||||||
|  | 
 | ||||||
|   statuses@2.0.1: {} |   statuses@2.0.1: {} | ||||||
| 
 | 
 | ||||||
|   streamsearch@1.1.0: {} |   streamsearch@1.1.0: {} | ||||||
|  | |||||||
| @ -1,10 +1,7 @@ | |||||||
| import { api } from "@/trpc/server"; | import { api } from "@/trpc/server"; | ||||||
| import type { User } from "next-auth"; |  | ||||||
| import { notFound } from "next/navigation"; | import { notFound } from "next/navigation"; | ||||||
| import React from "react"; | import React from "react"; | ||||||
| import type { PublicUser } from "@/server/auth/config"; | import LobbyPage from "@/app/_components/lobby/lobby-page"; | ||||||
| import { auth } from "@/server/auth"; |  | ||||||
| import LobbyPage from "@/app/_components/lobby-page"; |  | ||||||
| 
 | 
 | ||||||
| async function Page({ | async function Page({ | ||||||
|   params, |   params, | ||||||
| @ -13,17 +10,12 @@ async function Page({ | |||||||
|     id: string; |     id: string; | ||||||
|   }>; |   }>; | ||||||
| }) { | }) { | ||||||
|   const session = await auth(); |   const sessionPlayer = await api.player.getBySession(); | ||||||
|   const { id } = await params; |   const { id } = await params; | ||||||
|   const lobby = await api.lobby.get({ id }); |   const lobby = await api.lobby.get({ id }); | ||||||
|   if (!lobby) return notFound(); |   if (!lobby) return notFound(); | ||||||
| 
 | 
 | ||||||
|   const members: Array<{ leader: boolean } & PublicUser> = [ |   return <LobbyPage lobby={lobby} sessionPlayer={sessionPlayer} />; | ||||||
|     { ...lobby.leader, leader: true }, |  | ||||||
|     ...(lobby?.members?.map(({ user }) => ({ ...user, leader: false })) ?? []), |  | ||||||
|   ]; |  | ||||||
| 
 |  | ||||||
|   return <LobbyPage lobby={lobby} initialMembers={members} session={session} />; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export default Page; | export default Page; | ||||||
|  | |||||||
| @ -1,36 +1,29 @@ | |||||||
| import React from "react"; | import React from "react"; | ||||||
| import { api } from "@/trpc/server"; | import { api } from "@/trpc/server"; | ||||||
| import CreateLobbyDialog from "@/app/_components/create-lobby-dialog"; | import CreateLobbyDialog from "@/app/_components/lobby/create-lobby-dialog"; | ||||||
| import LobbyCard from "@/app/_components/lobby-card"; |  | ||||||
| import { Button } from "@/components/ui/button"; | import { Button } from "@/components/ui/button"; | ||||||
|  | import LobbyPage from "@/app/_components/lobby/lobby-page"; | ||||||
|  | import { redirect } from "next/navigation"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
| 
 | 
 | ||||||
| async function Page() { | async function Page() { | ||||||
|   const lobbies = await api.lobby.getAll(); |   const sessionPlayer = await api.player.getBySession(); | ||||||
| 
 |   if (!sessionPlayer) return redirect(appRoutes.signIn); | ||||||
|   return ( |   const lobby = sessionPlayer ? await api.lobby.getCurrentLobby() : null; | ||||||
|     <> |   if (!lobby) | ||||||
|       <div className="w-full max-w-md space-y-4"> |     return ( | ||||||
|         <div className="flex w-full gap-4"> |       <div className="flex w-full gap-4"> | ||||||
|           <CreateLobbyDialog className="grow" /> |         <CreateLobbyDialog className="grow" /> | ||||||
|           <Button |         <Button | ||||||
|             variant={"party"} |           variant={"party"} | ||||||
|             size={"xxl"} |           size={"xxl"} | ||||||
|             className="container-bg bg-border/10" |           className="container-bg bg-border/10" | ||||||
|           > |         > | ||||||
|             Placeholder |           Placeholder | ||||||
|           </Button> |         </Button> | ||||||
|         </div> |  | ||||||
| 
 |  | ||||||
|         <menu> |  | ||||||
|           {lobbies.map((lobby) => ( |  | ||||||
|             <li key={lobby.id}> |  | ||||||
|               <LobbyCard lobby={lobby} /> |  | ||||||
|             </li> |  | ||||||
|           ))} |  | ||||||
|         </menu> |  | ||||||
|       </div> |       </div> | ||||||
|     </> |     ); | ||||||
|   ); |   return <LobbyPage lobby={lobby} sessionPlayer={sessionPlayer} />; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export default Page; | export default Page; | ||||||
|  | |||||||
							
								
								
									
										7
									
								
								src/app/(routes)/me/edit/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/app/(routes)/me/edit/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | import React from "react"; | ||||||
|  | 
 | ||||||
|  | function page() { | ||||||
|  |   return <div>Session player profile edit page</div>; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default page; | ||||||
							
								
								
									
										7
									
								
								src/app/(routes)/me/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/app/(routes)/me/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | import React from "react"; | ||||||
|  | 
 | ||||||
|  | function Page() { | ||||||
|  |   return <div>Session player profile </div>; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Page; | ||||||
| @ -1,13 +1,17 @@ | |||||||
| import Link from "next/link"; | import Link from "next/link"; | ||||||
| import { Button } from "@/components/ui/button"; | import { Button } from "@/components/ui/button"; | ||||||
| import { Sparkles, Users, Plus } from "lucide-react"; | import { Sparkles, Users, Plus } from "lucide-react"; | ||||||
| import CreateLobbyDialog from "../_components/create-lobby-dialog"; | import CreateLobbyDialog from "../_components/lobby/create-lobby-dialog"; | ||||||
| import { auth } from "@/server/auth"; | import { auth } from "@/server/auth"; | ||||||
| import { Icons } from "@/components/icons"; | import { Icons } from "@/components/icons"; | ||||||
| import { appConfig } from "@/app.config"; | import { appConfig } from "@/config/app.config"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
| 
 | 
 | ||||||
| export default async function QuizGameStartPage() { | export default async function QuizGameStartPage() { | ||||||
|   const session = await auth(); |   const session = await auth(); | ||||||
|  |   const currentLobby = session ? await api.lobby.getCurrentLobby() : null; | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       <div className="mb-8 flex flex-col items-center justify-center gap-2 pt-12 text-center"> |       <div className="mb-8 flex flex-col items-center justify-center gap-2 pt-12 text-center"> | ||||||
| @ -21,43 +25,47 @@ export default async function QuizGameStartPage() { | |||||||
|         </p> |         </p> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       {/* Game card */} |       <div className="container-bg w-full max-w-md p-8"> | ||||||
|       <div className="container-bg max-w-md p-8"> |         {currentLobby ? ( | ||||||
|         {/* Create Lobby Button */} |           <Button variant={"party"} size={"xxl"} className="w-full" asChild> | ||||||
|         <div className="space-y-6"> |             <Link href={appRoutes.currentlobby}>Jump to Lobby</Link> | ||||||
|           {session ? ( |           </Button> | ||||||
|             <CreateLobbyDialog className="text-shadow-primary w-full" /> |         ) : ( | ||||||
|           ) : ( |           <div className="space-y-6"> | ||||||
|  |             {session ? ( | ||||||
|  |               <CreateLobbyDialog className="text-shadow-primary w-full" /> | ||||||
|  |             ) : ( | ||||||
|  |               <Button | ||||||
|  |                 size={"xxl"} | ||||||
|  |                 variant={"party"} | ||||||
|  |                 asChild | ||||||
|  |                 className="text-shadow-primary w-full font-black uppercase" | ||||||
|  |               > | ||||||
|  |                 <Link href={"/api/auth/signin"}>Sign-in to create a Lobby</Link> | ||||||
|  |               </Button> | ||||||
|  |             )} | ||||||
|  | 
 | ||||||
|  |             {/* Join Lobby Button */} | ||||||
|             <Button |             <Button | ||||||
|               size={"xxl"} |               size={"xxl"} | ||||||
|               variant={"party"} |               variant={"party"} | ||||||
|               asChild |               className="w-full rounded-xl bg-gradient-to-r from-blue-500 via-blue-600 to-indigo-600 text-white hover:from-blue-600 hover:to-indigo-700" | ||||||
|               className="text-shadow-primary w-full font-black uppercase" |  | ||||||
|             > |             > | ||||||
|               <Link href={"/api/auth/signin"}>Sign-in to create a Lobby</Link> |               <Users className="mr-2 h-6 w-6" /> | ||||||
|  |               Join Lobby | ||||||
|             </Button> |             </Button> | ||||||
|           )} |  | ||||||
| 
 | 
 | ||||||
|           {/* Join Lobby Button */} |             {/* Quick Play Option */} | ||||||
|           <Button |             <div className="border-t border-white/10 pt-4"> | ||||||
|             size={"xxl"} |               <Button | ||||||
|             variant={"party"} |                 variant="ghost" | ||||||
|             className="w-full rounded-xl bg-gradient-to-r from-blue-500 via-blue-600 to-indigo-600 text-white hover:from-blue-600 hover:to-indigo-700" |                 className="w-full text-indigo-200 hover:bg-white/10 hover:text-white" | ||||||
|           > |               > | ||||||
|             <Users className="mr-2 h-6 w-6" /> |                 Quick Play | ||||||
|             Join Lobby |               </Button> | ||||||
|           </Button> |             </div> | ||||||
| 
 |  | ||||||
|           {/* Quick Play Option */} |  | ||||||
|           <div className="border-t border-white/10 pt-4"> |  | ||||||
|             <Button |  | ||||||
|               variant="ghost" |  | ||||||
|               className="w-full text-indigo-200 hover:bg-white/10 hover:text-white" |  | ||||||
|             > |  | ||||||
|               Quick Play |  | ||||||
|             </Button> |  | ||||||
|           </div> |           </div> | ||||||
|         </div> |         )} | ||||||
|       </div> |       </div> | ||||||
|     </> |     </> | ||||||
|   ); |   ); | ||||||
|  | |||||||
| @ -1,9 +0,0 @@ | |||||||
| import React from 'react' |  | ||||||
| 
 |  | ||||||
| function page() { |  | ||||||
|   return ( |  | ||||||
|     <div>page</div> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export default page |  | ||||||
| @ -1,57 +0,0 @@ | |||||||
| "use client"; |  | ||||||
| 
 |  | ||||||
| import React from "react"; |  | ||||||
| import type { PublicUser } from "@/server/auth/config"; |  | ||||||
| import type { Lobby } from "@/server/db/schema"; |  | ||||||
| import UserCard from "@/components/user-card"; |  | ||||||
| import DeleteLobbyDialog from "@/app/_components/delete-lobby-dialog"; |  | ||||||
| import LobbyMembershipDialog from "@/app/_components/lobby-membership-dialog"; |  | ||||||
| import { type Session } from "next-auth"; |  | ||||||
| import { Badge } from "@/components/ui/badge"; |  | ||||||
| 
 |  | ||||||
| function LobbyPage({ |  | ||||||
|   session, |  | ||||||
|   initialMembers, |  | ||||||
|   lobby, |  | ||||||
| }: { |  | ||||||
|   session: Session | null; |  | ||||||
|   lobby: Pick<Lobby, "id" | "name" | "createdAt" | "createdById">; |  | ||||||
|   initialMembers: Array<{ leader: boolean } & PublicUser>; |  | ||||||
| }) { |  | ||||||
|   const [members, setMembers] = React.useState(initialMembers); |  | ||||||
|   const [memberPresence, setMemberPresence] = React.useState<Array<string>>(); |  | ||||||
|   const isJoined = members.find((member) => member.id === session?.user.id); |  | ||||||
|   const isOwner = lobby.createdById === session?.user.id; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|   return ( |  | ||||||
|     <div className="container-bg w-full max-w-md space-y-4 p-6"> |  | ||||||
|       <h1 className="text-2xl font-bold capitalize">{lobby.name}</h1> |  | ||||||
|       <ul className="space-y-2"> |  | ||||||
|         {members?.map((member, idx) => ( |  | ||||||
|           <li key={idx}> |  | ||||||
|             <UserCard |  | ||||||
|               name={member.name!} |  | ||||||
|               image={member.image!} |  | ||||||
|               className="relative" |  | ||||||
|             > |  | ||||||
|               {member?.leader && ( |  | ||||||
|                 <Badge className="absolute -top-2 right-2 p-px px-2 text-white"> |  | ||||||
|                   Leader |  | ||||||
|                 </Badge> |  | ||||||
|               )} |  | ||||||
|             </UserCard> |  | ||||||
|           </li> |  | ||||||
|         ))} |  | ||||||
|       </ul> |  | ||||||
|       {JSON.stringify(memberPresence)} |  | ||||||
|       {isOwner && <DeleteLobbyDialog lobbyId={lobby.id} />} |  | ||||||
|       {!isOwner && ( |  | ||||||
|         <LobbyMembershipDialog lobbyId={lobby.id} join={!isJoined} /> |  | ||||||
|       )} |  | ||||||
|     </div> |  | ||||||
|   ); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export default LobbyPage; |  | ||||||
| @ -12,7 +12,7 @@ import { | |||||||
|   DialogTitle, |   DialogTitle, | ||||||
|   DialogTrigger, |   DialogTrigger, | ||||||
| } from "@/components/ui/dialog"; | } from "@/components/ui/dialog"; | ||||||
| import LobbyForm from "./lobby/lobby-form"; | import LobbyForm from "./lobby-form"; | ||||||
| 
 | 
 | ||||||
| function CreateLobbyDialog({ className }: { className?: string }) { | function CreateLobbyDialog({ className }: { className?: string }) { | ||||||
|   const [open, setOpen] = React.useState(false); |   const [open, setOpen] = React.useState(false); | ||||||
| @ -16,6 +16,7 @@ import { | |||||||
| import { api } from "@/trpc/react"; | import { api } from "@/trpc/react"; | ||||||
| import { toast } from "sonner"; | import { toast } from "sonner"; | ||||||
| import { useRouter } from "next/navigation"; | import { useRouter } from "next/navigation"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
| 
 | 
 | ||||||
| function DeleteLobbyDialog({ lobbyId }: { lobbyId: string }) { | function DeleteLobbyDialog({ lobbyId }: { lobbyId: string }) { | ||||||
|   const [loading, setLoading] = React.useState(false); |   const [loading, setLoading] = React.useState(false); | ||||||
| @ -25,7 +26,8 @@ function DeleteLobbyDialog({ lobbyId }: { lobbyId: string }) { | |||||||
|     setLoading(true); |     setLoading(true); | ||||||
|     const result = await mutateAsync({ lobbyId }); |     const result = await mutateAsync({ lobbyId }); | ||||||
|     if (result) { |     if (result) { | ||||||
|       router.push("/"); |       router.push(appRoutes.home); | ||||||
|  |       router.refresh(); | ||||||
|     } else toast.error("Something went wrong"); |     } else toast.error("Something went wrong"); | ||||||
|     setLoading(false); |     setLoading(false); | ||||||
|   }; |   }; | ||||||
| @ -33,7 +35,7 @@ function DeleteLobbyDialog({ lobbyId }: { lobbyId: string }) { | |||||||
|   return ( |   return ( | ||||||
|     <AlertDialog> |     <AlertDialog> | ||||||
|       <Button asChild variant={"destructive"}> |       <Button asChild variant={"destructive"}> | ||||||
|         <AlertDialogTrigger>Remove Lobby</AlertDialogTrigger> |         <AlertDialogTrigger>Delete Lobby</AlertDialogTrigger> | ||||||
|       </Button> |       </Button> | ||||||
|       <AlertDialogContent> |       <AlertDialogContent> | ||||||
|         <AlertDialogHeader> |         <AlertDialogHeader> | ||||||
| @ -9,10 +9,11 @@ import { | |||||||
| import Link from "next/link"; | import Link from "next/link"; | ||||||
| import { formatDate } from "@/lib/utils"; | import { formatDate } from "@/lib/utils"; | ||||||
| import { ArrowRight, ChevronRight } from "lucide-react"; | import { ArrowRight, ChevronRight } from "lucide-react"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
| 
 | 
 | ||||||
| function LobbyCard({ lobby }: { lobby: Lobby }) { | function LobbyCard({ lobby }: { lobby: Lobby }) { | ||||||
|   return ( |   return ( | ||||||
|     <Link href={`/lobby/${lobby.id}`}> |     <Link href={appRoutes.lobby(lobby.id)}> | ||||||
|       <Card className="group border-none bg-transparent p-0 shadow-none"> |       <Card className="group border-none bg-transparent p-0 shadow-none"> | ||||||
|         <div className="container-bg py-4"> |         <div className="container-bg py-4"> | ||||||
|           <CardHeader className="flex w-full items-center justify-between"> |           <CardHeader className="flex w-full items-center justify-between"> | ||||||
| @ -19,6 +19,7 @@ import { toast } from "sonner"; | |||||||
| import { api } from "@/trpc/react"; | import { api } from "@/trpc/react"; | ||||||
| import { useRouter } from "next/navigation"; | import { useRouter } from "next/navigation"; | ||||||
| import { Button } from "@/components/ui/button"; | import { Button } from "@/components/ui/button"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
| 
 | 
 | ||||||
| function LobbyForm({ | function LobbyForm({ | ||||||
|   server_lobby, |   server_lobby, | ||||||
| @ -34,7 +35,8 @@ 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: "", |       name: server_lobby?.name ?? "", | ||||||
|  |       maxPlayers: server_lobby?.maxPlayers ?? 0, | ||||||
|     }, |     }, | ||||||
|   }); |   }); | ||||||
|   async function onSubmit(lobby: z.infer<typeof lobbyPatchSchema>) { |   async function onSubmit(lobby: z.infer<typeof lobbyPatchSchema>) { | ||||||
| @ -45,7 +47,10 @@ function LobbyForm({ | |||||||
|       : await createLobby({ lobby }); |       : await createLobby({ lobby }); | ||||||
|     cb?.(); |     cb?.(); | ||||||
|     if (result) { |     if (result) { | ||||||
|       if (!existingLobby) router.push(`/lobby/${result.id}`); |       if (!existingLobby) { | ||||||
|  |         router.push(appRoutes.currentlobby); | ||||||
|  |         router.refresh(); | ||||||
|  |       } | ||||||
|     } else toast.error("Something went wrong."); |     } else toast.error("Something went wrong."); | ||||||
|     setLoading(false); |     setLoading(false); | ||||||
|   } |   } | ||||||
| @ -70,9 +75,15 @@ function LobbyForm({ | |||||||
|             </FormItem> |             </FormItem> | ||||||
|           )} |           )} | ||||||
|         /> |         /> | ||||||
|         <Button type="submit" disabled={loading || !form.formState.isDirty}> |         <div className="flex items-center justify-end"> | ||||||
|           Create Lobby |           <Button | ||||||
|         </Button> |             type="submit" | ||||||
|  |             disabled={loading || !form.formState.isDirty} | ||||||
|  |             className="ml-auto" | ||||||
|  |           > | ||||||
|  |             {server_lobby?.id?.length ? "Update" : "Create"} Lobby | ||||||
|  |           </Button> | ||||||
|  |         </div> | ||||||
|       </form> |       </form> | ||||||
|     </Form> |     </Form> | ||||||
|   ); |   ); | ||||||
|  | |||||||
| @ -15,7 +15,6 @@ import { | |||||||
| } from "@/components/ui/alert-dialog"; | } from "@/components/ui/alert-dialog"; | ||||||
| import { api } from "@/trpc/react"; | import { api } from "@/trpc/react"; | ||||||
| import { toast } from "sonner"; | import { toast } from "sonner"; | ||||||
| import { useRouter } from "next/navigation"; |  | ||||||
| 
 | 
 | ||||||
| function LobbyMembershipDialog({ | function LobbyMembershipDialog({ | ||||||
|   lobbyId, |   lobbyId, | ||||||
| @ -25,23 +24,21 @@ function LobbyMembershipDialog({ | |||||||
|   join: boolean; |   join: boolean; | ||||||
| }) { | }) { | ||||||
|   const [loading, setLoading] = React.useState(false); |   const [loading, setLoading] = React.useState(false); | ||||||
|   const { mutateAsync } = api.lobby.membership.useMutation(); |   const membership = api.lobby.membership.useMutation(); | ||||||
|   const router = useRouter(); |   const labelText = join ? "join" : "leave"; | ||||||
|  | 
 | ||||||
|   const handleConfirm = async () => { |   const handleConfirm = async () => { | ||||||
|     setLoading(true); |     setLoading(true); | ||||||
|     const result = await mutateAsync({ lobbyId, join }); |     const result = await membership.mutateAsync({ lobbyId, join }); | ||||||
|     if (result) { |     if (result) { | ||||||
|       if (!join) router.push("/"); |       toast.success(`Successfully ${labelText} the lobby.`); | ||||||
|       else toast.success("Successfully joined the lobby."); |  | ||||||
|     } else toast.error("Something went wrong"); |     } else toast.error("Something went wrong"); | ||||||
|     setLoading(false); |     setLoading(false); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const labelText = join ? "join" : "leave"; |  | ||||||
| 
 |  | ||||||
|   return ( |   return ( | ||||||
|     <AlertDialog> |     <AlertDialog> | ||||||
|       <Button asChild> |       <Button asChild variant={join ? "default" : "destructive"}> | ||||||
|         <AlertDialogTrigger className="capitalize"> |         <AlertDialogTrigger className="capitalize"> | ||||||
|           {labelText} Lobby |           {labelText} Lobby | ||||||
|         </AlertDialogTrigger> |         </AlertDialogTrigger> | ||||||
| @ -62,7 +59,7 @@ function LobbyMembershipDialog({ | |||||||
|             disabled={loading} |             disabled={loading} | ||||||
|             className="capitalize" |             className="capitalize" | ||||||
|           > |           > | ||||||
|             {labelText} |             {labelText} Lobby | ||||||
|           </AlertDialogAction> |           </AlertDialogAction> | ||||||
|         </AlertDialogFooter> |         </AlertDialogFooter> | ||||||
|       </AlertDialogContent> |       </AlertDialogContent> | ||||||
							
								
								
									
										82
									
								
								src/app/_components/lobby/lobby-page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								src/app/_components/lobby/lobby-page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,82 @@ | |||||||
|  | "use client"; | ||||||
|  | 
 | ||||||
|  | import React from "react"; | ||||||
|  | import type { Lobby, LobbyMember, Player } from "@/server/db/schema"; | ||||||
|  | import LobbyPlayerCard from "@/app/_components/lobby/lobby_player-card"; | ||||||
|  | import DeleteLobbyDialog from "@/app/_components/lobby/delete-lobby-dialog"; | ||||||
|  | import LobbyMembershipDialog from "@/app/_components/lobby/lobby-membership-dialog"; | ||||||
|  | import { Badge } from "@/components/ui/badge"; | ||||||
|  | import { api } from "@/trpc/react"; | ||||||
|  | import CopyToClip from "@/components/copy-to-clip"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
|  | import LobbySettingsDialog from "./lobby-settings-dialog"; | ||||||
|  | 
 | ||||||
|  | function LobbyPage({ | ||||||
|  |   sessionPlayer, | ||||||
|  | 
 | ||||||
|  |   lobby, | ||||||
|  | }: { | ||||||
|  |   sessionPlayer?: Player | null; | ||||||
|  |   lobby: Lobby; | ||||||
|  | }) { | ||||||
|  |   const [members, setMembers] = React.useState<Array<LobbyMember>>( | ||||||
|  |     lobby?.members ?? [], | ||||||
|  |   ); | ||||||
|  |   api.lobby.onMemberUpdate.useSubscription(undefined, { | ||||||
|  |     onData({ data }) { | ||||||
|  |       if (data.joined) | ||||||
|  |         setMembers((prev) => [...prev, data.membership as LobbyMember]); | ||||||
|  |       else | ||||||
|  |         setMembers((prev) => | ||||||
|  |           prev.filter((m) => m.playerId !== data.membership), | ||||||
|  |         ); | ||||||
|  |       console.log("Data", data); | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const isJoined = members.find((m) => m.playerId === sessionPlayer?.id); | ||||||
|  |   const isAdmin = isJoined?.role === "admin"; | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <div className="container-bg w-full max-w-md space-y-4 p-6"> | ||||||
|  |       <div className="flex items-center justify-between"> | ||||||
|  |         <h1 className="w-full overflow-hidden text-2xl font-bold text-ellipsis capitalize"> | ||||||
|  |           {lobby.name} | ||||||
|  |         </h1> | ||||||
|  |         {isAdmin && <LobbySettingsDialog lobby={lobby} />} | ||||||
|  |       </div> | ||||||
|  | 
 | ||||||
|  |       <ul className="space-y-2"> | ||||||
|  |         {members?.map((member, idx) => ( | ||||||
|  |           <li key={idx}> | ||||||
|  |             <LobbyPlayerCard player={member?.player!} className="relative"> | ||||||
|  |               {member?.role === "admin" && ( | ||||||
|  |                 <Badge className="absolute -top-2 right-2 p-px px-2 text-white"> | ||||||
|  |                   Admin | ||||||
|  |                 </Badge> | ||||||
|  |               )} | ||||||
|  |             </LobbyPlayerCard> | ||||||
|  |           </li> | ||||||
|  |         ))} | ||||||
|  |       </ul> | ||||||
|  |       <div className="flex items-center gap-2"> | ||||||
|  |         {isJoined && ( | ||||||
|  |           <CopyToClip | ||||||
|  |             target="invite link" | ||||||
|  |             text={appRoutes.lobby(lobby.id)} | ||||||
|  |             buttonProps={{ variant: "outline" }} | ||||||
|  |           > | ||||||
|  |             Invite Players | ||||||
|  |           </CopyToClip> | ||||||
|  |         )} | ||||||
|  |         <div className="ml-auto"> | ||||||
|  |           {sessionPlayer && ( | ||||||
|  |             <LobbyMembershipDialog lobbyId={lobby.id} join={!isJoined} /> | ||||||
|  |           )} | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default LobbyPage; | ||||||
							
								
								
									
										41
									
								
								src/app/_components/lobby/lobby-settings-dialog.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								src/app/_components/lobby/lobby-settings-dialog.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | "use client"; | ||||||
|  | import { Button } from "@/components/ui/button"; | ||||||
|  | import { | ||||||
|  |   Dialog, | ||||||
|  |   DialogContent, | ||||||
|  |   DialogTitle, | ||||||
|  |   DialogTrigger, | ||||||
|  | } from "@/components/ui/dialog"; | ||||||
|  | import type { Lobby } from "@/server/db/schema"; | ||||||
|  | import { DialogDescription } from "@radix-ui/react-dialog"; | ||||||
|  | import React from "react"; | ||||||
|  | import LobbyForm from "./lobby-form"; | ||||||
|  | import DeleteLobbyDialog from "./delete-lobby-dialog"; | ||||||
|  | import { Settings } from "lucide-react"; | ||||||
|  | 
 | ||||||
|  | function LobbySettingsDialog({ lobby }: { lobby: Lobby }) { | ||||||
|  |   return ( | ||||||
|  |     <Dialog> | ||||||
|  |       <DialogTrigger asChild> | ||||||
|  |         <Button variant={"ghost"} size={"icon"}> | ||||||
|  |           <Settings className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |       </DialogTrigger> | ||||||
|  |       <DialogContent> | ||||||
|  |         <DialogTitle>Lobby Settings</DialogTitle> | ||||||
|  |         <DialogDescription></DialogDescription> | ||||||
|  | 
 | ||||||
|  |         <LobbyForm server_lobby={lobby} /> | ||||||
|  |         <div className="flex items-center"> | ||||||
|  |           <div className="bg-muted-foreground h-px w-12" /> | ||||||
|  |           <p className="relative z-10 px-2 text-xs">Danger Zone</p> | ||||||
|  |           <div className="bg-muted-foreground h-px grow" /> | ||||||
|  |         </div> | ||||||
|  | 
 | ||||||
|  |         <DeleteLobbyDialog lobbyId={lobby.id} /> | ||||||
|  |       </DialogContent> | ||||||
|  |     </Dialog> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default LobbySettingsDialog; | ||||||
| @ -1,17 +1,16 @@ | |||||||
| import React from "react"; | import React from "react"; | ||||||
| import Avatar from "./avatar"; | import Avatar from "../../../components/avatar"; | ||||||
| import { cn } from "@/lib/utils"; | import { cn } from "@/lib/utils"; | ||||||
| import { MoreHorizontal } from "lucide-react"; | import { MoreHorizontal } from "lucide-react"; | ||||||
| import { Button } from "./ui/button"; | import { Button } from "../../../components/ui/button"; | ||||||
|  | import type { Player } from "@/server/db/schema"; | ||||||
| 
 | 
 | ||||||
| function UserCard({ | function LobbyPlayerCard({ | ||||||
|   name, |   player, | ||||||
|   image, |  | ||||||
|   children, |   children, | ||||||
|   className, |   className, | ||||||
| }: { | }: { | ||||||
|   name: string; |   player: Pick<Player, "displayName" | "avatar">; | ||||||
|   image: string; |  | ||||||
|   children?: React.ReactNode; |   children?: React.ReactNode; | ||||||
|   className?: string; |   className?: string; | ||||||
| }) { | }) { | ||||||
| @ -23,8 +22,8 @@ function UserCard({ | |||||||
|       )} |       )} | ||||||
|     > |     > | ||||||
|       <div className="flex items-center gap-2"> |       <div className="flex items-center gap-2"> | ||||||
|         <Avatar src={image} fb={name} /> |         <Avatar src={player.avatar!} fb={player.displayName!} /> | ||||||
|         <h4 className="font-meidum text-xl">{name}</h4> |         <h4 className="font-meidum text-xl">{player.displayName}</h4> | ||||||
|       </div> |       </div> | ||||||
|       {children} |       {children} | ||||||
|       <Button |       <Button | ||||||
| @ -38,4 +37,4 @@ function UserCard({ | |||||||
|   ); |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export default UserCard; | export default LobbyPlayerCard; | ||||||
| @ -8,36 +8,37 @@ import { | |||||||
|   DropdownMenuSeparator, |   DropdownMenuSeparator, | ||||||
|   DropdownMenuTrigger, |   DropdownMenuTrigger, | ||||||
| } from "@/components/ui/dropdown-menu"; | } from "@/components/ui/dropdown-menu"; | ||||||
| import type { User } from "next-auth"; |  | ||||||
| import Avatar from "@/components/avatar"; | import Avatar from "@/components/avatar"; | ||||||
| import { Edit, Eye, LogOut } from "lucide-react"; | import { Edit, Eye, LogOut } from "lucide-react"; | ||||||
| import { cn } from "@/lib/utils"; | import { cn } from "@/lib/utils"; | ||||||
|  | import type { Player } from "@/server/db/schema"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
| 
 | 
 | ||||||
| function UserPopover({ user }: { user: User }) { | function UserPopover({ player }: { player: Player }) { | ||||||
|   const dropdownItemClassName = |   const dropdownItemClassName = | ||||||
|     "focus:bg-border/30 focus:text-white flex items-center gap-1 "; |     "focus:bg-border/30 focus:text-white flex items-center gap-1 "; | ||||||
|   return ( |   return ( | ||||||
|     <DropdownMenu> |     <DropdownMenu> | ||||||
|       <DropdownMenuTrigger> |       <DropdownMenuTrigger> | ||||||
|         <Avatar |         <Avatar | ||||||
|           src={user.image!} |           src={player.avatar!} | ||||||
|           fb={user.name!} |           fb={player.displayName!} | ||||||
|           className="border-border/20 size-10 cursor-pointer border-2" |           className="border-border/20 size-10 cursor-pointer border-2" | ||||||
|         /> |         /> | ||||||
|       </DropdownMenuTrigger> |       </DropdownMenuTrigger> | ||||||
|       <DropdownMenuContent className="bg-border/30 container-bg text-white"> |       <DropdownMenuContent className="bg-border/30 container-bg text-white"> | ||||||
|         <DropdownMenuLabel className="w-32 truncate overflow-hidden font-bold"> |         <DropdownMenuLabel className="w-32 truncate overflow-hidden font-bold"> | ||||||
|           {user.name} |           {player.displayName} | ||||||
|         </DropdownMenuLabel> |         </DropdownMenuLabel> | ||||||
|         <DropdownMenuSeparator /> |         <DropdownMenuSeparator /> | ||||||
|         <DropdownMenuItem className={dropdownItemClassName} asChild> |         <DropdownMenuItem className={dropdownItemClassName} asChild> | ||||||
|           <Link href={`/profile/${user.id}`}> |           <Link href={appRoutes.me}> | ||||||
|             <Eye className="size-4 text-white" /> |             <Eye className="size-4 text-white" /> | ||||||
|             View Profile |             View Profile | ||||||
|           </Link> |           </Link> | ||||||
|         </DropdownMenuItem> |         </DropdownMenuItem> | ||||||
|         <DropdownMenuItem className={dropdownItemClassName} asChild> |         <DropdownMenuItem className={dropdownItemClassName} asChild> | ||||||
|           <Link href={`/profile/${user.id}/edit`}> |           <Link href={appRoutes.editProfile}> | ||||||
|             <Edit className="size-4 text-white" /> |             <Edit className="size-4 text-white" /> | ||||||
|             Edit Profile |             Edit Profile | ||||||
|           </Link> |           </Link> | ||||||
| @ -50,7 +51,7 @@ function UserPopover({ user }: { user: User }) { | |||||||
|           )} |           )} | ||||||
|           asChild |           asChild | ||||||
|         > |         > | ||||||
|           <Link href="/api/auth/signout"> |           <Link href={appRoutes.signOut}> | ||||||
|             <LogOut className="group-focus:text-destructive size-4 text-white" /> |             <LogOut className="group-focus:text-destructive size-4 text-white" /> | ||||||
|             Log out |             Log out | ||||||
|           </Link> |           </Link> | ||||||
							
								
								
									
										28
									
								
								src/components/copy-to-clip.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/components/copy-to-clip.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | |||||||
|  | "use client"; | ||||||
|  | import React from "react"; | ||||||
|  | import { Button, type ButtonProps } from "./ui/button"; | ||||||
|  | import { toast } from "sonner"; | ||||||
|  | 
 | ||||||
|  | function CopyToClip({ | ||||||
|  |   text, | ||||||
|  |   target, | ||||||
|  |   buttonProps, | ||||||
|  |   children, | ||||||
|  | }: { | ||||||
|  |   text: string; | ||||||
|  |   target?: string; | ||||||
|  |   buttonProps?: ButtonProps; | ||||||
|  |   children?: React.ReactNode; | ||||||
|  | }) { | ||||||
|  |   const handleclick = () => { | ||||||
|  |     navigator.clipboard.writeText(text); | ||||||
|  |     toast.success(`Copied ${target ?? ""} to clipboard`); | ||||||
|  |   }; | ||||||
|  |   return ( | ||||||
|  |     <Button {...buttonProps} onClick={handleclick}> | ||||||
|  |       {children} | ||||||
|  |     </Button> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default CopyToClip; | ||||||
| @ -1,7 +1,7 @@ | |||||||
| import Link from "next/link"; | import Link from "next/link"; | ||||||
| import React from "react"; | import React from "react"; | ||||||
| import { Icons } from "./icons"; | import { Icons } from "./icons"; | ||||||
| import { appConfig } from "@/app.config"; | import { appConfig } from "@/config/app.config"; | ||||||
| 
 | 
 | ||||||
| function Footer() { | function Footer() { | ||||||
|   return ( |   return ( | ||||||
|  | |||||||
| @ -1,13 +1,16 @@ | |||||||
| import React from "react"; | import React from "react"; | ||||||
| import { auth } from "@/server/auth"; |  | ||||||
| import { Button } from "./ui/button"; | import { Button } from "./ui/button"; | ||||||
| import Link from "next/link"; | import Link from "next/link"; | ||||||
| import UserPopover from "@/app/_components/user-popover"; | import UserPopover from "@/app/_components/profile-popover"; | ||||||
| import { appConfig } from "@/app.config"; | import { appConfig } from "@/config/app.config"; | ||||||
| import NavLink from "./nav-link"; | import NavLink from "./nav-link"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | import { Icons } from "./icons"; | ||||||
|  | import { appRoutes } from "@/config/app.routes"; | ||||||
| 
 | 
 | ||||||
| async function Navbar() { | async function Navbar() { | ||||||
|   const session = await auth(); |   const sessionPlayer = await api.player.getBySession(); | ||||||
|  |   const currentLobby = sessionPlayer ? await api.lobby.getCurrentLobby() : null; | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <nav className="flex w-full items-center justify-center p-4"> |     <nav className="flex w-full items-center justify-center p-4"> | ||||||
| @ -19,14 +22,24 @@ async function Navbar() { | |||||||
|             </li> |             </li> | ||||||
|           ))} |           ))} | ||||||
|         </menu> |         </menu> | ||||||
|         {/* Login Button or Profile Avatar */} | 
 | ||||||
|         {session?.user ? ( |         <div className="bg-border/10 border-border/20 flex items-center rounded-full border-l-2"> | ||||||
|           <UserPopover user={session.user} /> |           {currentLobby ? ( | ||||||
|         ) : ( |             <Link href={appRoutes.currentlobby}> | ||||||
|           <Button asChild className="bg-primary font-black"> |               <div className="border-border/20 flex h-10 items-center justify-center gap-2 pr-2 pl-4 font-semibold"> | ||||||
|             <Link href="/api/auth/signin">Sign In</Link> |                 <Icons.logo className="size-4" /> | ||||||
|           </Button> |                 <span>Lobby</span> | ||||||
|         )} |               </div> | ||||||
|  |             </Link> | ||||||
|  |           ) : null} | ||||||
|  |           {sessionPlayer ? ( | ||||||
|  |             <UserPopover player={sessionPlayer} /> | ||||||
|  |           ) : ( | ||||||
|  |             <Button asChild className="bg-primary font-black"> | ||||||
|  |               <Link href="/api/auth/signin">Sign In</Link> | ||||||
|  |             </Button> | ||||||
|  |           )} | ||||||
|  |         </div> | ||||||
|       </div> |       </div> | ||||||
|     </nav> |     </nav> | ||||||
|   ); |   ); | ||||||
|  | |||||||
| @ -1,15 +1,15 @@ | |||||||
| "use client" | "use client"; | ||||||
| 
 | 
 | ||||||
| import * as React from "react" | import * as React from "react"; | ||||||
| import * as AlertDialogPrimitive from "@radix-ui/react-alert-dialog" | import * as AlertDialogPrimitive from "@radix-ui/react-alert-dialog"; | ||||||
| 
 | 
 | ||||||
| import { cn } from "@/lib/utils" | import { cn } from "@/lib/utils"; | ||||||
| import { buttonVariants } from "@/components/ui/button" | import { buttonVariants } from "@/components/ui/button"; | ||||||
| 
 | 
 | ||||||
| function AlertDialog({ | function AlertDialog({ | ||||||
|   ...props |   ...props | ||||||
| }: React.ComponentProps<typeof AlertDialogPrimitive.Root>) { | }: React.ComponentProps<typeof AlertDialogPrimitive.Root>) { | ||||||
|   return <AlertDialogPrimitive.Root data-slot="alert-dialog" {...props} /> |   return <AlertDialogPrimitive.Root data-slot="alert-dialog" {...props} />; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogTrigger({ | function AlertDialogTrigger({ | ||||||
| @ -17,7 +17,7 @@ function AlertDialogTrigger({ | |||||||
| }: React.ComponentProps<typeof AlertDialogPrimitive.Trigger>) { | }: React.ComponentProps<typeof AlertDialogPrimitive.Trigger>) { | ||||||
|   return ( |   return ( | ||||||
|     <AlertDialogPrimitive.Trigger data-slot="alert-dialog-trigger" {...props} /> |     <AlertDialogPrimitive.Trigger data-slot="alert-dialog-trigger" {...props} /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogPortal({ | function AlertDialogPortal({ | ||||||
| @ -25,7 +25,7 @@ function AlertDialogPortal({ | |||||||
| }: React.ComponentProps<typeof AlertDialogPrimitive.Portal>) { | }: React.ComponentProps<typeof AlertDialogPrimitive.Portal>) { | ||||||
|   return ( |   return ( | ||||||
|     <AlertDialogPrimitive.Portal data-slot="alert-dialog-portal" {...props} /> |     <AlertDialogPrimitive.Portal data-slot="alert-dialog-portal" {...props} /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogOverlay({ | function AlertDialogOverlay({ | ||||||
| @ -37,11 +37,11 @@ function AlertDialogOverlay({ | |||||||
|       data-slot="alert-dialog-overlay" |       data-slot="alert-dialog-overlay" | ||||||
|       className={cn( |       className={cn( | ||||||
|         "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50", |         "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50", | ||||||
|         className |         className, | ||||||
|       )} |       )} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogContent({ | function AlertDialogContent({ | ||||||
| @ -54,13 +54,13 @@ function AlertDialogContent({ | |||||||
|       <AlertDialogPrimitive.Content |       <AlertDialogPrimitive.Content | ||||||
|         data-slot="alert-dialog-content" |         data-slot="alert-dialog-content" | ||||||
|         className={cn( |         className={cn( | ||||||
|           "bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg", |           "bg-background/20 container-bg data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg", | ||||||
|           className |           className, | ||||||
|         )} |         )} | ||||||
|         {...props} |         {...props} | ||||||
|       /> |       /> | ||||||
|     </AlertDialogPortal> |     </AlertDialogPortal> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogHeader({ | function AlertDialogHeader({ | ||||||
| @ -73,7 +73,7 @@ function AlertDialogHeader({ | |||||||
|       className={cn("flex flex-col gap-2 text-center sm:text-left", className)} |       className={cn("flex flex-col gap-2 text-center sm:text-left", className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogFooter({ | function AlertDialogFooter({ | ||||||
| @ -85,11 +85,11 @@ function AlertDialogFooter({ | |||||||
|       data-slot="alert-dialog-footer" |       data-slot="alert-dialog-footer" | ||||||
|       className={cn( |       className={cn( | ||||||
|         "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end", |         "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end", | ||||||
|         className |         className, | ||||||
|       )} |       )} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogTitle({ | function AlertDialogTitle({ | ||||||
| @ -102,7 +102,7 @@ function AlertDialogTitle({ | |||||||
|       className={cn("text-lg font-semibold", className)} |       className={cn("text-lg font-semibold", className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogDescription({ | function AlertDialogDescription({ | ||||||
| @ -115,7 +115,7 @@ function AlertDialogDescription({ | |||||||
|       className={cn("text-muted-foreground text-sm", className)} |       className={cn("text-muted-foreground text-sm", className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogAction({ | function AlertDialogAction({ | ||||||
| @ -127,7 +127,7 @@ function AlertDialogAction({ | |||||||
|       className={cn(buttonVariants(), className)} |       className={cn(buttonVariants(), className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function AlertDialogCancel({ | function AlertDialogCancel({ | ||||||
| @ -139,7 +139,7 @@ function AlertDialogCancel({ | |||||||
|       className={cn(buttonVariants({ variant: "outline" }), className)} |       className={cn(buttonVariants({ variant: "outline" }), className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export { | export { | ||||||
| @ -154,4 +154,4 @@ export { | |||||||
|   AlertDialogDescription, |   AlertDialogDescription, | ||||||
|   AlertDialogAction, |   AlertDialogAction, | ||||||
|   AlertDialogCancel, |   AlertDialogCancel, | ||||||
| } | }; | ||||||
|  | |||||||
| @ -14,14 +14,13 @@ const buttonVariants = cva( | |||||||
|         destructive: |         destructive: | ||||||
|           "bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60", |           "bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60", | ||||||
|         outline: |         outline: | ||||||
|           "border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50", |           "border bg-transparent shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50", | ||||||
|         secondary: |         secondary: | ||||||
|           "bg-secondary text-secondary-foreground shadow-xs hover:bg-secondary/80", |           "bg-secondary text-secondary-foreground shadow-xs hover:bg-secondary/80", | ||||||
|         ghost: |         ghost: "hover:bg-background/10  dark:hover:bg-accent/50 rounded-full", | ||||||
|           "hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50 rounded-full", |  | ||||||
|         link: "text-primary underline-offset-4 hover:underline", |         link: "text-primary underline-offset-4 hover:underline", | ||||||
|         party: |         party: | ||||||
|           "bg-gradient-to-r from-secondary via-primary to-secondary text-shadow-primary text-primary-foreground rounded-xl border-b-4 shadow-lg  font-bold hover:translate-y-[0.1rem] hover:border-b-2 ",     |           "bg-gradient-to-r from-secondary via-primary to-secondary text-shadow-primary text-primary-foreground rounded-xl border-b-4 shadow-lg  font-bold hover:translate-y-[0.1rem] hover:border-b-2 ", | ||||||
|       }, |       }, | ||||||
|       size: { |       size: { | ||||||
|         default: "h-9 px-4 py-2 has-[>svg]:px-3", |         default: "h-9 px-4 py-2 has-[>svg]:px-3", | ||||||
| @ -38,16 +37,18 @@ const buttonVariants = cva( | |||||||
|   }, |   }, | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
|  | export type ButtonProps = React.ComponentProps<"button"> & | ||||||
|  |   VariantProps<typeof buttonVariants> & { | ||||||
|  |     asChild?: boolean; | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
| function Button({ | function Button({ | ||||||
|   className, |   className, | ||||||
|   variant, |   variant, | ||||||
|   size, |   size, | ||||||
|   asChild = false, |   asChild = false, | ||||||
|   ...props |   ...props | ||||||
| }: React.ComponentProps<"button"> & | }: ButtonProps) { | ||||||
|   VariantProps<typeof buttonVariants> & { |  | ||||||
|     asChild?: boolean; |  | ||||||
|   }) { |  | ||||||
|   const Comp = asChild ? Slot : "button"; |   const Comp = asChild ? Slot : "button"; | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|  | |||||||
| @ -1,33 +1,33 @@ | |||||||
| "use client" | "use client"; | ||||||
| 
 | 
 | ||||||
| import * as React from "react" | import * as React from "react"; | ||||||
| import * as DialogPrimitive from "@radix-ui/react-dialog" | import * as DialogPrimitive from "@radix-ui/react-dialog"; | ||||||
| import { XIcon } from "lucide-react" | import { XIcon } from "lucide-react"; | ||||||
| 
 | 
 | ||||||
| import { cn } from "@/lib/utils" | import { cn } from "@/lib/utils"; | ||||||
| 
 | 
 | ||||||
| function Dialog({ | function Dialog({ | ||||||
|   ...props |   ...props | ||||||
| }: React.ComponentProps<typeof DialogPrimitive.Root>) { | }: React.ComponentProps<typeof DialogPrimitive.Root>) { | ||||||
|   return <DialogPrimitive.Root data-slot="dialog" {...props} /> |   return <DialogPrimitive.Root data-slot="dialog" {...props} />; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogTrigger({ | function DialogTrigger({ | ||||||
|   ...props |   ...props | ||||||
| }: React.ComponentProps<typeof DialogPrimitive.Trigger>) { | }: React.ComponentProps<typeof DialogPrimitive.Trigger>) { | ||||||
|   return <DialogPrimitive.Trigger data-slot="dialog-trigger" {...props} /> |   return <DialogPrimitive.Trigger data-slot="dialog-trigger" {...props} />; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogPortal({ | function DialogPortal({ | ||||||
|   ...props |   ...props | ||||||
| }: React.ComponentProps<typeof DialogPrimitive.Portal>) { | }: React.ComponentProps<typeof DialogPrimitive.Portal>) { | ||||||
|   return <DialogPrimitive.Portal data-slot="dialog-portal" {...props} /> |   return <DialogPrimitive.Portal data-slot="dialog-portal" {...props} />; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogClose({ | function DialogClose({ | ||||||
|   ...props |   ...props | ||||||
| }: React.ComponentProps<typeof DialogPrimitive.Close>) { | }: React.ComponentProps<typeof DialogPrimitive.Close>) { | ||||||
|   return <DialogPrimitive.Close data-slot="dialog-close" {...props} /> |   return <DialogPrimitive.Close data-slot="dialog-close" {...props} />; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogOverlay({ | function DialogOverlay({ | ||||||
| @ -39,11 +39,11 @@ function DialogOverlay({ | |||||||
|       data-slot="dialog-overlay" |       data-slot="dialog-overlay" | ||||||
|       className={cn( |       className={cn( | ||||||
|         "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50", |         "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50", | ||||||
|         className |         className, | ||||||
|       )} |       )} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogContent({ | function DialogContent({ | ||||||
| @ -57,8 +57,8 @@ function DialogContent({ | |||||||
|       <DialogPrimitive.Content |       <DialogPrimitive.Content | ||||||
|         data-slot="dialog-content" |         data-slot="dialog-content" | ||||||
|         className={cn( |         className={cn( | ||||||
|           "bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg", |           "bg-background/20 container-bg data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg", | ||||||
|           className |           className, | ||||||
|         )} |         )} | ||||||
|         {...props} |         {...props} | ||||||
|       > |       > | ||||||
| @ -69,7 +69,7 @@ function DialogContent({ | |||||||
|         </DialogPrimitive.Close> |         </DialogPrimitive.Close> | ||||||
|       </DialogPrimitive.Content> |       </DialogPrimitive.Content> | ||||||
|     </DialogPortal> |     </DialogPortal> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogHeader({ className, ...props }: React.ComponentProps<"div">) { | function DialogHeader({ className, ...props }: React.ComponentProps<"div">) { | ||||||
| @ -79,7 +79,7 @@ function DialogHeader({ className, ...props }: React.ComponentProps<"div">) { | |||||||
|       className={cn("flex flex-col gap-2 text-center sm:text-left", className)} |       className={cn("flex flex-col gap-2 text-center sm:text-left", className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogFooter({ className, ...props }: React.ComponentProps<"div">) { | function DialogFooter({ className, ...props }: React.ComponentProps<"div">) { | ||||||
| @ -88,11 +88,11 @@ function DialogFooter({ className, ...props }: React.ComponentProps<"div">) { | |||||||
|       data-slot="dialog-footer" |       data-slot="dialog-footer" | ||||||
|       className={cn( |       className={cn( | ||||||
|         "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end", |         "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end", | ||||||
|         className |         className, | ||||||
|       )} |       )} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogTitle({ | function DialogTitle({ | ||||||
| @ -105,7 +105,7 @@ function DialogTitle({ | |||||||
|       className={cn("text-lg leading-none font-semibold", className)} |       className={cn("text-lg leading-none font-semibold", className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function DialogDescription({ | function DialogDescription({ | ||||||
| @ -118,7 +118,7 @@ function DialogDescription({ | |||||||
|       className={cn("text-muted-foreground text-sm", className)} |       className={cn("text-muted-foreground text-sm", className)} | ||||||
|       {...props} |       {...props} | ||||||
|     /> |     /> | ||||||
|   ) |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export { | export { | ||||||
| @ -132,4 +132,4 @@ export { | |||||||
|   DialogPortal, |   DialogPortal, | ||||||
|   DialogTitle, |   DialogTitle, | ||||||
|   DialogTrigger, |   DialogTrigger, | ||||||
| } | }; | ||||||
|  | |||||||
| @ -1,3 +1,5 @@ | |||||||
|  | import { appRoutes } from "./app.routes"; | ||||||
|  | 
 | ||||||
| type AppConfig = { | type AppConfig = { | ||||||
|   name: string; |   name: string; | ||||||
|   description: string; |   description: string; | ||||||
| @ -10,15 +12,11 @@ export const appConfig: AppConfig = { | |||||||
|   navigation: [ |   navigation: [ | ||||||
|     { |     { | ||||||
|       label: "Home", |       label: "Home", | ||||||
|       path: "/", |       path: appRoutes.home, | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       label: "Lobbies", |  | ||||||
|       path: "/lobby", |  | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       label: "Games", |       label: "Games", | ||||||
|       path: "/game", |       path: appRoutes.game, | ||||||
|     }, |     }, | ||||||
|   ], |   ], | ||||||
| }; | }; | ||||||
							
								
								
									
										13
									
								
								src/config/app.routes.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								src/config/app.routes.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | |||||||
|  | export const appRoutes = { | ||||||
|  |   home: "/", | ||||||
|  |   game: "/game", | ||||||
|  |   currentlobby: "/lobby", | ||||||
|  |   lobby: (id: string) => `/lobby/${id}`, | ||||||
|  | 
 | ||||||
|  |   me: "/me", | ||||||
|  |   editProfile: "/me/edit", | ||||||
|  |   playerProfile: (id: string) => `/player/${id}`, | ||||||
|  | 
 | ||||||
|  |   signIn: "/api/auth/signin", | ||||||
|  |   signOut: "/api/auth/signout", | ||||||
|  | }; | ||||||
							
								
								
									
										2
									
								
								src/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @ -2,3 +2,5 @@ type NavLink = { | |||||||
|   label: string; |   label: string; | ||||||
|   path: string; |   path: string; | ||||||
| }; | }; | ||||||
|  | 
 | ||||||
|  | type LobbyMemberRole = "player" | "admin"; | ||||||
|  | |||||||
| @ -1,5 +1,6 @@ | |||||||
| import { lobbyRouter } from "@/server/api/routers/lobby"; | import { lobbyRouter } from "@/server/api/routers/lobby"; | ||||||
| import { createCallerFactory, createTRPCRouter } from "@/server/api/trpc"; | import { createCallerFactory, createTRPCRouter } from "@/server/api/trpc"; | ||||||
|  | import { playerRouter } from "./routers/player"; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * This is the primary router for your server. |  * This is the primary router for your server. | ||||||
| @ -8,6 +9,7 @@ import { createCallerFactory, createTRPCRouter } from "@/server/api/trpc"; | |||||||
|  */ |  */ | ||||||
| export const appRouter = createTRPCRouter({ | export const appRouter = createTRPCRouter({ | ||||||
|   lobby: lobbyRouter, |   lobby: lobbyRouter, | ||||||
|  |   player: playerRouter, | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| // export type definition of API
 | // export type definition of API
 | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| import { z } from "zod"; | import { z } from "zod"; | ||||||
| 
 | 
 | ||||||
| import { lobbies, lobbyMembers } from "@/server/db/schema"; | import { lobbies, lobbyMembers, players } from "@/server/db/schema"; | ||||||
| import { | import { | ||||||
|   createTRPCRouter, |   createTRPCRouter, | ||||||
|   protectedProcedure, |   protectedProcedure, | ||||||
| @ -8,25 +8,31 @@ import { | |||||||
| } from "@/server/api/trpc"; | } from "@/server/api/trpc"; | ||||||
| import { lobbyPatchSchema } from "@/lib/validations/lobby"; | import { lobbyPatchSchema } from "@/lib/validations/lobby"; | ||||||
| import { and, eq } from "drizzle-orm"; | import { and, eq } from "drizzle-orm"; | ||||||
| import { time } from "console"; | import { | ||||||
|  |   combineRedisIterators, | ||||||
|  |   redisAsyncIterator, | ||||||
|  |   redisPublish, | ||||||
|  | } from "@/server/redis/sse-redis"; | ||||||
|  | import { tracked } from "@trpc/server"; | ||||||
| 
 | 
 | ||||||
| export const lobbyRouter = createTRPCRouter({ | export const lobbyRouter = createTRPCRouter({ | ||||||
|   // queries
 |   // queries
 | ||||||
|   getAll: protectedProcedure.query(async ({ ctx }) => { |   getCurrentLobby: protectedProcedure.query(async ({ ctx }) => { | ||||||
|     const ownedLobbies = await ctx.db.query.lobbies.findMany({ |     const reuslt = await ctx.db.query.lobbyMembers.findFirst({ | ||||||
|       where: eq(lobbies.createdById, ctx.session.user.id), |       where: eq(lobbyMembers.playerId, ctx.session.user.id), | ||||||
|     }); |  | ||||||
|     const joinedLobbies = await ctx.db.query.lobbyMembers.findMany({ |  | ||||||
|       where: eq(lobbyMembers.userId, ctx.session.user.id), |  | ||||||
|       with: { |       with: { | ||||||
|         lobby: true, |         lobby: { | ||||||
|  |           with: { | ||||||
|  |             members: { | ||||||
|  |               with: { | ||||||
|  |                 player: true, | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|       }, |       }, | ||||||
|     }); |     }); | ||||||
| 
 |     return reuslt?.lobby!; | ||||||
|     return [ |  | ||||||
|       ...ownedLobbies, |  | ||||||
|       ...(joinedLobbies?.map(({ lobby }) => lobby) ?? []), |  | ||||||
|     ]; |  | ||||||
|   }), |   }), | ||||||
| 
 | 
 | ||||||
|   get: publicProcedure |   get: publicProcedure | ||||||
| @ -40,24 +46,9 @@ export const lobbyRouter = createTRPCRouter({ | |||||||
|         await ctx.db.query.lobbies.findFirst({ |         await ctx.db.query.lobbies.findFirst({ | ||||||
|           where: eq(lobbies.id, input.id), |           where: eq(lobbies.id, input.id), | ||||||
|           with: { |           with: { | ||||||
|             leader: { |  | ||||||
|               columns: { |  | ||||||
|                 image: true, |  | ||||||
|                 name: true, |  | ||||||
|                 id: true, |  | ||||||
|                 joinedAt: true, |  | ||||||
|               }, |  | ||||||
|             }, |  | ||||||
|             members: { |             members: { | ||||||
|               with: { |               with: { | ||||||
|                 user: { |                 player: true, | ||||||
|                   columns: { |  | ||||||
|                     image: true, |  | ||||||
|                     name: true, |  | ||||||
|                     id: true, |  | ||||||
|                     joinedAt: true, |  | ||||||
|                   }, |  | ||||||
|                 }, |  | ||||||
|               }, |               }, | ||||||
|             }, |             }, | ||||||
|           }, |           }, | ||||||
| @ -79,6 +70,15 @@ export const lobbyRouter = createTRPCRouter({ | |||||||
|           createdById: ctx.session.user.id, |           createdById: ctx.session.user.id, | ||||||
|         }) |         }) | ||||||
|         .returning({ id: lobbies.id }); |         .returning({ id: lobbies.id }); | ||||||
|  |       if (!lobby) throw new Error("Error creating lobby"); | ||||||
|  |       await ctx.db.insert(lobbyMembers).values({ | ||||||
|  |         lobbyId: lobby.id, | ||||||
|  |         playerId: ctx.session.user.id, | ||||||
|  |         isReady: false, | ||||||
|  | 
 | ||||||
|  |         role: "admin", | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|       return lobby; |       return lobby; | ||||||
|     }), |     }), | ||||||
|   update: protectedProcedure |   update: protectedProcedure | ||||||
| @ -89,6 +89,8 @@ export const lobbyRouter = createTRPCRouter({ | |||||||
|       }), |       }), | ||||||
|     ) |     ) | ||||||
|     .mutation(async ({ ctx, input }) => { |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       console.log("Check if user is admin"); | ||||||
|  | 
 | ||||||
|       const [lobby] = await ctx.db |       const [lobby] = await ctx.db | ||||||
|         .update(lobbies) |         .update(lobbies) | ||||||
|         .set(input.lobby) |         .set(input.lobby) | ||||||
| @ -108,6 +110,8 @@ export const lobbyRouter = createTRPCRouter({ | |||||||
|       }), |       }), | ||||||
|     ) |     ) | ||||||
|     .mutation(async ({ ctx, input }) => { |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       console.log("Check if user is admin"); | ||||||
|  | 
 | ||||||
|       const [lobby] = await ctx.db |       const [lobby] = await ctx.db | ||||||
|         .delete(lobbies) |         .delete(lobbies) | ||||||
|         .where( |         .where( | ||||||
| @ -117,6 +121,7 @@ export const lobbyRouter = createTRPCRouter({ | |||||||
|           ), |           ), | ||||||
|         ) |         ) | ||||||
|         .returning({ id: lobbies.id }); |         .returning({ id: lobbies.id }); | ||||||
|  | 
 | ||||||
|       return lobby; |       return lobby; | ||||||
|     }), |     }), | ||||||
| 
 | 
 | ||||||
| @ -129,30 +134,65 @@ export const lobbyRouter = createTRPCRouter({ | |||||||
|     ) |     ) | ||||||
|     .mutation(async ({ ctx, input }) => { |     .mutation(async ({ ctx, input }) => { | ||||||
|       if (input.join) { |       if (input.join) { | ||||||
|         return ( |         const [member] = await ctx.db | ||||||
|           await ctx.db |           .insert(lobbyMembers) | ||||||
|             .insert(lobbyMembers) |           .values({ | ||||||
|             .values({ |             lobbyId: input.lobbyId, | ||||||
|               lobbyId: input.lobbyId, |             playerId: ctx.session.user.id, | ||||||
|               userId: ctx.session.user.id, |             isReady: false, | ||||||
|               isReady: false, |             role: "player", | ||||||
|               joinedAt: new Date(), |           }) | ||||||
|               role: "member", |           .returning(); | ||||||
|  |         const player = member | ||||||
|  |           ? await ctx.db.query.players.findFirst({ | ||||||
|  |               where: eq(players.id, member.playerId), | ||||||
|             }) |             }) | ||||||
|             .returning() |           : undefined; | ||||||
|         )[0]; |         if (member) redisPublish("lobby:member:join", { ...member, player }); | ||||||
|  |         return member; | ||||||
|       } else { |       } else { | ||||||
|         return ( |         const [member] = await ctx.db | ||||||
|           await ctx.db |           .delete(lobbyMembers) | ||||||
|             .delete(lobbyMembers) |           .where( | ||||||
|             .where( |             and( | ||||||
|               and( |               eq(lobbyMembers.lobbyId, input.lobbyId), | ||||||
|                 eq(lobbyMembers.lobbyId, input.lobbyId), |               eq(lobbyMembers.playerId, ctx.session.user.id), | ||||||
|                 eq(lobbyMembers.userId, ctx.session.user.id), |             ), | ||||||
|               ), |           ) | ||||||
|             ) |           .returning(); | ||||||
|             .returning() |         if (member) redisPublish("lobby:member:leave", member.playerId); | ||||||
|         )[0]; |         return member; | ||||||
|  |       } | ||||||
|  |     }), | ||||||
|  | 
 | ||||||
|  |   // subscriptions
 | ||||||
|  |   onMemberUpdate: protectedProcedure | ||||||
|  |     .input( | ||||||
|  |       z | ||||||
|  |         .object({ | ||||||
|  |           lastEventId: z.string().nullish(), | ||||||
|  |         }) | ||||||
|  |         .optional(), | ||||||
|  |     ) | ||||||
|  |     .subscription(async function* (opts) { | ||||||
|  |       // Create an async iterator for our Redis channel.
 | ||||||
|  | 
 | ||||||
|  |       if (opts.input?.lastEventId) { | ||||||
|  |         // fetch posts from a database that were missed.
 | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       for await (const { event, data: membership } of combineRedisIterators([ | ||||||
|  |         "lobby:member:join", | ||||||
|  |         "lobby:member:leave", | ||||||
|  |       ])) { | ||||||
|  |         const joined = event === "lobby:member:join"; | ||||||
|  |         const id = | ||||||
|  |           typeof membership === "string" ? membership : membership.playerId; | ||||||
|  | 
 | ||||||
|  |         yield tracked(String(id), { | ||||||
|  |           joined, | ||||||
|  |           membership, | ||||||
|  |         }); | ||||||
|       } |       } | ||||||
|     }), |     }), | ||||||
| }); | }); | ||||||
|  | |||||||
							
								
								
									
										13
									
								
								src/server/api/routers/player.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								src/server/api/routers/player.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | |||||||
|  | import { eq } from "drizzle-orm"; | ||||||
|  | import { createTRPCRouter, protectedProcedure, publicProcedure } from "../trpc"; | ||||||
|  | import { players } from "@/server/db/schema"; | ||||||
|  | 
 | ||||||
|  | export const playerRouter = createTRPCRouter({ | ||||||
|  |   getBySession: publicProcedure.query(async ({ ctx }) => { | ||||||
|  |     return ctx?.session?.user | ||||||
|  |       ? await ctx.db.query.players.findFirst({ | ||||||
|  |           where: eq(players.id, ctx.session.user.id), | ||||||
|  |         }) | ||||||
|  |       : null; | ||||||
|  |   }), | ||||||
|  | }); | ||||||
| @ -1,15 +1,6 @@ | |||||||
| import { DrizzleAdapter } from "@auth/drizzle-adapter"; |  | ||||||
| import { type DefaultSession, type NextAuthConfig, type User } from "next-auth"; | import { type DefaultSession, type NextAuthConfig, type User } from "next-auth"; | ||||||
| import DiscordProvider from "next-auth/providers/discord"; | import DiscordProvider from "next-auth/providers/discord"; | ||||||
| 
 | import { CustomDrizzleAdapter } from "./custom-drizzle-adapter"; | ||||||
| import { db } from "@/server/db"; |  | ||||||
| import { |  | ||||||
|   accounts, |  | ||||||
|   sessions, |  | ||||||
|   users, |  | ||||||
|   verificationTokens, |  | ||||||
| } from "@/server/db/schema"; |  | ||||||
| import type { Adapter } from "next-auth/adapters"; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Module augmentation for `next-auth` types. Allows us to add custom properties to the `session` |  * Module augmentation for `next-auth` types. Allows us to add custom properties to the `session` | ||||||
| @ -17,23 +8,16 @@ import type { Adapter } from "next-auth/adapters"; | |||||||
|  * |  * | ||||||
|  * @see https://next-auth.js.org/getting-started/typescript#module-augmentation
 |  * @see https://next-auth.js.org/getting-started/typescript#module-augmentation
 | ||||||
|  */ |  */ | ||||||
|  | 
 | ||||||
| declare module "next-auth" { | declare module "next-auth" { | ||||||
|   interface Session extends DefaultSession { |   interface Session extends DefaultSession { | ||||||
|     user: { |     user: { | ||||||
|       id: string; |       id: string; | ||||||
|       joinedAt: Date; |       email: string; | ||||||
|       // ...other properties
 |       name: string; | ||||||
|       // role: UserRole;
 |     }; | ||||||
|     } & DefaultSession["user"]; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   interface User { |  | ||||||
|     joinedAt: Date; |  | ||||||
|     // ...other properties
 |  | ||||||
|     // role: UserRole;
 |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
| export type PublicUser = Pick<User, "name" | "image" | "id" | "joinedAt">; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Options for NextAuth.js used to configure adapters, providers, callbacks, etc. |  * Options for NextAuth.js used to configure adapters, providers, callbacks, etc. | ||||||
| @ -41,33 +25,18 @@ export type PublicUser = Pick<User, "name" | "image" | "id" | "joinedAt">; | |||||||
|  * @see https://next-auth.js.org/configuration/options
 |  * @see https://next-auth.js.org/configuration/options
 | ||||||
|  */ |  */ | ||||||
| export const authConfig = { | export const authConfig = { | ||||||
|   providers: [ |   providers: [DiscordProvider], | ||||||
|     DiscordProvider, |   adapter: CustomDrizzleAdapter, | ||||||
|     /** |  | ||||||
|      * ...add more providers here. |  | ||||||
|      * |  | ||||||
|      * Most other providers require a bit more work than the Discord provider. For example, the |  | ||||||
|      * GitHub provider requires you to add the `refresh_token_expires_in` field to the Account |  | ||||||
|      * model. Refer to the NextAuth.js docs for the provider you want to use. Example: |  | ||||||
|      * |  | ||||||
|      * @see https://next-auth.js.org/providers/github
 |  | ||||||
|      */ |  | ||||||
|   ], |  | ||||||
|   adapter: DrizzleAdapter(db, { |  | ||||||
|     usersTable: users, |  | ||||||
|     accountsTable: accounts, |  | ||||||
|     sessionsTable: sessions, |  | ||||||
|     verificationTokensTable: verificationTokens, |  | ||||||
|   }) as Adapter, |  | ||||||
| 
 | 
 | ||||||
|   callbacks: { |   callbacks: { | ||||||
|     session: ({ session, user }) => ({ |     session: async ({ session, user }) => { | ||||||
|       ...session, |       return { | ||||||
|       user: { |         ...session, | ||||||
|         ...session.user, |         user: { | ||||||
|         id: user.id, |           ...session.user, | ||||||
|         joinedAt: user.joinedAt, |           id: user.id, | ||||||
|       }, |         }, | ||||||
|     }), |       }; | ||||||
|  |     }, | ||||||
|   }, |   }, | ||||||
| } satisfies NextAuthConfig; | } satisfies NextAuthConfig; | ||||||
|  | |||||||
							
								
								
									
										39
									
								
								src/server/auth/custom-drizzle-adapter.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/server/auth/custom-drizzle-adapter.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | |||||||
|  | import { DrizzleAdapter } from "@auth/drizzle-adapter"; | ||||||
|  | import { db } from "@/server/db"; | ||||||
|  | import { | ||||||
|  |   accounts, | ||||||
|  |   players, | ||||||
|  |   sessions, | ||||||
|  |   users, | ||||||
|  |   verificationTokens, | ||||||
|  | } from "@/server/db/schema"; | ||||||
|  | import type { Adapter } from "next-auth/adapters"; | ||||||
|  | 
 | ||||||
|  | export const CustomDrizzleAdapter: Adapter = { | ||||||
|  |   ...(DrizzleAdapter(db, { | ||||||
|  |     usersTable: users as any, | ||||||
|  |     accountsTable: accounts, | ||||||
|  |     sessionsTable: sessions, | ||||||
|  |     verificationTokensTable: verificationTokens, | ||||||
|  |   }) as Adapter), | ||||||
|  | 
 | ||||||
|  |   async createUser(rawUser) { | ||||||
|  |     // Insert user **without** the image field
 | ||||||
|  |     const [user] = await db | ||||||
|  |       .insert(users) | ||||||
|  |       .values({ | ||||||
|  |         name: rawUser.name, | ||||||
|  |         email: rawUser.email, | ||||||
|  |         emailVerified: rawUser.emailVerified, | ||||||
|  |       }) | ||||||
|  |       .returning(); | ||||||
|  |     if (!user) throw new Error("Error creating user"); | ||||||
|  | 
 | ||||||
|  |     await db.insert(players).values({ | ||||||
|  |       id: user.id, | ||||||
|  |       displayName: rawUser.name, | ||||||
|  |       avatar: rawUser.image, | ||||||
|  |     }); | ||||||
|  |     return user; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @ -10,6 +10,12 @@ import { createId } from "@paralleldrive/cuid2"; | |||||||
|  */ |  */ | ||||||
| export const createTable = pgTableCreator((name) => `game-master_${name}`); | export const createTable = pgTableCreator((name) => `game-master_${name}`); | ||||||
| 
 | 
 | ||||||
|  | const defaultTimeStamp = (name: string, d: any) => | ||||||
|  |   d | ||||||
|  |     .timestamp(name, { withTimezone: true }) | ||||||
|  |     .default(sql`CURRENT_TIMESTAMP`) | ||||||
|  |     .notNull(); | ||||||
|  | 
 | ||||||
| export const lobbies = createTable("lobby", (d) => ({ | export const lobbies = createTable("lobby", (d) => ({ | ||||||
|   id: d |   id: d | ||||||
|     .varchar({ length: 255 }) |     .varchar({ length: 255 }) | ||||||
| @ -21,22 +27,22 @@ export const lobbies = createTable("lobby", (d) => ({ | |||||||
|   createdById: d |   createdById: d | ||||||
|     .varchar({ length: 255 }) |     .varchar({ length: 255 }) | ||||||
|     .notNull() |     .notNull() | ||||||
|     .references(() => users.id, { onDelete: "cascade" }), |     .references(() => players.id, { onDelete: "cascade" }), | ||||||
|   createdAt: d |   createdAt: defaultTimeStamp("created_at", d), | ||||||
|     .timestamp("created_at", { withTimezone: true }) |  | ||||||
|     .default(sql`CURRENT_TIMESTAMP`) |  | ||||||
|     .notNull(), |  | ||||||
|   updatedAt: d |   updatedAt: d | ||||||
|     .timestamp("updated_at", { withTimezone: true }) |     .timestamp("updated_at", { withTimezone: true }) | ||||||
|     .$onUpdate(() => new Date()), |     .$onUpdate(() => new Date()), | ||||||
| })); | })); | ||||||
| 
 | 
 | ||||||
| export type Lobby = typeof lobbies.$inferSelect; | export type Lobby = typeof lobbies.$inferSelect & { | ||||||
|  |   members?: LobbyMember[]; | ||||||
|  |   leader?: Player; | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| export const lobbyRelations = relations(lobbies, ({ many, one }) => ({ | export const lobbyRelations = relations(lobbies, ({ many, one }) => ({ | ||||||
|   leader: one(users, { |   leader: one(players, { | ||||||
|     fields: [lobbies.createdById], |     fields: [lobbies.createdById], | ||||||
|     references: [users.id], |     references: [players.id], | ||||||
|   }), |   }), | ||||||
|   members: many(lobbyMembers), |   members: many(lobbyMembers), | ||||||
| })); | })); | ||||||
| @ -44,38 +50,65 @@ export const lobbyRelations = relations(lobbies, ({ many, one }) => ({ | |||||||
| export const lobbyMembers = createTable( | export const lobbyMembers = createTable( | ||||||
|   "lobby_member", |   "lobby_member", | ||||||
|   (d) => ({ |   (d) => ({ | ||||||
|     userId: d |     playerId: d | ||||||
|       .varchar({ length: 255 }) |       .varchar({ length: 255 }) | ||||||
|       .notNull() |       .notNull() | ||||||
|       .references(() => users.id, { onDelete: "cascade" }), |       .references(() => players.id, { onDelete: "cascade" }), | ||||||
|     lobbyId: d |     lobbyId: d | ||||||
|       .varchar({ length: 255 }) |       .varchar({ length: 255 }) | ||||||
|       .notNull() |       .notNull() | ||||||
|       .references(() => lobbies.id, { onDelete: "cascade" }), |       .references(() => lobbies.id, { onDelete: "cascade" }), | ||||||
|     joinedAt: d.timestamp("created_at", { withTimezone: true }).notNull(), |     joinedAt: defaultTimeStamp("joined_at", d), | ||||||
|     role: d.varchar({ length: 255 }).notNull(), |     role: d | ||||||
|  |       .varchar({ length: 255 }) | ||||||
|  |       .notNull() | ||||||
|  |       .$type<LobbyMemberRole>() | ||||||
|  |       .default("player"), | ||||||
|     isReady: d.boolean().notNull(), |     isReady: d.boolean().notNull(), | ||||||
|   }), |   }), | ||||||
|   (t) => [primaryKey({ columns: [t.lobbyId, t.userId] })], |   (t) => [primaryKey({ columns: [t.playerId] })], | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
|  | export type LobbyMember = typeof lobbyMembers.$inferSelect & { | ||||||
|  |   player?: Player; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| export const lobbyMembersRelations = relations(lobbyMembers, ({ one }) => ({ | export const lobbyMembersRelations = relations(lobbyMembers, ({ one }) => ({ | ||||||
|   lobby: one(lobbies, { |   lobby: one(lobbies, { | ||||||
|     fields: [lobbyMembers.lobbyId], |     fields: [lobbyMembers.lobbyId], | ||||||
|     references: [lobbies.id], |     references: [lobbies.id], | ||||||
|   }), |   }), | ||||||
|   user: one(users, { |   player: one(players, { | ||||||
|     fields: [lobbyMembers.userId], |     fields: [lobbyMembers.playerId], | ||||||
|     references: [users.id], |     references: [players.id], | ||||||
|   }), |   }), | ||||||
| })); | })); | ||||||
| 
 | 
 | ||||||
|  | export const players = createTable( | ||||||
|  |   "player", | ||||||
|  |   (d) => ({ | ||||||
|  |     id: d | ||||||
|  |       .varchar({ length: 255 }) | ||||||
|  |       .notNull() | ||||||
|  |       .references(() => users.id, { onDelete: "cascade" }), | ||||||
|  |     displayName: d.varchar({ length: 255 }), | ||||||
|  |     avatar: d.varchar({ length: 255 }), | ||||||
|  |     joinedAt: defaultTimeStamp("joined_at", d), | ||||||
|  |   }), | ||||||
|  |   (t) => [ | ||||||
|  |     primaryKey({ | ||||||
|  |       columns: [t.id], | ||||||
|  |     }), | ||||||
|  |   ], | ||||||
|  | ); | ||||||
|  | export type Player = typeof players.$inferSelect; | ||||||
|  | 
 | ||||||
| export const users = createTable("user", (d) => ({ | export const users = createTable("user", (d) => ({ | ||||||
|   id: d |   id: d | ||||||
|     .varchar({ length: 255 }) |     .varchar({ length: 255 }) | ||||||
|     .notNull() |     .notNull() | ||||||
|     .primaryKey() |     .primaryKey() | ||||||
|     .$defaultFn(() => crypto.randomUUID()), |     .$defaultFn(() => createId()), | ||||||
|   name: d.varchar({ length: 255 }), |   name: d.varchar({ length: 255 }), | ||||||
|   email: d.varchar({ length: 255 }).notNull(), |   email: d.varchar({ length: 255 }).notNull(), | ||||||
|   emailVerified: d |   emailVerified: d | ||||||
| @ -84,11 +117,6 @@ export const users = createTable("user", (d) => ({ | |||||||
|       withTimezone: true, |       withTimezone: true, | ||||||
|     }) |     }) | ||||||
|     .default(sql`CURRENT_TIMESTAMP`), |     .default(sql`CURRENT_TIMESTAMP`), | ||||||
|   image: d.varchar({ length: 255 }), |  | ||||||
|   joinedAt: d |  | ||||||
|     .timestamp("joined_at", { withTimezone: true }) |  | ||||||
|     .default(sql`CURRENT_TIMESTAMP`) |  | ||||||
|     .notNull(), |  | ||||||
| })); | })); | ||||||
| 
 | 
 | ||||||
| export const usersRelations = relations(users, ({ many }) => ({ | export const usersRelations = relations(users, ({ many }) => ({ | ||||||
| @ -102,7 +130,7 @@ export const accounts = createTable( | |||||||
|     userId: d |     userId: d | ||||||
|       .varchar({ length: 255 }) |       .varchar({ length: 255 }) | ||||||
|       .notNull() |       .notNull() | ||||||
|       .references(() => users.id), |       .references(() => users.id, { onDelete: "cascade" }), | ||||||
|     type: d.varchar({ length: 255 }).$type<AdapterAccount["type"]>().notNull(), |     type: d.varchar({ length: 255 }).$type<AdapterAccount["type"]>().notNull(), | ||||||
|     provider: d.varchar({ length: 255 }).notNull(), |     provider: d.varchar({ length: 255 }).notNull(), | ||||||
|     providerAccountId: d.varchar({ length: 255 }).notNull(), |     providerAccountId: d.varchar({ length: 255 }).notNull(), | ||||||
| @ -133,7 +161,7 @@ export const sessions = createTable( | |||||||
|     userId: d |     userId: d | ||||||
|       .varchar({ length: 255 }) |       .varchar({ length: 255 }) | ||||||
|       .notNull() |       .notNull() | ||||||
|       .references(() => users.id), |       .references(() => users.id, { onDelete: "cascade" }), | ||||||
|     expires: d.timestamp({ mode: "date" }).notNull(), |     expires: d.timestamp({ mode: "date" }).notNull(), | ||||||
|   }), |   }), | ||||||
|   (t) => [index("session_user_id_idx").on(t.userId)], |   (t) => [index("session_user_id_idx").on(t.userId)], | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								src/server/redis/events.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/server/redis/events.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,6 @@ | |||||||
|  | import type { LobbyMember, Post } from "../db/schema"; | ||||||
|  | 
 | ||||||
|  | export type PubSubEvents = { | ||||||
|  |   "lobby:member:join": LobbyMember; | ||||||
|  |   "lobby:member:leave": string; | ||||||
|  | }; | ||||||
							
								
								
									
										72
									
								
								src/server/redis/sse-redis.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								src/server/redis/sse-redis.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,72 @@ | |||||||
|  | import Redis from "ioredis"; | ||||||
|  | import type { PubSubEvents } from "./events"; | ||||||
|  | 
 | ||||||
|  | const redisPub = new Redis(); | ||||||
|  | const redisSub = new Redis(); | ||||||
|  | 
 | ||||||
|  | export const redisPublish = <T extends keyof PubSubEvents>( | ||||||
|  |   event: T, | ||||||
|  |   data: PubSubEvents[T], | ||||||
|  | ) => { | ||||||
|  |   redisPub.publish(event, JSON.stringify(data)); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const redisSubscribe = <T extends keyof PubSubEvents>( | ||||||
|  |   event: T, | ||||||
|  |   callback: (data: PubSubEvents[T]) => void, | ||||||
|  | ) => { | ||||||
|  |   redisSub.subscribe(event, (err) => { | ||||||
|  |     if (err) console.error(`Redis subscription error for ${event}:`, err); | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   redisSub.on("message", (channel, message) => { | ||||||
|  |     if (channel === event) { | ||||||
|  |       callback(JSON.parse(message) as PubSubEvents[T]); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Creates an async iterator for a Redis subscription channel. | ||||||
|  |  * @param event - The Redis event to subscribe to. | ||||||
|  |  * @returns An async iterator that yields tracked events. | ||||||
|  |  */ | ||||||
|  | export function redisAsyncIterator<K extends keyof PubSubEvents>(event: K) { | ||||||
|  |   return { | ||||||
|  |     [Symbol.asyncIterator]() { | ||||||
|  |       return { | ||||||
|  |         next(): Promise<IteratorResult<PubSubEvents[K]>> { | ||||||
|  |           return new Promise((resolve) => { | ||||||
|  |             redisSubscribe(event, (data: PubSubEvents[K]) => { | ||||||
|  |               resolve({ value: data, done: false }); | ||||||
|  |             }); | ||||||
|  |           }); | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function* combineRedisIterators<T extends keyof PubSubEvents>( | ||||||
|  |   events: T[], | ||||||
|  | ) { | ||||||
|  |   const iterators = events.map((event) => | ||||||
|  |     redisAsyncIterator(event)[Symbol.asyncIterator](), | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   while (true) { | ||||||
|  |     // Wait for the next event from any iterator
 | ||||||
|  |     const result = await Promise.race( | ||||||
|  |       iterators.map((iterator, index) => | ||||||
|  |         iterator.next().then((untypedRes) => { | ||||||
|  |           const res = untypedRes as IteratorResult<PubSubEvents[T]>; | ||||||
|  |           return { res, event: events[index] }; | ||||||
|  |         }), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     if (result.res.done) break; | ||||||
|  | 
 | ||||||
|  |     yield { event: result.event, data: result.res.value }; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @ -1,7 +1,12 @@ | |||||||
| "use client"; | "use client"; | ||||||
| 
 | 
 | ||||||
| import { QueryClientProvider, type QueryClient } from "@tanstack/react-query"; | import { QueryClientProvider, type QueryClient } from "@tanstack/react-query"; | ||||||
| import { httpBatchStreamLink, loggerLink } from "@trpc/client"; | import { | ||||||
|  |   httpBatchStreamLink, | ||||||
|  |   httpSubscriptionLink, | ||||||
|  |   loggerLink, | ||||||
|  |   splitLink, | ||||||
|  | } from "@trpc/client"; | ||||||
| import { createTRPCReact } from "@trpc/react-query"; | import { createTRPCReact } from "@trpc/react-query"; | ||||||
| import { type inferRouterInputs, type inferRouterOutputs } from "@trpc/server"; | import { type inferRouterInputs, type inferRouterOutputs } from "@trpc/server"; | ||||||
| import { useState } from "react"; | import { useState } from "react"; | ||||||
| @ -50,14 +55,22 @@ export function TRPCReactProvider(props: { children: React.ReactNode }) { | |||||||
|             process.env.NODE_ENV === "development" || |             process.env.NODE_ENV === "development" || | ||||||
|             (op.direction === "down" && op.result instanceof Error), |             (op.direction === "down" && op.result instanceof Error), | ||||||
|         }), |         }), | ||||||
|         httpBatchStreamLink({ |         splitLink({ | ||||||
|           transformer: SuperJSON, |           // uses the httpSubscriptionLink for subscriptions
 | ||||||
|           url: getBaseUrl() + "/api/trpc", |           condition: (op) => op.type === "subscription", | ||||||
|           headers: () => { |           true: httpSubscriptionLink({ | ||||||
|             const headers = new Headers(); |             transformer: SuperJSON, | ||||||
|             headers.set("x-trpc-source", "nextjs-react"); |             url: `${getBaseUrl()}/api/trpc`, | ||||||
|             return headers; |           }), | ||||||
|           }, |           false: httpBatchStreamLink({ | ||||||
|  |             transformer: SuperJSON, | ||||||
|  |             url: `${getBaseUrl()}/api/trpc`, | ||||||
|  |             headers: () => { | ||||||
|  |               const headers = new Headers(); | ||||||
|  |               headers.set("x-trpc-source", "nextjs-react"); | ||||||
|  |               return headers; | ||||||
|  |             }, | ||||||
|  |           }), | ||||||
|         }), |         }), | ||||||
|       ], |       ], | ||||||
|     }), |     }), | ||||||
|  | |||||||
| @ -1,60 +1,82 @@ | |||||||
| #!/usr/bin/env bash | #!/usr/bin/env bash | ||||||
| # Use this script to start a docker container for a local development database |  | ||||||
| 
 |  | ||||||
| # TO RUN ON WINDOWS: |  | ||||||
| # 1. Install WSL (Windows Subsystem for Linux) - https://learn.microsoft.com/en-us/windows/wsl/install |  | ||||||
| # 2. Install Docker Desktop for Windows - https://docs.docker.com/docker-for-windows/install/ |  | ||||||
| # 3. Open WSL - `wsl` |  | ||||||
| # 4. Run this script - `./start-database.sh` |  | ||||||
| 
 |  | ||||||
| # On Linux and macOS you can run this script directly - `./start-database.sh` |  | ||||||
| 
 |  | ||||||
| DB_CONTAINER_NAME="game-master" |  | ||||||
| 
 |  | ||||||
| if ! [ -x "$(command -v docker)" ]; then |  | ||||||
|   echo -e "Docker is not installed. Please install docker and try again.\nDocker install guide: https://docs.docker.com/engine/install/" |  | ||||||
|   exit 1 |  | ||||||
| fi |  | ||||||
| 
 |  | ||||||
| if ! docker info > /dev/null 2>&1; then |  | ||||||
|   echo "Docker daemon is not running. Please start Docker and try again." |  | ||||||
|   exit 1 |  | ||||||
| fi |  | ||||||
| 
 |  | ||||||
| if [ "$(docker ps -q -f name=$DB_CONTAINER_NAME)" ]; then |  | ||||||
|   echo "Database container '$DB_CONTAINER_NAME' already running" |  | ||||||
|   exit 0 |  | ||||||
| fi |  | ||||||
| 
 |  | ||||||
| if [ "$(docker ps -q -a -f name=$DB_CONTAINER_NAME)" ]; then |  | ||||||
|   docker start "$DB_CONTAINER_NAME" |  | ||||||
|   echo "Existing database container '$DB_CONTAINER_NAME' started" |  | ||||||
|   exit 0 |  | ||||||
| fi |  | ||||||
| 
 | 
 | ||||||
| # import env variables from .env | # import env variables from .env | ||||||
| set -a | set -a | ||||||
| source .env | source .env | ||||||
| 
 | 
 | ||||||
| DB_PASSWORD=$(echo "$DATABASE_URL" | awk -F':' '{print $3}' | awk -F'@' '{print $1}') | DB_PASSWORD=$(echo "$DATABASE_URL" | awk -F':' '{print $3}' | awk -F'@' '{print $1}') | ||||||
| DB_PORT=$(echo "$DATABASE_URL" | awk -F':' '{print $4}' | awk -F'\/' '{print $1}') | DB_PORT=$(echo "$DATABASE_URL" | awk -F':' '{print $4}' | awk -F'/' '{print $1}') | ||||||
|  | DB_NAME=$(echo "$DATABASE_URL" | awk -F'/' '{print $4}') | ||||||
|  | DB_CONTAINER_NAME="$DB_NAME-postgres" | ||||||
| 
 | 
 | ||||||
| if [ "$DB_PASSWORD" = "password" ]; then | REDIS_PORT=${REDIS_PORT:-6379} | ||||||
|   echo "You are using the default database password" | REDIS_CONTAINER_NAME="$DB_NAME-redis" | ||||||
|   read -p "Should we generate a random password for you? [y/N]: " -r REPLY | 
 | ||||||
|   if ! [[ $REPLY =~ ^[Yy]$ ]]; then | if ! [ -x "$(command -v docker)" ] && ! [ -x "$(command -v podman)" ]; then | ||||||
|     echo "Please change the default password in the .env file and try again" |   echo -e "Docker or Podman is not installed. Please install docker or podman and try again.\nDocker install guide: https://docs.docker.com/engine/install/\nPodman install guide: https://podman.io/getting-started/installation" | ||||||
|     exit 1 |   exit 1 | ||||||
|   fi |  | ||||||
|   # Generate a random URL-safe password |  | ||||||
|   DB_PASSWORD=$(openssl rand -base64 12 | tr '+/' '-_') |  | ||||||
|   sed -i -e "s#:password@#:$DB_PASSWORD@#" .env |  | ||||||
| fi | fi | ||||||
| 
 | 
 | ||||||
| docker run -d \ | # determine which docker command to use | ||||||
|   --name $DB_CONTAINER_NAME \ | if [ -x "$(command -v docker)" ]; then | ||||||
|   -e POSTGRES_USER="postgres" \ |   DOCKER_CMD="docker" | ||||||
|   -e POSTGRES_PASSWORD="$DB_PASSWORD" \ | elif [ -x "$(command -v podman)" ]; then | ||||||
|   -e POSTGRES_DB=game-master \ |   DOCKER_CMD="podman" | ||||||
|   -p "$DB_PORT":5432 \ | fi | ||||||
|   docker.io/postgres && echo "Database container '$DB_CONTAINER_NAME' was successfully created" | 
 | ||||||
|  | if ! $DOCKER_CMD info > /dev/null 2>&1; then | ||||||
|  |   echo "$DOCKER_CMD daemon is not running. Please start $DOCKER_CMD and try again." | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | # # Check if ports are in use | ||||||
|  | # if command -v nc >/dev/null 2>&1; then | ||||||
|  | #   if nc -z localhost "$DB_PORT" 2>/dev/null; then | ||||||
|  | #     echo "Port $DB_PORT is already in use." | ||||||
|  | #     exit 1 | ||||||
|  | #   fi | ||||||
|  | #   if nc -z localhost "$REDIS_PORT" 2>/dev/null; then | ||||||
|  | #     echo "Port $REDIS_PORT is already in use." | ||||||
|  | #     exit 1 | ||||||
|  | #   fi | ||||||
|  | # else | ||||||
|  | #   echo "Warning: Unable to check if ports are in use (netcat not installed)" | ||||||
|  | #   read -p "Do you want to continue anyway? [y/N]: " -r REPLY | ||||||
|  | #   if ! [[ $REPLY =~ ^[Yy]$ ]]; then | ||||||
|  | #     echo "Aborting." | ||||||
|  | #     exit 1 | ||||||
|  | #   fi | ||||||
|  | # fi | ||||||
|  | 
 | ||||||
|  | # Start PostgreSQL if not running | ||||||
|  | if [ "$($DOCKER_CMD ps -q -f name=$DB_CONTAINER_NAME)" ]; then | ||||||
|  |   echo "Database container '$DB_CONTAINER_NAME' already running" | ||||||
|  | else | ||||||
|  |   if [ "$($DOCKER_CMD ps -q -a -f name=$DB_CONTAINER_NAME)" ]; then | ||||||
|  |     $DOCKER_CMD start "$DB_CONTAINER_NAME" | ||||||
|  |     echo "Existing database container '$DB_CONTAINER_NAME' started" | ||||||
|  |   else | ||||||
|  |     $DOCKER_CMD run -d \ | ||||||
|  |       --name $DB_CONTAINER_NAME \ | ||||||
|  |       -e POSTGRES_USER="postgres" \ | ||||||
|  |       -e POSTGRES_PASSWORD="$DB_PASSWORD" \ | ||||||
|  |       -e POSTGRES_DB="$DB_NAME" \ | ||||||
|  |       -p "$DB_PORT":5432 \ | ||||||
|  |       docker.io/postgres && echo "Database container '$DB_CONTAINER_NAME' was successfully created" | ||||||
|  |   fi | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | # Start Redis if not running | ||||||
|  | if [ "$($DOCKER_CMD ps -q -f name=$REDIS_CONTAINER_NAME)" ]; then | ||||||
|  |   echo "Redis container '$REDIS_CONTAINER_NAME' already running" | ||||||
|  | else | ||||||
|  |   if [ "$($DOCKER_CMD ps -q -a -f name=$REDIS_CONTAINER_NAME)" ]; then | ||||||
|  |     $DOCKER_CMD start "$REDIS_CONTAINER_NAME" | ||||||
|  |     echo "Existing Redis container '$REDIS_CONTAINER_NAME' started" | ||||||
|  |   else | ||||||
|  |     $DOCKER_CMD run -d \ | ||||||
|  |       --name $REDIS_CONTAINER_NAME \ | ||||||
|  |       -p "$REDIS_PORT":6379 \ | ||||||
|  |       docker.io/redis && echo "Redis container '$REDIS_CONTAINER_NAME' was successfully created" | ||||||
|  |   fi | ||||||
|  | fi | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user