Spaces:
				
			
			
	
			
			
		Sleeping
		
	
	
	
			
			
	
	
	
	
		
		
		Sleeping
		
	| import { useRef, useState } from 'react'; | |
| import PixiGame from './PixiGame.tsx'; | |
| import { useElementSize } from 'usehooks-ts'; | |
| import { Stage } from '@pixi/react'; | |
| import { ConvexProvider, useConvex, useQuery } from 'convex/react'; | |
| import PlayerDetails from './PlayerDetails.tsx'; | |
| import { api } from '../../convex/_generated/api'; | |
| import { useWorldHeartbeat } from '../hooks/useWorldHeartbeat.ts'; | |
| import { useHistoricalTime } from '../hooks/useHistoricalTime.ts'; | |
| import { DebugTimeManager } from './DebugTimeManager.tsx'; | |
| import { GameId } from '../../convex/aiTown/ids.ts'; | |
| import { Game as GameObj} from '../../convex/aiTown/game.ts'; | |
| import { ServerGame, useServerGame } from '../hooks/serverGame.ts'; | |
| import { GameCycle } from '../../convex/aiTown/gameCycle.ts'; | |
| import { PlayerDescription } from '../../convex/aiTown/playerDescription.ts'; | |
| import { Cloud } from './Cloud.tsx'; | |
| import VotingPopover from './LLMVote.tsx'; | |
| import { createPortal } from 'react-dom'; | |
| import GameVote from './GameVote.tsx'; | |
| import { EndGame } from './EndGame.tsx'; | |
| export const SHOW_DEBUG_UI = !!import.meta.env.VITE_SHOW_DEBUG_UI; | |
| const oauth = JSON.parse(localStorage.getItem('oauth')); | |
| const oauthToken = oauth ? oauth.userInfo.fullname : undefined; | |
| export function GameStateLabel(game: GameObj, me: PlayerDescription | undefined) { | |
| switch (game.world.gameCycle.cycleState) { | |
| case 'Day': | |
| return { | |
| label: 'Day', | |
| desc: 'Find out who is a werewolf', | |
| }; | |
| case 'WerewolfVoting': | |
| return { | |
| label: 'Werewolf Vote', | |
| desc: 'Select a player who is a warewolf', | |
| }; | |
| case 'PlayerKillVoting': | |
| return { | |
| label: 'Player Kill Vote', | |
| desc: me?.type === 'werewolf' ? 'Select a player to kill' : 'Hide in your home!!', | |
| }; | |
| case 'LobbyState': | |
| return { | |
| label: 'Lobby (waiting for start)', | |
| desc: 'Waiting for the game to start', | |
| }; | |
| case 'Night': | |
| return { | |
| label: 'Night', | |
| desc: me?.type === 'werewolf' ? 'Discuss who to kill with other warewolves' : 'Hide in your home!!', | |
| }; | |
| case 'EndGame': | |
| return { | |
| label: 'The End', | |
| desc: `Winners are ${ game.world.winner }!`, | |
| }; | |
| } | |
| } | |
| export function canVote(game: ServerGame, me: PlayerDescription | undefined) { | |
| return me && (game.world.gameCycle.cycleState === "WerewolfVoting" || (game.world.gameCycle.cycleState === "PlayerKillVoting" && me.type === "werewolf")); | |
| } | |
| export function isEndGame(game: ServerGame) { | |
| return game.world.gameCycle.cycleState === "EndGame"; | |
| } | |
| function showMap(gameCycle: GameCycle, me: PlayerDescription | undefined) { | |
| // Here also check for player description | |
| return (gameCycle.cycleState === "Day" || gameCycle.cycleState === "WerewolfVoting") || me?.type === "werewolf"; | |
| } | |
| export default function Game() { | |
| const convex = useConvex(); | |
| const [selectedElement, setSelectedElement] = useState<{ | |
| kind: 'player'; | |
| id: GameId<'players'>; | |
| }>(); | |
| const [gameWrapperRef, { width, height }] = useElementSize(); | |
| const worldStatus = useQuery(api.world.defaultWorldStatus); | |
| const worldId = worldStatus?.worldId; | |
| const engineId = worldStatus?.engineId; | |
| const game = useServerGame(worldId); | |
| // Send a periodic heartbeat to our world to keep it alive. | |
| useWorldHeartbeat(); | |
| const worldState = useQuery(api.world.worldState, worldId ? { worldId } : 'skip'); | |
| const { historicalTime, timeManager } = useHistoricalTime(worldState?.engine); | |
| const scrollViewRef = useRef<HTMLDivElement>(null); | |
| const humanTokenIdentifier =oauthToken | |
| if (!worldId || !engineId || !game ) { | |
| return null; | |
| } | |
| console.log("humanTokenIdentifier",humanTokenIdentifier) | |
| console.log("game.playerDescriptions",game.playerDescriptions) | |
| console.log("game.world.players",game.world.players) | |
| const playerId = [...game.world.players.values()].find( | |
| (p) => !!p && p.human === humanTokenIdentifier, | |
| )?.id; | |
| console.log("playerId",playerId) | |
| const meDescription = playerId ? game?.playerDescriptions.get(playerId) : undefined; | |
| return ( | |
| <> | |
| {SHOW_DEBUG_UI && <DebugTimeManager timeManager={timeManager} width={200} height={100} />} | |
| <div className="mx-auto w-full max-w grid grid-rows-[240px_1fr] lg:grid-rows-[1fr] lg:grid-cols-[1fr_auto] lg:grow max-w-[1400px] min-h-[480px] game-frame"> | |
| {/* Game area */} | |
| <div className="relative overflow-hidden bg-brown-900" ref={gameWrapperRef}> | |
| <div className={`absolute inset-0 ${showMap(game.world.gameCycle, meDescription) ? '' : 'invisible' }`}> | |
| <div className="container"> | |
| <Stage width={width} height={height} options={{ backgroundColor: 0x7ab5ff }}> | |
| {/* Re-propagate context because contexts are not shared between renderers. | |
| https://github.com/michalochman/react-pixi-fiber/issues/145#issuecomment-531549215 */} | |
| <ConvexProvider client={convex}> | |
| <PixiGame | |
| game={game} | |
| worldId={worldId} | |
| engineId={engineId} | |
| width={width} | |
| height={height} | |
| historicalTime={historicalTime} | |
| setSelectedElement={setSelectedElement} | |
| /> | |
| </ConvexProvider> | |
| </Stage> | |
| </div> | |
| </div> | |
| <div className={`absolute inset-0 ${!showMap(game.world.gameCycle, meDescription) ? '' : 'invisible' }`}> | |
| <Cloud worldId={worldId} /> | |
| </div> | |
| </div> | |
| {/* Right column area */} | |
| <div | |
| className="flex flex-col overflow-y-auto shrink-0 px-4 py-6 sm:px-6 lg:w-96 xl:pr-6 border-t-8 sm:border-t-0 sm:border-l-8 border-brown-900 bg-brown-800 text-brown-100" | |
| ref={scrollViewRef} | |
| > | |
| <div className="flex flex-col items-center mb-4 gap-4"> | |
| <h2 className="text-2xl font-bold">{GameStateLabel(game as GameObj, meDescription).label}</h2> | |
| <p className="text-lg text-center">{GameStateLabel(game as GameObj, meDescription).desc}</p> | |
| </div> | |
| {playerId && !isEndGame(game) && canVote(game, meDescription) && <GameVote engineId={engineId} game={game} playerId={playerId} />} | |
| {!isEndGame(game) && !canVote(game, meDescription) && playerId &&<PlayerDetails | |
| worldId={worldId} | |
| engineId={engineId} | |
| game={game} | |
| playerId={selectedElement?.id} | |
| setSelectedElement={setSelectedElement} | |
| scrollViewRef={scrollViewRef} | |
| currentPlayerId={playerId} | |
| />} | |
| {playerId && isEndGame(game) && <EndGame game={game} playerId={playerId} />} | |
| </div> | |
| </div> | |
| {createPortal( | |
| <div className="max-w-[1400px] mx-auto"> | |
| {playerId && <VotingPopover engineId={engineId} game={game} playerId={playerId} />} | |
| </div>, | |
| document.getElementById('footer-buttons') | |
| )} | |
| </> | |
| ); | |
| } | |
 
			
