diff --git a/BackendTests/Services/ActionServiceTests.cs b/BackendTests/Services/ActionServiceTests.cs index bb43dda..35405e4 100644 --- a/BackendTests/Services/ActionServiceTests.cs +++ b/BackendTests/Services/ActionServiceTests.cs @@ -11,27 +11,49 @@ namespace BackendTests.Services; public class ActionServiceTests { - private readonly IPlayer _blackPlayer = Players.Create("black"); - private readonly IPlayer _redPlayer = Players.Create("red"); - private readonly IPlayer _whitePlayer = Players.Create("white"); + private readonly Player _blackPlayer = (Player)Players.Create("black"); + private readonly Player _redPlayer = (Player)Players.Create("red"); + + private readonly Player _whitePlayer = (Player)Players.Create("white"); private ActionMessage _blackMessage = null!; private ActionMessage _redMessage = null!; private IActionService _service = null!; + + private Queue _spawns = null!; + private ActionMessage _whiteMessage = null!; private IWebSocketService _wssSub = null!; [SetUp] public void Setup() { + _spawns = CreateQueue(); _whiteMessage = new ActionMessage - { Action = GameAction.PlayerInfo, Data = JsonSerializer.Serialize(_whitePlayer) }; + { + Action = GameAction.PlayerInfo, + Data = JsonSerializer.Serialize(new { Player = _whitePlayer, Spawns = CreateQueue() }) + }; _blackMessage = new ActionMessage - { Action = GameAction.PlayerInfo, Data = JsonSerializer.Serialize(_blackPlayer) }; - _redMessage = new ActionMessage { Action = GameAction.PlayerInfo, Data = JsonSerializer.Serialize(_redPlayer) }; + { + Action = GameAction.PlayerInfo, + Data = JsonSerializer.Serialize(new { Player = _blackPlayer, Spawns = CreateQueue() }) + }; + _redMessage = new ActionMessage + { + Action = GameAction.PlayerInfo, + Data = JsonSerializer.Serialize(new { Player = _redPlayer, Spawns = CreateQueue() }) + }; _wssSub = Substitute.For(Substitute.For>()); _service = new ActionService(Substitute.For>(), _wssSub); } + private static Queue CreateQueue() => + new(new[] + { + new DirectionalPosition { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up }, + new() { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down } + }); + #region RollDice() [Test] @@ -74,6 +96,9 @@ public class ActionServiceTests { var players = _service.SetPlayerInfo(_whiteMessage); + var pos = _spawns.Dequeue(); + _whitePlayer.PacMan.Position = pos; + _whitePlayer.PacMan.SpawnPosition = pos; Assert.That(new List { _whitePlayer }, Is.EqualTo(players)); } @@ -140,7 +165,7 @@ public class ActionServiceTests [Test] public void Ready_TwoReady() { - var group = new GameGroup { Players = { _blackPlayer, _whitePlayer } }; + var group = new GameGroup(new Queue()) { Players = { _blackPlayer, _whitePlayer } }; _service.Group = group; _service.Player = _blackPlayer; diff --git a/BackendTests/Services/GameGroupTests.cs b/BackendTests/Services/GameGroupTests.cs index f9c2196..d5b5baa 100644 --- a/BackendTests/Services/GameGroupTests.cs +++ b/BackendTests/Services/GameGroupTests.cs @@ -9,17 +9,34 @@ namespace BackendTests.Services; public class GameGroupTests { + private readonly DirectionalPosition _spawn3By3Up = new() + { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up }; + + private readonly DirectionalPosition _spawn7By7Down = new() + { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down }; + + private readonly DirectionalPosition _spawn7By7Left = new() + { At = new Position { X = 7, Y = 7 }, Direction = Direction.Left }; + + private readonly DirectionalPosition _spawn7By7Right = new() + { At = new Position { X = 7, Y = 7 }, Direction = Direction.Right }; + private IPlayer _bluePlayer = null!; private GameGroup _gameGroup = null!; private IPlayer _greenPlayer = null!; private IPlayer _purplePlayer = null!; private IPlayer _redPlayer = null!; + + private Queue _spawns = null!; private IPlayer _yellowPlayer = null!; [SetUp] public void Setup() { - _gameGroup = new GameGroup(); + _spawns = new Queue( + new[] { _spawn3By3Up, _spawn7By7Left, _spawn7By7Down, _spawn7By7Right }); + + _gameGroup = new GameGroup(_spawns); _redPlayer = Players.Create("red"); _bluePlayer = Players.Create("blue"); _yellowPlayer = Players.Create("yellow"); @@ -70,6 +87,14 @@ public class GameGroupTests Assert.That(_redPlayer.State, Is.EqualTo(State.WaitingForPlayers)); } + [Test] + public void AddPlayer_AddSpawnPosition() + { + _gameGroup.AddPlayer(_redPlayer); + Assert.That(_redPlayer.PacMan.SpawnPosition, Is.Not.Null); + Assert.That(_redPlayer.PacMan.SpawnPosition, Is.EqualTo(_spawn3By3Up)); + } + #endregion #region Sendtoall(ArraySegment segment) diff --git a/BackendTests/Services/WebSocketServiceTests.cs b/BackendTests/Services/WebSocketServiceTests.cs index 8ac2819..213696d 100644 --- a/BackendTests/Services/WebSocketServiceTests.cs +++ b/BackendTests/Services/WebSocketServiceTests.cs @@ -11,12 +11,25 @@ namespace BackendTests.Services; public class WebSocketServiceTests { + private readonly DirectionalPosition _spawn3By3Up = new() + { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up }; + private IWebSocketService _service = null!; + private Queue _spawns = null!; + + [SetUp] public void SetUp() { _service = new WebSocketService(Substitute.For>()); + _spawns = new Queue(new[] + { + _spawn3By3Up, + new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down }, + new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down }, + new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down } + }); } #region Send(Websocket, ArraySegment) @@ -135,7 +148,7 @@ public class WebSocketServiceTests public void AddPlayer_ToEmptyGroup() { var player = Players.Create("white"); - var group = _service.AddPlayer(player); + var group = _service.AddPlayer(player, _spawns); Assert.Multiple(() => { @@ -151,12 +164,12 @@ public class WebSocketServiceTests for (var i = 0; i < 4; i++) { var player = Players.Create(i.ToString()); - _service.AddPlayer(player); + _service.AddPlayer(player, _spawns); } var player5 = Players.Create("white"); - var group = _service.AddPlayer(player5); + var group = _service.AddPlayer(player5, new Queue(new[] { _spawn3By3Up })); Assert.Multiple(() => { diff --git a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx index c76a4ca..6199f6e 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -4,10 +4,9 @@ import findPossiblePositions from "../game/possibleMovesAlgorithm"; import {GameTile} from "./gameTile"; import {TileType} from "../game/tileType"; import {useAtomValue} from "jotai"; -import {allCharactersAtom} from "../utils/state"; +import {allCharactersAtom, selectedDiceAtom} from "../utils/state"; interface BoardProps extends ComponentProps { - selectedDice?: SelectedDice, onMove?: Action, map: GameMap } @@ -15,12 +14,12 @@ interface BoardProps extends ComponentProps { const Board: Component = ( { className, - selectedDice, onMove, map }) => { const characters = useAtomValue(allCharactersAtom); + const selectedDice = useAtomValue(selectedDiceAtom); const [selectedCharacter, setSelectedCharacter] = useState(); const [possiblePositions, setPossiblePositions] = useState([]); // TODO reset when other client moves a character const [hoveredPosition, setHoveredPosition] = useState(); @@ -63,15 +62,15 @@ const Board: Component = ( const pacMan = selectedCharacter as PacMan; for (const tile of [...destination.Path ?? [], destination.End]) { - const currentTile = map[tile.y][tile.x]; + const currentTile = map[tile.Y][tile.X]; if (currentTile === TileType.pellet) { // pacMan.box.addPellet(new Pellet()); // TODO update to current player - map[tile.y][tile.x] = TileType.empty; + map[tile.Y][tile.X] = TileType.empty; positions.push(tile); } else if (currentTile === TileType.powerPellet) { // pacMan.box.addPellet(new Pellet(true)); - map[tile.y][tile.x] = TileType.empty; + map[tile.Y][tile.X] = TileType.empty; positions.push(tile); } } @@ -98,9 +97,9 @@ const Board: Component = ( p.End.x === colIndex && p.End.y === rowIndex)} - character={characters.find(c => c.isAt({x: colIndex, y: rowIndex}))} - isSelected={selectedCharacter?.isAt({x: colIndex, y: rowIndex})} + possiblePath={possiblePositions.find(p => p.End.X === colIndex && p.End.Y === rowIndex)} + character={characters.find(c => c.isAt({X: colIndex, Y: rowIndex}))} + isSelected={selectedCharacter?.isAt({X: colIndex, Y: rowIndex})} showPath={hoveredPosition?.Path?.find(pos => pos.x === colIndex && pos.y === rowIndex) !== undefined} handleMoveCharacter={handleMoveCharacter} handleSelectCharacter={handleSelectCharacter} diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index 7bf51c7..c648ff8 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -3,11 +3,12 @@ import {AllDice} from "./dice"; import {doAction, GameAction} from "../utils/actions"; import GameBoard from "./gameBoard"; import WebSocketService from "../websockets/WebSocketService"; -import {testMap} from "../game/map"; +import {getCharacterSpawns, testMap} from "../game/map"; import Player, {State} from "../game/player"; import PlayerStats from "../components/playerStats"; import {getDefaultStore, useAtom, useAtomValue} from "jotai"; import {currentPlayerAtom, diceAtom, ghostsAtom, playersAtom, selectedDiceAtom} from "../utils/state"; +import {CharacterType} from "../game/character"; const wsService = new WebSocketService(import.meta.env.VITE_API); @@ -43,7 +44,19 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => { } async function sendPlayer(): Promise { - wsService.send({Action: GameAction.playerInfo, Data: player}); + // TODO set spawn position and position + wsService.send({ + Action: GameAction.playerInfo, + Data: { + Player: player, Spawns: getCharacterSpawns(testMap) + .filter(s => s.type === CharacterType.pacMan) + .map(s => s.position) + } + }); + } + + function sendReady(): void { + wsService.send({Action: GameAction.ready}); } useEffect(() => { @@ -55,10 +68,6 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => { return () => wsService.close(); }, []); - function sendReady(): void { - wsService.send({Action: GameAction.ready}); - } - return ( <>
@@ -70,9 +79,7 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => {
{players?.map(p => )} - + ); }; diff --git a/pac-man-board-game/ClientApp/src/components/gameTile.tsx b/pac-man-board-game/ClientApp/src/components/gameTile.tsx index a36f670..6b8eacc 100644 --- a/pac-man-board-game/ClientApp/src/components/gameTile.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameTile.tsx @@ -3,6 +3,7 @@ import {TileType} from "../game/tileType"; import {Character, Dummy} from "../game/character"; import {Direction} from "../game/direction"; import {getCSSColour} from "../utils/colours"; +import {Colour} from "../game/colour"; interface TileWithCharacterProps extends ComponentProps { possiblePath?: Path, @@ -113,8 +114,8 @@ const Tile: Component = ( onMouseEnter={onMouseEnter} onMouseLeave={onMouseLeave}> {children} - {type === TileType.pellet && } - {type === TileType.powerPellet && } + {type === TileType.pellet && } + {type === TileType.powerPellet && } ); }; @@ -146,7 +147,7 @@ const CharacterComponent: Component = ( }) => { function getSide() { - switch (character?.Position.Direction) { + switch (character?.Position?.Direction) { case Direction.up: return "right-1/4 top-0"; case Direction.down: diff --git a/pac-man-board-game/ClientApp/src/game/character.ts b/pac-man-board-game/ClientApp/src/game/character.ts index 700f06d..2bfa635 100644 --- a/pac-man-board-game/ClientApp/src/game/character.ts +++ b/pac-man-board-game/ClientApp/src/game/character.ts @@ -57,7 +57,7 @@ export class Character { } public isAt(position: Position): boolean { - return this.Position !== null && this.Position.End.x === position.x && this.Position.End.y === position.y; + return this.Position !== null && this.Position.End.X === position.X && this.Position.End.Y === position.Y; } } @@ -83,7 +83,7 @@ export class Dummy extends Character { Colour: Colour.Grey, Position: position, IsEatable: false, - SpawnPosition: {At: {x: 0, y: 0}, Direction: Direction.up}, + SpawnPosition: {At: {X: 0, Y: 0}, Direction: Direction.up}, Type: CharacterType.dummy, }); } diff --git a/pac-man-board-game/ClientApp/src/game/map.ts b/pac-man-board-game/ClientApp/src/game/map.ts index 61b3d73..0e50342 100644 --- a/pac-man-board-game/ClientApp/src/game/map.ts +++ b/pac-man-board-game/ClientApp/src/game/map.ts @@ -30,10 +30,10 @@ export function getCharacterSpawns(map: GameMap): { type: CharacterType, positio for (let col = 0; col < map.length; col++) { // TODO find direction if (map[row][col] === 4) { - result.push({type: CharacterType.ghost, position: {At: {x: col, y: row}, Direction: Direction.up}}); + result.push({type: CharacterType.ghost, position: {At: {X: col, Y: row}, Direction: Direction.up}}); } else if (map[row][col] === 5) { result.push({ - type: CharacterType.pacMan, position: {At: {x: col, y: row}, Direction: Direction.up} + type: CharacterType.pacMan, position: {At: {X: col, Y: row}, Direction: Direction.up} }); } } diff --git a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts index 6653514..cbb7718 100644 --- a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts +++ b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts @@ -108,23 +108,23 @@ function tryMove(board: GameMap, path: Path, direction: Direction, steps: number switch (direction) { case Direction.left: return { - x: path.End.x - 1, - y: path.End.y + X: path.End.X - 1, + Y: path.End.Y }; case Direction.up: return { - x: path.End.x, - y: path.End.y - 1 + X: path.End.X, + Y: path.End.Y - 1 }; case Direction.right: return { - x: path.End.x + 1, - y: path.End.y + X: path.End.X + 1, + Y: path.End.Y }; case Direction.down: return { - x: path.End.x, - y: path.End.y + 1 + X: path.End.X, + Y: path.End.Y + 1 }; } } @@ -150,8 +150,8 @@ function addTeleportationTiles(board: GameMap, currentPath: Path, steps: number, const possiblePositions = findTeleportationTiles(board); const paths: Path[] = []; for (const pos of possiblePositions) { - if (pos.End.x !== interval(0, board.length - 1, currentPath.End.x) || - pos.End.y !== interval(0, board.length - 1, currentPath.End.y)) { + if (pos.End.X !== interval(0, board.length - 1, currentPath.End.X) || + pos.End.Y !== interval(0, board.length - 1, currentPath.End.Y)) { pos.Path = currentPath.Path; paths.push(...findPossibleRecursive(board, pos, steps, character, characters)); @@ -193,7 +193,7 @@ function findTeleportationTiles(board: GameMap): Path[] { */ function pushPath(board: GameMap, possiblePositions: Path[], x: number, y: number): void { if (board[x][y] !== TileType.wall) { - possiblePositions.push({End: {x, y}, Direction: findDirection(x, y, board.length)}); + possiblePositions.push({End: {X: x, Y: y}, Direction: findDirection(x, y, board.length)}); } } @@ -224,7 +224,7 @@ function findDirection(x: number, y: number, boardSize: number): Direction { */ function isOutsideBoard(currentPos: Path, boardSize: number): boolean { const pos = currentPos.End; - return pos.x < 0 || pos.x >= boardSize || pos.y < 0 || pos.y >= boardSize; + return pos.X < 0 || pos.X >= boardSize || pos.Y < 0 || pos.Y >= boardSize; } /** @@ -234,7 +234,7 @@ function isOutsideBoard(currentPos: Path, boardSize: number): boolean { */ function isWall(board: GameMap, currentPos: Path): boolean { const pos = currentPos.End; - return board[pos.y][pos.x] === TileType.wall; // Shouldn't work, but it does + return board[pos.Y][pos.X] === TileType.wall; // Shouldn't work, but it does } /** @@ -244,7 +244,7 @@ function isWall(board: GameMap, currentPos: Path): boolean { */ function isSpawn(board: GameMap, currentPos: Path) { const pos = currentPos.End; - return board[pos.y][pos.x] === TileType.pacmanSpawn || board[pos.y][pos.x] === TileType.ghostSpawn; + return board[pos.Y][pos.X] === TileType.pacmanSpawn || board[pos.Y][pos.X] === TileType.ghostSpawn; } /** @@ -255,6 +255,6 @@ function isSpawn(board: GameMap, currentPos: Path) { function isOwnSpawn(currentPos: Path, character: Character): boolean { const pos = currentPos.End; const charPos = character.SpawnPosition!.At; - return charPos.x === pos.x && charPos.y === pos.y; + return charPos.X === pos.X && charPos.Y === pos.Y; } diff --git a/pac-man-board-game/ClientApp/src/types/types.d.ts b/pac-man-board-game/ClientApp/src/types/types.d.ts index 68b380e..6426dcd 100644 --- a/pac-man-board-game/ClientApp/src/types/types.d.ts +++ b/pac-man-board-game/ClientApp/src/types/types.d.ts @@ -20,7 +20,7 @@ type SelectedDice = { index: number }; -type Position = { x: number, y: number }; +type Position = { X: number, Y: number }; type GameMap = number[][]; diff --git a/pac-man-board-game/ClientApp/src/utils/actions.ts b/pac-man-board-game/ClientApp/src/utils/actions.ts index 949fbf9..f49f6e2 100644 --- a/pac-man-board-game/ClientApp/src/utils/actions.ts +++ b/pac-man-board-game/ClientApp/src/utils/actions.ts @@ -89,7 +89,7 @@ function removeEatenPellets(data?: MoveCharacterData): void { const pellets = data?.EatenPellets; for (const pellet of pellets ?? []) { - testMap[pellet.y][pellet.x] = TileType.empty; + testMap[pellet.Y][pellet.X] = TileType.empty; } } diff --git a/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts b/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts index fdbc33a..902ca8e 100644 --- a/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts +++ b/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts @@ -8,7 +8,7 @@ let pacMan: Character; beforeEach(() => { pacMan = new PacMan({ - Colour: "yellow", SpawnPosition: {At: {x: 3, y: 3}, Direction: Direction.up} + Colour: "yellow", SpawnPosition: {At: {X: 3, Y: 3}, Direction: Direction.up} }); }); @@ -29,21 +29,21 @@ test("Pac-Man rolls two from start, should return one position", () => { test("Pac-Man rolls three from start, should return two positions", () => { const result = possibleMovesAlgorithm(testMap, pacMan, 3, []); expect(result.length).toBe(2); - arrayEquals(result, [{End: {x: 2, y: 1}, Direction: Direction.left, Path: [{x: 3, y: 2}, {x: 3, y: 1}]}, - {End: {x: 4, y: 1}, Direction: Direction.right, Path: [{x: 3, y: 2}, {x: 3, y: 1}]}]); + arrayEquals(result, [{End: {X: 2, Y: 1}, Direction: Direction.left, Path: [{X: 3, Y: 2}, {X: 3, Y: 1}]}, + {End: {X: 4, Y: 1}, Direction: Direction.right, Path: [{X: 3, Y: 2}, {X: 3, Y: 1}]}]); }); test("Pac-Man rolls four from start, should return two positions", () => { const result = possibleMovesAlgorithm(testMap, pacMan, 4, []); expect(result.length).toBe(2); arrayEquals(result, [{ - End: {x: 1, y: 1}, + End: {X: 1, Y: 1}, Direction: Direction.left, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 2, Y: 1}] }, { - End: {x: 5, y: 1}, + End: {X: 5, Y: 1}, Direction: Direction.right, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}] }]); }); @@ -51,21 +51,21 @@ test("Pac-Man rolls five from start, should return four positions", () => { const result = possibleMovesAlgorithm(testMap, pacMan, 5, []); expect(result.length).toBe(4); arrayEquals(result, [{ - End: {x: 5, y: 0}, + End: {X: 5, Y: 0}, Direction: Direction.up, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}] }, { - End: {x: 6, y: 1}, + End: {X: 6, Y: 1}, Direction: Direction.right, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}] }, { - End: {x: 1, y: 2}, + End: {X: 1, Y: 2}, Direction: Direction.down, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}, {x: 1, y: 1}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 2, Y: 1}, {X: 1, Y: 1}] }, { - End: {x: 5, y: 2}, + End: {X: 5, Y: 2}, Direction: Direction.down, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}] } ]); }); @@ -75,72 +75,72 @@ test("Pac-Man rolls six from start, should return six positions", () => { expect(result.length).toBe(6); arrayEquals(result, [ { - End: {x: 1, y: 3}, + End: {X: 1, Y: 3}, Direction: Direction.down, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}, {x: 1, y: 1}, {x: 1, y: 2}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 2, Y: 1}, {X: 1, Y: 1}, {X: 1, Y: 2}] }, { - End: {x: 0, y: 5}, + End: {X: 0, Y: 5}, Direction: Direction.right, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 0}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 0}] }, { - End: {x: 5, y: 3}, + End: {X: 5, Y: 3}, Direction: Direction.down, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 2}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 2}] }, { - End: {x: 7, y: 1}, + End: {X: 7, Y: 1}, Direction: Direction.right, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 6, y: 1}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 6, Y: 1}] }, { - End: {x: 10, y: 5}, + End: {X: 10, Y: 5}, Direction: Direction.left, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 0}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 0}] }, { - End: {x: 5, y: 10}, + End: {X: 5, Y: 10}, Direction: Direction.up, - Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 0}] + Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 0}] } ]); }); test("Pac-Man rolls four from position [5,1] (right), should return 11", () => { - pacMan.follow({End: {x: 5, y: 1}, Direction: Direction.right}); + pacMan.follow({End: {X: 5, Y: 1}, Direction: Direction.right}); const result = possibleMovesAlgorithm(testMap, pacMan, 4, []); expect(result.length).toBe(11); }); test("Pac-Man rolls four from position [5,1] (left), should return 12", () => { - pacMan.follow({End: {x: 5, y: 1}, Direction: Direction.left}); + pacMan.follow({End: {X: 5, Y: 1}, Direction: Direction.left}); const result = possibleMovesAlgorithm(testMap, pacMan, 4, []); expect(result.length).toBe(12); }); test("Pac-Man rolls three from position [1,5] (left), should return 5", () => { - pacMan.follow({End: {x: 1, y: 5}, Direction: Direction.left}); + pacMan.follow({End: {X: 1, Y: 5}, Direction: Direction.left}); const result = possibleMovesAlgorithm(testMap, pacMan, 3, []); arrayEquals(result, [ - {End: {x: 1, y: 2}, Direction: Direction.up, Path: [{x: 1, y: 4}, {x: 1, y: 3}]}, - {End: {x: 1, y: 8}, Direction: Direction.down, Path: [{x: 1, y: 6}, {x: 1, y: 7}]}, - {End: {x: 5, y: 1}, Direction: Direction.down, Path: [{x: 0, y: 5}, {x: 5, y: 0}]}, - {End: {x: 9, y: 5}, Direction: Direction.left, Path: [{x: 0, y: 5}, {x: 10, y: 5}]}, - {End: {x: 5, y: 9}, Direction: Direction.up, Path: [{x: 0, y: 5}, {x: 5, y: 10}]}, + {End: {X: 1, Y: 2}, Direction: Direction.up, Path: [{X: 1, Y: 4}, {X: 1, Y: 3}]}, + {End: {X: 1, Y: 8}, Direction: Direction.down, Path: [{X: 1, Y: 6}, {X: 1, Y: 7}]}, + {End: {X: 5, Y: 1}, Direction: Direction.down, Path: [{X: 0, Y: 5}, {X: 5, Y: 0}]}, + {End: {X: 9, Y: 5}, Direction: Direction.left, Path: [{X: 0, Y: 5}, {X: 10, Y: 5}]}, + {End: {X: 5, Y: 9}, Direction: Direction.up, Path: [{X: 0, Y: 5}, {X: 5, Y: 10}]}, ]); expect(result.length).toBe(5); }); test("Pac-Man rolls six from position [1,5] (down), should return 17", () => { - pacMan.follow({End: {x: 1, y: 5}, Direction: Direction.down}); + pacMan.follow({End: {X: 1, Y: 5}, Direction: Direction.down}); const result = possibleMovesAlgorithm(testMap, pacMan, 6, []); expect(result.length).toBe(17); }); test("Pac-Man rolls six from position [7,1] (right), path to [9,5] should be five tiles long", () => { - pacMan.follow({End: {x: 7, y: 1}, Direction: Direction.right}); + pacMan.follow({End: {X: 7, Y: 1}, Direction: Direction.right}); const result = possibleMovesAlgorithm(testMap, pacMan, 6, []); expect(result[0].Path?.length).toBe(5); }); test("Pac-Man rolls 5 from position [9,3] (down), should return 5", () => { - pacMan.follow({End: {x: 9, y: 3}, Direction: Direction.down}); + pacMan.follow({End: {X: 9, Y: 3}, Direction: Direction.down}); const result = possibleMovesAlgorithm(testMap, pacMan, 5, []); expect(result.length).toBe(5); }); diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index ba2c874..d2e27b4 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -9,7 +9,7 @@ namespace pacMan.Controllers; [ApiController] [Route("api/[controller]")] -public class GameController : GenericController +public class GameController : GenericController // TODO reconnect using player id { private readonly IActionService _actionService; diff --git a/pac-man-board-game/Game/Actions.cs b/pac-man-board-game/Game/Actions.cs index 5c1ceff..6e2995e 100644 --- a/pac-man-board-game/Game/Actions.cs +++ b/pac-man-board-game/Game/Actions.cs @@ -12,15 +12,10 @@ public enum GameAction public class ActionMessage { - public GameAction Action { get; set; } + public GameAction Action { get; init; } public T? Data { get; set; } - public static ActionMessage FromJson(string json) - { - return JsonSerializer.Deserialize(json)!; - } + public static ActionMessage FromJson(string json) => JsonSerializer.Deserialize(json)!; } -public class ActionMessage : ActionMessage -{ -} \ No newline at end of file +public class ActionMessage : ActionMessage { } diff --git a/pac-man-board-game/Game/Character.cs b/pac-man-board-game/Game/Character.cs index 6b6c6b4..ae251c9 100644 --- a/pac-man-board-game/Game/Character.cs +++ b/pac-man-board-game/Game/Character.cs @@ -2,11 +2,11 @@ namespace pacMan.Game; public class Character : IEquatable { - public required string Colour { get; set; } + public required string Colour { get; init; } public MovePath? Position { get; set; } public bool IsEatable { get; set; } = true; public DirectionalPosition? SpawnPosition { get; set; } - public required CharacterType Type { get; set; } + public required CharacterType? Type { get; init; } public bool Equals(Character? other) { diff --git a/pac-man-board-game/Game/Items/Box.cs b/pac-man-board-game/Game/Items/Box.cs index 3e3bf66..1b05ab9 100644 --- a/pac-man-board-game/Game/Items/Box.cs +++ b/pac-man-board-game/Game/Items/Box.cs @@ -2,7 +2,7 @@ namespace pacMan.Game.Items; public class Box : IEquatable { - public required List? Pellets { get; init; } = new(); + public List? Pellets { get; init; } = new(); public required string Colour { get; init; } public int CountNormal => Pellets?.Count(pellet => !pellet.IsPowerPellet) ?? 0; diff --git a/pac-man-board-game/Game/Items/Player.cs b/pac-man-board-game/Game/Items/Player.cs index ff909cc..2acbf70 100644 --- a/pac-man-board-game/Game/Items/Player.cs +++ b/pac-man-board-game/Game/Items/Player.cs @@ -5,7 +5,7 @@ public interface IPlayer string Name { get; init; } Character PacMan { get; init; } string Colour { get; init; } - Box Box { get; init; } + Box? Box { get; init; } State State { get; set; } } @@ -29,7 +29,7 @@ public class Player : IPlayer, IEquatable public required string Name { get; init; } public required Character PacMan { get; init; } public required string Colour { get; init; } - public required Box Box { get; init; } + public Box? Box { get; init; } public State State { get; set; } = State.WaitingForPlayers; public override bool Equals(object? obj) diff --git a/pac-man-board-game/Game/Positions.cs b/pac-man-board-game/Game/Positions.cs index 3a26b44..5b87162 100644 --- a/pac-man-board-game/Game/Positions.cs +++ b/pac-man-board-game/Game/Positions.cs @@ -1,16 +1,55 @@ namespace pacMan.Game; -public class MovePath +public class MovePath : IEquatable { public Position[]? Path { get; set; } - public required Position End { get; set; } - public required Direction Direction { get; set; } + public required Position End { get; init; } + public required Direction Direction { get; init; } + + public bool Equals(MovePath? other) + { + if (ReferenceEquals(null, other)) return false; + if (ReferenceEquals(this, other)) return true; + return Equals(Path, other.Path) && End.Equals(other.End) && Direction == other.Direction; + } + + public static implicit operator MovePath(DirectionalPosition path) => + new() + { + End = path.At, + Direction = path.Direction + }; + + public override bool Equals(object? obj) + { + if (ReferenceEquals(null, obj)) return false; + if (ReferenceEquals(this, obj)) return true; + return obj.GetType() == GetType() && Equals((MovePath)obj); + } + + public override int GetHashCode() => HashCode.Combine(Path, End, (int)Direction); } -public class Position +public class Position : IEquatable { - public int X { get; set; } = 0; - public int Y { get; set; } = 0; + public int X { get; init; } + public int Y { get; init; } + + public bool Equals(Position? other) + { + if (ReferenceEquals(null, other)) return false; + if (ReferenceEquals(this, other)) return true; + return X == other.X && Y == other.Y; + } + + public override bool Equals(object? obj) + { + if (ReferenceEquals(null, obj)) return false; + if (ReferenceEquals(this, obj)) return true; + return obj.GetType() == GetType() && Equals((Position)obj); + } + + public override int GetHashCode() => HashCode.Combine(X, Y); } public enum Direction @@ -21,8 +60,31 @@ public enum Direction Down } -public class DirectionalPosition +public class DirectionalPosition : IEquatable { - public required Position At { get; set; } - public required Direction Direction { get; set; } -} \ No newline at end of file + public required Position At { get; init; } + public required Direction Direction { get; init; } + + public bool Equals(DirectionalPosition? other) + { + if (ReferenceEquals(null, other)) return false; + if (ReferenceEquals(this, other)) return true; + return At.Equals(other.At) && Direction == other.Direction; + } + + public static explicit operator DirectionalPosition(MovePath path) => + new() + { + At = path.End, + Direction = path.Direction + }; + + public override bool Equals(object? obj) + { + if (ReferenceEquals(null, obj)) return false; + if (ReferenceEquals(this, obj)) return true; + return obj.GetType() == GetType() && Equals((DirectionalPosition)obj); + } + + public override int GetHashCode() => HashCode.Combine(At, (int)Direction); +} diff --git a/pac-man-board-game/Interfaces/IWebSocketService.cs b/pac-man-board-game/Interfaces/IWebSocketService.cs index d8095b9..4f789b4 100644 --- a/pac-man-board-game/Interfaces/IWebSocketService.cs +++ b/pac-man-board-game/Interfaces/IWebSocketService.cs @@ -1,4 +1,5 @@ using System.Net.WebSockets; +using pacMan.Game; using pacMan.Game.Items; using pacMan.Services; @@ -13,5 +14,5 @@ public interface IWebSocketService void SendToAll(ArraySegment segment); Task Receive(WebSocket webSocket, byte[] buffer); Task Close(WebSocket webSocket, WebSocketCloseStatus closeStatus, string? closeStatusDescription); - GameGroup AddPlayer(IPlayer player); + GameGroup AddPlayer(IPlayer player, Queue spawns); } diff --git a/pac-man-board-game/Services/ActionService.cs b/pac-man-board-game/Services/ActionService.cs index 55128ae..5d0f61c 100644 --- a/pac-man-board-game/Services/ActionService.cs +++ b/pac-man-board-game/Services/ActionService.cs @@ -29,7 +29,7 @@ public class ActionService : IActionService _wsService = wsService; } - public GameGroup Group { get; set; } = new(); + public GameGroup? Group { get; set; } public IPlayer? Player { get; set; } @@ -56,14 +56,16 @@ public class ActionService : IActionService { try { - // Receieved JsonElement from frontend - Player = JsonSerializer.Deserialize(message.Data); - Group = _wsService.AddPlayer(Player); + // Receieving JsonElement from frontend + PlayerInfoData data = JsonSerializer.Deserialize(message.Data); + Player = data.Player; + + Group = _wsService.AddPlayer(Player, data.Spawns); } catch (RuntimeBinderException e) { Console.WriteLine(e); - if (message.Data == null) throw new NullReferenceException(); + if (message.Data is null) throw new NullReferenceException(); throw; } @@ -74,7 +76,7 @@ public class ActionService : IActionService public object Ready() { object data; - if (Player != null) + if (Player != null && Group != null) { var players = Group.SetReady(Player).ToArray(); if (players.All(p => p.State == State.Ready)) @@ -96,3 +98,9 @@ public class ActionService : IActionService return data; } } + +public class PlayerInfoData +{ + public required Player Player { get; set; } + public required Queue Spawns { get; set; } +} diff --git a/pac-man-board-game/Services/GameGroup.cs b/pac-man-board-game/Services/GameGroup.cs index 5dd1f48..6c95eb9 100644 --- a/pac-man-board-game/Services/GameGroup.cs +++ b/pac-man-board-game/Services/GameGroup.cs @@ -8,14 +8,20 @@ namespace pacMan.Services; public class GameGroup : IEnumerable { private readonly Random _random = new(); + + public GameGroup(Queue spawns) => Spawns = spawns; + public List Players { get; } = new(); + private Queue Spawns { get; } public IPlayer RandomPlayer => Players[_random.Next(Count)]; + public int Count => Players.Count; public IEnumerator GetEnumerator() => Players.GetEnumerator(); IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); + public event Func, Task>? Connections; public bool AddPlayer(IPlayer player) // TODO if name exists, use that player instead @@ -25,9 +31,18 @@ public class GameGroup : IEnumerable player.State = State.WaitingForPlayers; if (Players.Exists(p => p.Name == player.Name)) return true; Players.Add(player); + if (player.PacMan.SpawnPosition is null) SetSpawn(player); return true; } + private void SetSpawn(IPlayer player) + { + if (player.PacMan.SpawnPosition is not null) return; + var spawn = Spawns.Dequeue(); + player.PacMan.SpawnPosition = spawn; + player.PacMan.Position = spawn; + } + public void SendToAll(ArraySegment segment) => Connections?.Invoke(segment); public IEnumerable SetReady(IPlayer player) diff --git a/pac-man-board-game/Services/WebSocketService.cs b/pac-man-board-game/Services/WebSocketService.cs index 8f32d49..e176719 100644 --- a/pac-man-board-game/Services/WebSocketService.cs +++ b/pac-man-board-game/Services/WebSocketService.cs @@ -1,4 +1,5 @@ using System.Net.WebSockets; +using pacMan.Game; using pacMan.Game.Items; using pacMan.Interfaces; using pacMan.Utils; @@ -56,7 +57,7 @@ public class WebSocketService : IWebSocketService _logger.Log(LogLevel.Information, "WebSocket connection closed"); } - public GameGroup AddPlayer(IPlayer player) + public GameGroup AddPlayer(IPlayer player, Queue spawns) { var index = 0; try @@ -65,7 +66,7 @@ public class WebSocketService : IWebSocketService } catch (ArgumentOutOfRangeException) { - var game = new GameGroup(); + var game = new GameGroup(spawns); game.AddPlayer(player); Games.Add(game); }