From 0cb436799933842dc89d58477b577884f600c245 Mon Sep 17 00:00:00 2001 From: martin Date: Sun, 9 Jul 2023 19:10:36 +0200 Subject: [PATCH] Fixed bugs, sending players instead of characters --- .../ClientApp/src/components/gameBoard.tsx | 5 +- .../src/components/gameComponent.tsx | 30 ++++----- .../ClientApp/src/types/props.d.ts | 2 +- .../ClientApp/src/utils/actions.ts | 64 +++++++++++-------- .../ClientApp/src/utils/state.ts | 8 ++- pac-man-board-game/Services/GameGroup.cs | 4 +- 6 files changed, 60 insertions(+), 53 deletions(-) diff --git a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx index 8982deb..c76a4ca 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -3,9 +3,10 @@ import {Character, PacMan} from "../game/character"; import findPossiblePositions from "../game/possibleMovesAlgorithm"; import {GameTile} from "./gameTile"; import {TileType} from "../game/tileType"; +import {useAtomValue} from "jotai"; +import {allCharactersAtom} from "../utils/state"; interface BoardProps extends ComponentProps { - characters: Character[], selectedDice?: SelectedDice, onMove?: Action, map: GameMap @@ -14,12 +15,12 @@ interface BoardProps extends ComponentProps { const Board: Component = ( { className, - characters, selectedDice, onMove, map }) => { + const characters = useAtomValue(allCharactersAtom); const [selectedCharacter, setSelectedCharacter] = useState(); const [possiblePositions, setPossiblePositions] = useState([]); // TODO reset when other client moves a character const [hoveredPosition, setHoveredPosition] = useState(); diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index e210e0e..7bf51c7 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -6,19 +6,17 @@ import WebSocketService from "../websockets/WebSocketService"; import {testMap} from "../game/map"; import Player, {State} from "../game/player"; import PlayerStats from "../components/playerStats"; -import {useAtom} from "jotai"; -import {charactersAtom, currentPlayerAtom, diceAtom, playersAtom, selectedDiceAtom} from "../utils/state"; +import {getDefaultStore, useAtom, useAtomValue} from "jotai"; +import {currentPlayerAtom, diceAtom, ghostsAtom, playersAtom, selectedDiceAtom} from "../utils/state"; const wsService = new WebSocketService(import.meta.env.VITE_API); export const GameComponent: Component<{ player: Player }> = ({player}) => { - // TODO find spawn points - const [characters] = useAtom(charactersAtom); - const [players] = useAtom(playersAtom); + const players = useAtomValue(playersAtom); - const [dice] = useAtom(diceAtom); + const dice = useAtomValue(diceAtom); const [selectedDice, setSelectedDice] = useAtom(selectedDiceAtom); - const [currentPlayer] = useAtom(currentPlayerAtom); + const currentPlayer = useAtomValue(currentPlayerAtom); function startGameLoop(): void { if (currentPlayer?.Name !== player.Name) return; @@ -35,9 +33,10 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => { const data: ActionMessage = { Action: GameAction.moveCharacter, Data: { - dice: dice?.length ?? 0 > 0 ? dice : null, - characters: characters, - eatenPellets: eatenPellets + Dice: dice?.length ?? 0 > 0 ? dice : null, + Players: players, + Ghosts: getDefaultStore().get(ghostsAtom), + EatenPellets: eatenPellets } }; wsService.send(data); @@ -70,13 +69,10 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => { } - {players?.map(p => )} - {characters && - - } + {players?.map(p => )} + ); }; diff --git a/pac-man-board-game/ClientApp/src/types/props.d.ts b/pac-man-board-game/ClientApp/src/types/props.d.ts index 3e0781c..97b4139 100644 --- a/pac-man-board-game/ClientApp/src/types/props.d.ts +++ b/pac-man-board-game/ClientApp/src/types/props.d.ts @@ -34,7 +34,7 @@ interface BoxProps { interface PlayerProps { readonly Name: string, - readonly PacMan?: CharacterProps, + readonly PacMan: CharacterProps, readonly Colour: import("../game/colour").Colour, readonly Box?: BoxProps, State?: import("../game/player").State, diff --git a/pac-man-board-game/ClientApp/src/utils/actions.ts b/pac-man-board-game/ClientApp/src/utils/actions.ts index dcd56db..949fbf9 100644 --- a/pac-man-board-game/ClientApp/src/utils/actions.ts +++ b/pac-man-board-game/ClientApp/src/utils/actions.ts @@ -1,9 +1,9 @@ import Player from "../game/player"; -import {Character, CharacterType, Ghost, PacMan} from "../game/character"; +import {CharacterType, Ghost} from "../game/character"; import {getCharacterSpawns, testMap} from "../game/map"; import {TileType} from "../game/tileType"; import {getDefaultStore} from "jotai"; -import {charactersAtom, currentPlayerAtom, diceAtom, playersAtom} from "./state"; +import {currentPlayerAtom, diceAtom, ghostsAtom, playersAtom} from "./state"; import {Colour} from "../game/colour"; export enum GameAction { @@ -13,19 +13,20 @@ export enum GameAction { ready, } +const store = getDefaultStore(); + const ghostsProps: CharacterProps[] = [ {Colour: Colour.Purple}, {Colour: Colour.Purple}, ]; - let spawns = getCharacterSpawns(testMap).filter(spawn => spawn.type === CharacterType.ghost); ghostsProps.forEach(ghost => { ghost.SpawnPosition = spawns.pop()?.position; -}); +}); const ghosts = ghostsProps.map(props => new Ghost(props)); -const store = getDefaultStore(); +store.set(ghostsAtom, ghosts); export const doAction: MessageEventFunction = (event): void => { // TODO divide into smaller functions const message: ActionMessage = JSON.parse(event.data); @@ -51,28 +52,41 @@ function setDice(data?: number[]): void { store.set(diceAtom, data); } -type MoveCharacterData = { dice: number[], characters: CharacterProps[], eatenPellets: Position[] }; +type MoveCharacterData = { Dice: number[], Players: PlayerProps[], Ghosts: CharacterProps[], EatenPellets: Position[] }; function moveCharacter(data?: MoveCharacterData): void { - store.set(diceAtom, data?.dice); - updateCharacters(data); + store.set(diceAtom, data?.Dice); + updatePlayers(data); + updateGhosts(data); removeEatenPellets(data); } -function updateCharacters(data?: MoveCharacterData): void { - const updatedCharacters = data?.characters; +function updatePlayers(data?: MoveCharacterData): void { + const updatedPlayers = data?.Players; - if (updatedCharacters) { - const newList: Character[] = []; - for (const character of updatedCharacters) { - newList.push(new Character(character)); + if (updatedPlayers) { + const newList: Player[] = []; + for (const player of updatedPlayers) { + newList.push(new Player(player)); } - store.set(charactersAtom, newList); + store.set(playersAtom, newList); + } +} + +function updateGhosts(data?: MoveCharacterData): void { + const updatedGhosts = data?.Ghosts; + + if (updatedGhosts) { + const newList: Ghost[] = []; + for (const ghost of updatedGhosts) { + newList.push(new Ghost(ghost)); + } + store.set(ghostsAtom, newList); } } function removeEatenPellets(data?: MoveCharacterData): void { - const pellets = data?.eatenPellets; + const pellets = data?.EatenPellets; for (const pellet of pellets ?? []) { testMap[pellet.y][pellet.x] = TileType.empty; @@ -81,20 +95,15 @@ function removeEatenPellets(data?: MoveCharacterData): void { function playerInfo(data?: PlayerProps[]): void { const playerProps = data ?? []; - store.set(playersAtom, playerProps.map(p => new Player(p))); - spawns = getCharacterSpawns(testMap).filter(spawn => spawn.type === CharacterType.pacMan); - const pacMen = playerProps.filter(p => p.PacMan).map(p => { - if (!p.PacMan!.SpawnPosition) { - p.PacMan!.SpawnPosition = spawns.pop()?.position; + store.set(playersAtom, playerProps.map(p => { + if (!p.PacMan.SpawnPosition) { + p.PacMan.SpawnPosition = spawns.pop()?.position; } - return new PacMan(p.PacMan!); - }); - - store.set(charactersAtom, [...pacMen, ...ghosts]); + return new Player(p); + })); } - type ReadyData = | { AllReady: true, Starter: PlayerProps, Players: PlayerProps[] } | { AllReady: false, Players: PlayerProps[] } @@ -102,8 +111,7 @@ type ReadyData = function ready(data?: ReadyData): void { if (data && typeof data !== "string") { - const isReady = data.AllReady; - if (isReady) { + if (data.AllReady) { store.set(currentPlayerAtom, new Player(data.Starter)); } store.set(playersAtom, data.Players.map(p => new Player(p))); diff --git a/pac-man-board-game/ClientApp/src/utils/state.ts b/pac-man-board-game/ClientApp/src/utils/state.ts index 7b6d1e9..5eabe18 100644 --- a/pac-man-board-game/ClientApp/src/utils/state.ts +++ b/pac-man-board-game/ClientApp/src/utils/state.ts @@ -1,13 +1,15 @@ import Player from "../game/player"; import {atom} from "jotai"; -import {Character} from "../game/character"; import {atomWithStorage, createJSONStorage} from "jotai/utils"; +import {Ghost} from "../game/character"; const playerStorage = createJSONStorage(() => sessionStorage); -// TODO merge character and player atoms, since the player is the owner of the character -export const charactersAtom = atom(undefined); +// TODO derived from playersAtom export const playersAtom = atom([]); +export const playerCharactersAtom = atom(get => get(playersAtom).map(player => player.PacMan)); +export const ghostsAtom = atom([]); +export const allCharactersAtom = atom(get => [...get(playerCharactersAtom), ...get(ghostsAtom)]); export const thisPlayerAtom = atomWithStorage("player", undefined, { ...playerStorage, getItem(key, initialValue): Player | undefined { diff --git a/pac-man-board-game/Services/GameGroup.cs b/pac-man-board-game/Services/GameGroup.cs index 1e03804..ea8c433 100644 --- a/pac-man-board-game/Services/GameGroup.cs +++ b/pac-man-board-game/Services/GameGroup.cs @@ -11,12 +11,12 @@ public class GameGroup // TODO tests public IPlayer RandomPlayer => Players[_random.Next(Players.Count)]; public event Func, Task>? Connections; - public bool AddPlayer(IPlayer player) + public bool AddPlayer(IPlayer player) // TODO if name exists, use that player instead { if (Players.Count >= Rules.MaxPlayers) return false; - if (Players.Exists(p => p.Name == player.Name)) return false; player.State = State.WaitingForPlayers; + if (Players.Exists(p => p.Name == player.Name)) return true; Players.Add(player); return true; }