diff --git a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx index 68b492a..8982deb 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -1,10 +1,8 @@ import React, {useEffect, useState} from "react"; import {Character, PacMan} from "../game/character"; import findPossiblePositions from "../game/possibleMovesAlgorithm"; -import {Direction} from "../game/direction"; import {GameTile} from "./gameTile"; import {TileType} from "../game/tileType"; -import Pellet from "../game/pellet"; interface BoardProps extends ComponentProps { characters: Character[], @@ -49,9 +47,9 @@ const Board: Component = ( setSelectedCharacter(undefined); } } - + function tryMovePacManToSpawn(destination: Path): void { - const takenChar = characters.find(c => c.isPacMan() && c.isAt(destination.end)); + const takenChar = characters.find(c => c.isPacMan() && c.isAt(destination.End)); if (takenChar) { takenChar.moveToSpawn(); // TODO steal from player @@ -63,15 +61,15 @@ const Board: Component = ( if (selectedCharacter instanceof PacMan) { const pacMan = selectedCharacter as PacMan; - for (const tile of [...destination.path ?? [], destination.end]) { + for (const tile of [...destination.Path ?? [], destination.End]) { const currentTile = map[tile.y][tile.x]; - + if (currentTile === TileType.pellet) { - pacMan.box.addPellet(new Pellet()); + // pacMan.box.addPellet(new Pellet()); // TODO update to current player map[tile.y][tile.x] = TileType.empty; positions.push(tile); } else if (currentTile === TileType.powerPellet) { - pacMan.box.addPellet(new Pellet(true)); + // pacMan.box.addPellet(new Pellet(true)); map[tile.y][tile.x] = TileType.empty; positions.push(tile); } @@ -99,10 +97,10 @@ const Board: Component = ( p.end.x === colIndex && p.end.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} + showPath={hoveredPosition?.Path?.find(pos => pos.x === colIndex && pos.y === rowIndex) !== undefined} handleMoveCharacter={handleMoveCharacter} handleSelectCharacter={handleSelectCharacter} handleStartShowPath={handleShowPath} diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index 8c050e5..624ae6f 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -11,20 +11,20 @@ import Player from "../game/player"; const wsService = new WebSocketService("wss://localhost:3000/api/game"); -export const GameComponent: Component<{ player: Player }> = ({player = new Player({colour: "yellow"})}) => { +export const GameComponent: Component<{ player: Player }> = ( + { + player = new Player({ + name: "Martin", + colour: "yellow", + }) + }) => { // TODO find spawn points const [characters, setCharacters] = useState([ - new PacMan({ - colour: "yellow", spawnPosition: {at: {x: 3, y: 3}, direction: Direction.up} - }), - new PacMan({ - colour: "blue", spawnPosition: {at: {x: 7, y: 7}, direction: Direction.down} + new Ghost({ + colour: "purple", spawnPosition: {At: {x: 7, y: 3}, Direction: Direction.up} }), new Ghost({ - colour: "purple", spawnPosition: {at: {x: 7, y: 3}, direction: Direction.up} - }), - new Ghost({ - colour: "purple", spawnPosition: {at: {x: 3, y: 7}, direction: Direction.down} + colour: "purple", spawnPosition: {At: {x: 3, y: 7}, Direction: Direction.down} }) ]); @@ -63,6 +63,10 @@ export const GameComponent: Component<{ player: Player }> = ({player = new Playe updateCharacters(parsed); removeEatenPellets(parsed); break; + case GameAction.playerInfo: + const players = parsed.Data as Player[]; + // TODO set all characters + break; } } @@ -102,11 +106,16 @@ export const GameComponent: Component<{ player: Player }> = ({player = new Playe wsService.send(data); } + async function sendPlayer(): Promise { + await wsService.waitForOpen(); + wsService.send({Action: GameAction.playerInfo, Data: player}); + } + useEffect(() => { wsService.onReceive = doAction; wsService.open(); - // TODO send player info to backend + void sendPlayer(); // TODO send action to backend when all players are ready // The backend should then send the first player as current player return () => wsService.close(); @@ -121,10 +130,10 @@ export const GameComponent: Component<{ player: Player }> = ({player = new Playe { (characters.filter(c => c instanceof PacMan) as PacMan[]).map(c => -
-

Player: {player.colour}

-

Pellets: {player.box.count}

-

PowerPellets: {player.box.countPowerPellets}

+
+

Player: {player.Colour}

+

Pellets: {player.Box.count}

+

PowerPellets: {player.Box.countPowerPellets}

) } = ( isSelected = false, showPath = false }) => ( - handleMoveCharacter?.(possiblePath) : undefined} onMouseEnter={possiblePath ? () => handleStartShowPath?.(possiblePath) : undefined} @@ -146,7 +146,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: @@ -162,7 +162,7 @@ const CharacterComponent: Component = ( return (
onClick?.(character)}>
diff --git a/pac-man-board-game/ClientApp/src/game/box.ts b/pac-man-board-game/ClientApp/src/game/box.ts index 36e0007..f92eadf 100644 --- a/pac-man-board-game/ClientApp/src/game/box.ts +++ b/pac-man-board-game/ClientApp/src/game/box.ts @@ -1,28 +1,28 @@ import Pellet from "./pellet"; export default class Box { - public pellets: Pellet[]; - public readonly colour: Colour; + public Pellets: Pellet[]; + public readonly Colour: Colour; public constructor({colour, pellets = []}: BoxProps) { - this.colour = colour; - this.pellets = pellets; - } - - public addPellet(pellet: Pellet): void { - this.pellets.push(pellet); + this.Colour = colour; + this.Pellets = pellets; } get powerPellet(): Pellet | undefined { - return this.pellets.find(pellet => pellet.isPowerPellet); + return this.Pellets.find(pellet => pellet.isPowerPellet); } get count(): number { - return this.pellets.filter(pellet => !pellet.isPowerPellet).length; + return this.Pellets.filter(pellet => !pellet.isPowerPellet).length; } get countPowerPellets(): number { - return this.pellets.filter(pellet => pellet.isPowerPellet).length; + return this.Pellets.filter(pellet => pellet.isPowerPellet).length; + } + + public addPellet(pellet: Pellet): void { + this.Pellets.push(pellet); } } diff --git a/pac-man-board-game/ClientApp/src/game/character.ts b/pac-man-board-game/ClientApp/src/game/character.ts index ababa3a..8eb29a3 100644 --- a/pac-man-board-game/ClientApp/src/game/character.ts +++ b/pac-man-board-game/ClientApp/src/game/character.ts @@ -1,53 +1,62 @@ import {Direction} from "./direction"; export enum CharacterType { - pacMan = "pacMan", - ghost = "ghost", - dummy = "dummy", + pacMan, + ghost, + dummy, } export class Character { - public readonly colour: Colour; - public position: Path; - public isEatable: boolean; - public readonly spawnPosition: DirectionalPosition; - public readonly type: CharacterType; + public readonly Colour: Colour; + public Position: Path | null; + public IsEatable: boolean; + public readonly SpawnPosition: DirectionalPosition | null; + public readonly Type: CharacterType; public constructor( { colour, - position, + position = null, type = CharacterType.dummy, isEatable = type === CharacterType.pacMan, - spawnPosition + spawnPosition = null }: CharacterProps) { - this.colour = colour; - this.position = position ?? {end: spawnPosition.at, direction: spawnPosition.direction}; - this.isEatable = isEatable; - this.spawnPosition = spawnPosition; - this.type = type; + this.Colour = colour; + this.IsEatable = isEatable; + this.SpawnPosition = spawnPosition; + + this.Position = position ?? spawnPosition ? { + End: spawnPosition!.At, + Direction: spawnPosition!.Direction + } : null; + this.Type = type; } public follow(path: Path): void { - this.position.end = path.end; - this.position.direction = path.direction; - this.position.path = undefined; + if (!this.Position) { + this.Position = path; + } else { + this.Position.End = path.End; + this.Position.Direction = path.Direction; + this.Position.Path = undefined; + } } public isPacMan(): boolean { - return this.type === CharacterType.pacMan; + return this.Type === CharacterType.pacMan; } public isGhost(): boolean { - return this.type === CharacterType.ghost; + return this.Type === CharacterType.ghost; } public moveToSpawn(): void { - this.follow({end: this.spawnPosition.at, direction: this.spawnPosition.direction}); + if (!this.SpawnPosition) return; + this.follow({End: this.SpawnPosition.At, Direction: this.SpawnPosition.Direction}); } public isAt(position: Position): boolean { - return 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; } } @@ -73,7 +82,7 @@ export class Dummy extends Character { colour: "grey", 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/player.ts b/pac-man-board-game/ClientApp/src/game/player.ts index d4d9571..79c29ca 100644 --- a/pac-man-board-game/ClientApp/src/game/player.ts +++ b/pac-man-board-game/ClientApp/src/game/player.ts @@ -1,27 +1,27 @@ import {Character, CharacterType} from "./character"; import Box from "./box"; -import {Direction} from "./direction"; export default class Player { - public readonly pacMan: Character; - public readonly colour: Colour; - public readonly box: Box; + public readonly Name: string; + public readonly PacMan: Character; + public readonly Colour: Colour; + public readonly Box: Box; constructor(props: PlayerProps) { - this.colour = props.colour; - this.box = new Box(props.box ?? {colour: props.colour}); - this.pacMan = new Character(props.pacMan ?? { + this.Name = props.name; + this.Colour = props.colour; + this.Box = new Box(props.box ?? {colour: props.colour}); + this.PacMan = new Character(props.pacMan ?? { colour: props.colour, - spawnPosition: {at: {x: 0, y: 0}, direction: Direction.up}, type: CharacterType.pacMan }); } public stealFrom(other: Player): void { for (let i = 0; i < 2; i++) { - const pellet = other.box.pellets.pop(); + const pellet = other.Box.Pellets.pop(); if (pellet) - this.box.addPellet(pellet); + this.Box.addPellet(pellet); } } diff --git a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts index f95c362..3536d0f 100644 --- a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts +++ b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts @@ -13,7 +13,7 @@ import {Direction, getDirections} from "./direction"; * @returns An array of paths the character can move to */ export default function findPossiblePositions(board: GameMap, character: Character, steps: number, characters: Character[]): Path[] { - return findPossibleRecursive(board, character.position, steps, character, characters); + return findPossibleRecursive(board, character.Position, steps, character, characters); } /** @@ -64,7 +64,7 @@ function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number, * @returns True if the character is a ghost and hits Pac-Man */ function ghostHitsPacMan(character: Character, currentPath: Path, characters: Character[]): boolean { - return character.isGhost() && characters.find(c => c.isPacMan() && c.isAt(currentPath.end)) !== undefined; + return character.isGhost() && characters.find(c => c.isPacMan() && c.isAt(currentPath.End)) !== undefined; } /** @@ -75,7 +75,7 @@ function ghostHitsPacMan(character: Character, currentPath: Path, characters: Ch * @returns True if the character hits another character */ function characterHitsAnotherCharacter(character: Character, currentPath: Path, characters: Character[]): boolean { - return characters.find(c => c !== character && c.isAt(currentPath.end)) !== undefined; + return characters.find(c => c !== character && c.isAt(currentPath.End)) !== undefined; } /** @@ -83,10 +83,10 @@ function characterHitsAnotherCharacter(character: Character, currentPath: Path, * @param currentPos The current path the character is on */ function addToPath(currentPos: Path): void { - if (!currentPos.path) { - currentPos.path = []; - } else if (!currentPos.path.includes(currentPos.end)) { - currentPos.path = [...currentPos.path, currentPos.end]; + if (!currentPos.Path) { + currentPos.Path = []; + } else if (!currentPos.Path.includes(currentPos.End)) { + currentPos.Path = [...currentPos.Path, currentPos.End]; } } @@ -107,31 +107,31 @@ 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 }; } } - if (path.direction !== (direction + 2) % 4) { + if (path.Direction !== (direction + 2) % 4) { // TODO getNewPosition() and check if a character is on the new position return findPossibleRecursive(board, { - end: getNewPosition(), direction: direction, path: path.path + End: getNewPosition(), Direction: direction, Path: path.Path }, steps, character, characters); } return []; @@ -149,10 +149,10 @@ 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; + pos.Path = currentPath.Path; paths.push(...findPossibleRecursive(board, pos, steps, character, characters)); } } @@ -192,7 +192,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, y}, Direction: findDirection(x, y, board.length)}); } } @@ -222,7 +222,7 @@ function findDirection(x: number, y: number, boardSize: number): Direction { * @param boardSize The size of the board */ function isOutsideBoard(currentPos: Path, boardSize: number): boolean { - const pos = currentPos.end; + const pos = currentPos.End; return pos.x < 0 || pos.x >= boardSize || pos.y < 0 || pos.y >= boardSize; } @@ -232,7 +232,7 @@ function isOutsideBoard(currentPos: Path, boardSize: number): boolean { * @param currentPos The current position of the character */ function isWall(board: GameMap, currentPos: Path): boolean { - const pos = currentPos.end; + const pos = currentPos.End; return board[pos.y][pos.x] === TileType.wall; // Shouldn't work, but it does } @@ -242,7 +242,7 @@ function isWall(board: GameMap, currentPos: Path): boolean { * @param currentPos The current position of the character */ function isSpawn(board: GameMap, currentPos: Path) { - const pos = currentPos.end; + const pos = currentPos.End; return board[pos.y][pos.x] === TileType.pacmanSpawn || board[pos.y][pos.x] === TileType.ghostSpawn; } @@ -252,8 +252,8 @@ function isSpawn(board: GameMap, currentPos: Path) { * @param character The current character */ function isOwnSpawn(currentPos: Path, character: Character): boolean { - const pos = currentPos.end; - const charPos = character.spawnPosition.at; + const pos = currentPos.End; + const charPos = character.SpawnPosition.At; return charPos.x === pos.x && charPos.y === pos.y; } 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 bbe2073..0105230 100644 --- a/pac-man-board-game/ClientApp/src/types/props.d.ts +++ b/pac-man-board-game/ClientApp/src/types/props.d.ts @@ -13,9 +13,9 @@ interface ChildProps extends ComponentProps { interface CharacterProps { colour: Colour, - position?: Path, + position?: Path | null, isEatable?: boolean, - spawnPosition: DirectionalPosition, + spawnPosition?: DirectionalPosition | null, type?: import("../game/character").CharacterType, } @@ -25,6 +25,7 @@ interface BoxProps { } interface PlayerProps { + readonly name: string, readonly pacMan?: CharacterProps, readonly colour: Colour, readonly box?: BoxProps, 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 ea0cfb9..b5d91fb 100644 --- a/pac-man-board-game/ClientApp/src/types/types.d.ts +++ b/pac-man-board-game/ClientApp/src/types/types.d.ts @@ -25,14 +25,14 @@ type Position = { x: number, y: number }; type GameMap = number[][]; type DirectionalPosition = { - at: Position, - direction: import("../game/direction").Direction + At: Position, + Direction: import("../game/direction").Direction } type Path = { - path?: Position[], - end: Position, - direction: import("../game/direction").Direction + Path?: Position[] | null, + End: Position, + Direction: import("../game/direction").Direction } type Colour = "white" | "red" | "blue" | "yellow" | "green" | "purple" | "grey"; diff --git a/pac-man-board-game/ClientApp/src/utils/colours.ts b/pac-man-board-game/ClientApp/src/utils/colours.ts index 3f64e60..44136e9 100644 --- a/pac-man-board-game/ClientApp/src/utils/colours.ts +++ b/pac-man-board-game/ClientApp/src/utils/colours.ts @@ -1,27 +1,3 @@ export function getCSSColour(colour: Colour): string { - let tailwindColour: string; - switch (colour) { - case "red": - tailwindColour = "bg-red-500"; - break; - case "blue": - tailwindColour = "bg-blue-500"; - break; - case "yellow": - tailwindColour = "bg-yellow-500"; - break; - case "green": - tailwindColour = "bg-green-500"; - break; - case "purple": - tailwindColour = "bg-purple-500"; - break; - case "grey": - tailwindColour = "bg-gray-500"; - break; - default: - tailwindColour = "bg-white"; - break; - } - return tailwindColour; + return `bg-${colour}${colour === "white" ? "-500" : ""}`; } diff --git a/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts b/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts index 6cf2e67..1701b33 100644 --- a/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts +++ b/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts @@ -8,10 +8,6 @@ interface IWebSocket { export default class WebSocketService { private ws?: WebSocket; private readonly _url: string; - private _onOpen?: VoidFunction; - private _onReceive?: MessageEventFunction; - private _onClose?: VoidFunction; - private _onError?: VoidFunction; constructor(url: string, {onOpen, onReceive, onClose, onError}: IWebSocket = {}) { this._url = url; @@ -21,8 +17,40 @@ export default class WebSocketService { this._onError = onError; } + private _onOpen?: VoidFunction; + + set onOpen(onOpen: VoidFunction) { + this._onOpen = onOpen; + if (!this.ws) return; + this.ws.onopen = onOpen; + } + + private _onReceive?: MessageEventFunction; + + set onReceive(onReceive: MessageEventFunction) { + this._onReceive = onReceive; + if (!this.ws) return; + this.ws.onmessage = onReceive; + } + + private _onClose?: VoidFunction; + + set onClose(onClose: VoidFunction) { + this._onClose = onClose; + if (!this.ws) return; + this.ws.onclose = onClose; + } + + private _onError?: VoidFunction; + + set onError(onError: VoidFunction) { + this._onError = onError; + if (!this.ws) return; + this.ws.onerror = onError; + } + public open(): void { - if (typeof WebSocket === "undefined") return; + if (typeof WebSocket === "undefined" || this.isConnecting()) return; this.ws = new WebSocket(this._url); if (this._onOpen) this.ws.onopen = this._onOpen; if (this._onReceive) this.ws.onmessage = this._onReceive; @@ -30,13 +58,27 @@ export default class WebSocketService { if (this._onError) this.ws.onerror = this._onError; } + public waitForOpen(): Promise { + return new Promise((resolve) => { + const f = () => { + if (this.isOpen()) { + if (this._onOpen) this.onOpen = this._onOpen; + return resolve(); + } + setTimeout(f, 50); + }; + + f(); + }); + } + public send(data: ActionMessage | string): void { if (typeof data !== "string") { data = JSON.stringify(data); } this.ws?.send(data); } - + public async sendAndReceive(data: ActionMessage): Promise { if (!this.isOpen()) return Promise.reject("WebSocket is not open"); @@ -69,27 +111,11 @@ export default class WebSocketService { return this.ws?.readyState === WebSocket?.OPEN; } - set onOpen(onOpen: VoidFunction) { - this._onOpen = onOpen; - if (!this.ws) return; - this.ws.onopen = onOpen; + public isConnecting(): boolean { + return this.ws?.readyState === WebSocket?.CONNECTING; } - set onReceive(onReceive: MessageEventFunction) { - this._onReceive = onReceive; - if (!this.ws) return; - this.ws.onmessage = onReceive; - } - - set onClose(onClose: VoidFunction) { - this._onClose = onClose; - if (!this.ws) return; - this.ws.onclose = onClose; - } - - set onError(onError: VoidFunction) { - this._onError = onError; - if (!this.ws) return; - this.ws.onerror = onError; + public isClosed(): boolean { + return this.ws?.readyState === WebSocket?.CLOSED; } } diff --git a/pac-man-board-game/ClientApp/src/websockets/actions.ts b/pac-man-board-game/ClientApp/src/websockets/actions.ts index 40c919f..43605b8 100644 --- a/pac-man-board-game/ClientApp/src/websockets/actions.ts +++ b/pac-man-board-game/ClientApp/src/websockets/actions.ts @@ -1,4 +1,6 @@ export enum GameAction { rollDice, moveCharacter, + playerInfo, + ready, } \ No newline at end of file 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 3ef46aa..b4b761f 100644 --- a/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts +++ b/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts @@ -8,42 +8,42 @@ 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} }); }); test("Pac-Man rolls one from start, should return one position", () => { const result = possibleMovesAlgorithm(testMap, pacMan, 1, []); expect(result.length).toBe(1); - expect(result[0].path?.length).toBe(0); + expect(result[0].Path?.length).toBe(0); expect(result).toEqual([{end: {x: 3, y: 2}, direction: Direction.up, path: []}]); }); test("Pac-Man rolls two from start, should return one position", () => { const result = possibleMovesAlgorithm(testMap, pacMan, 2, []); expect(result.length).toBe(1); - expect(result[0].path?.length).toBe(1); + expect(result[0].Path?.length).toBe(1); expect(result).toEqual([{end: {x: 3, y: 1}, direction: Direction.up, path: [{x: 3, y: 2}]}]); }); 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}, - direction: Direction.left, - path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}] + End: {x: 1, y: 1}, + Direction: Direction.left, + Path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}] }, { - end: {x: 5, y: 1}, - direction: Direction.right, - path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}] + End: {x: 5, y: 1}, + Direction: Direction.right, + 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}, - direction: Direction.up, - path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}] + 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}] }, { - 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}] + 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}] }, { - 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}] + 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}] }, { - 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}] + 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}] } ]); }); @@ -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}, - direction: Direction.down, - path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}, {x: 1, y: 1}, {x: 1, y: 2}] + 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}] }, { - 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}] + 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}] }, { - 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}] + 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}] }, { - 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}] + 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}] }, { - 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}] + 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}] }, { - 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}] + 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}] } ]); }); 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); + 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 a3cd7a3..b63e52d 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -1,4 +1,5 @@ using System.Net.WebSockets; +using System.Text.Json; using Microsoft.AspNetCore.Mvc; using pacMan.Game; using pacMan.Game.Interfaces; @@ -13,19 +14,17 @@ namespace pacMan.Controllers; public class GameController : GenericController { private readonly IDiceCup _diceCup; - private readonly IPlayer _player; // TODO recieve player from client and choose a starter public GameController(ILogger logger, IWebSocketService wsService) : base(logger, wsService) { _diceCup = new DiceCup(); - _player = new Player - { - Box = new Box() - }; } [HttpGet] - public override async Task Accept() => await base.Accept(); + public override async Task Accept() + { + await base.Accept(); + } protected override ArraySegment Run(WebSocketReceiveResult result, byte[] data) { @@ -48,10 +47,14 @@ public class GameController : GenericController message.Data = rolls; break; - case GameAction.AppendBox: - // TODO - // Add pellets to box - // Forward box to all clients + case GameAction.PlayerInfo: + Player player = JsonSerializer.Deserialize(message.Data); + var group = WsService.AddPlayer(player); // TODO missing some data? + + message.Data = group.Players; + break; + case GameAction.Ready: + // TODO select starter player break; default: Logger.Log(LogLevel.Information, "Forwarding message to all clients"); diff --git a/pac-man-board-game/Controllers/GenericController.cs b/pac-man-board-game/Controllers/GenericController.cs index 4b60e98..9799cb5 100644 --- a/pac-man-board-game/Controllers/GenericController.cs +++ b/pac-man-board-game/Controllers/GenericController.cs @@ -6,15 +6,15 @@ namespace pacMan.Controllers; public abstract class GenericController : ControllerBase { - protected readonly ILogger Logger; - private readonly IWebSocketService _wsService; - private WebSocket? _webSocket; private const int BufferSize = 1024 * 4; + protected readonly ILogger Logger; + protected readonly IWebSocketService WsService; + private WebSocket? _webSocket; protected GenericController(ILogger logger, IWebSocketService wsService) { Logger = logger; - _wsService = wsService; + WsService = wsService; Logger.Log(LogLevel.Debug, "WebSocket Controller created"); } @@ -25,7 +25,7 @@ public abstract class GenericController : ControllerBase using var webSocket = await HttpContext.WebSockets.AcceptWebSocketAsync(); Logger.Log(LogLevel.Information, "WebSocket connection established to {}", HttpContext.Connection.Id); _webSocket = webSocket; - _wsService.Connections += WsServiceOnFire; + WsService.Connections += WsServiceOnFire; await Echo(); } else @@ -37,7 +37,7 @@ public abstract class GenericController : ControllerBase private async Task WsServiceOnFire(ArraySegment segment) { if (_webSocket == null) return; - await _wsService.Send(_webSocket, segment); + await WsService.Send(_webSocket, segment); } @@ -50,23 +50,23 @@ public abstract class GenericController : ControllerBase do { var buffer = new byte[BufferSize]; - result = await _wsService.Receive(_webSocket, buffer); + result = await WsService.Receive(_webSocket, buffer); if (result.CloseStatus.HasValue) break; var segment = Run(result, buffer); - _wsService.SendToAll(segment); + WsService.SendToAll(segment); } while (true); - await _wsService.Close(_webSocket, result.CloseStatus.Value, result.CloseStatusDescription ?? "No reason"); + await WsService.Close(_webSocket, result.CloseStatus.Value, result.CloseStatusDescription ?? "No reason"); } catch (WebSocketException e) { Logger.Log(LogLevel.Error, "{}", e.Message); } - _wsService.Connections -= WsServiceOnFire; + WsService.Connections -= WsServiceOnFire; } protected abstract ArraySegment Run(WebSocketReceiveResult result, byte[] data); diff --git a/pac-man-board-game/Game/Actions.cs b/pac-man-board-game/Game/Actions.cs index 447b5d4..5c1ceff 100644 --- a/pac-man-board-game/Game/Actions.cs +++ b/pac-man-board-game/Game/Actions.cs @@ -6,7 +6,8 @@ public enum GameAction { RollDice, MoveCharacter, - AppendBox + PlayerInfo, + Ready } public class ActionMessage @@ -14,7 +15,10 @@ public class ActionMessage public GameAction Action { get; set; } public T? Data { get; set; } - public static ActionMessage FromJson(string json) => JsonSerializer.Deserialize(json)!; + public static ActionMessage FromJson(string json) + { + return JsonSerializer.Deserialize(json)!; + } } public class ActionMessage : ActionMessage diff --git a/pac-man-board-game/Game/Character.cs b/pac-man-board-game/Game/Character.cs new file mode 100644 index 0000000..d2b5d5b --- /dev/null +++ b/pac-man-board-game/Game/Character.cs @@ -0,0 +1,17 @@ +namespace pacMan.Game; + +public class Character +{ + public required string Colour { get; set; } + public MovePath? Position { get; set; } + public required bool IsEatable { get; set; } + public DirectionalPosition? SpawnPosition { get; set; } + public required CharacterType Type { get; set; } +} + +public enum CharacterType +{ + PacMan, + Ghost, + Dummy +} \ No newline at end of file diff --git a/pac-man-board-game/Game/Interfaces/IBox.cs b/pac-man-board-game/Game/Interfaces/IBox.cs index 337fcd0..6d2b03b 100644 --- a/pac-man-board-game/Game/Interfaces/IBox.cs +++ b/pac-man-board-game/Game/Interfaces/IBox.cs @@ -1,8 +1,9 @@ +using pacMan.Game.Items; + namespace pacMan.Game.Interfaces; public interface IBox : IEnumerable { - void Add(IPellet pellet); - int CountNormal { get; } + void Add(Pellet pellet); } \ No newline at end of file diff --git a/pac-man-board-game/Game/Interfaces/IPellet.cs b/pac-man-board-game/Game/Interfaces/IPellet.cs index 529af4c..39b8f34 100644 --- a/pac-man-board-game/Game/Interfaces/IPellet.cs +++ b/pac-man-board-game/Game/Interfaces/IPellet.cs @@ -1,12 +1,6 @@ namespace pacMan.Game.Interfaces; -public enum PelletType -{ - Normal, - PowerPellet -} - public interface IPellet { - PelletType Get { get; set; } + bool IsPowerPellet { get; init; } } \ No newline at end of file diff --git a/pac-man-board-game/Game/Interfaces/IPlayer.cs b/pac-man-board-game/Game/Interfaces/IPlayer.cs index d6531ae..114ceee 100644 --- a/pac-man-board-game/Game/Interfaces/IPlayer.cs +++ b/pac-man-board-game/Game/Interfaces/IPlayer.cs @@ -1,6 +1,11 @@ +using pacMan.Game.Items; + namespace pacMan.Game.Interfaces; public interface IPlayer { - IBox Box { get; init; } + string Name { get; init; } + Character PacMan { get; init; } + string Colour { get; init; } + Box Box { get; init; } } \ No newline at end of file diff --git a/pac-man-board-game/Game/Items/Box.cs b/pac-man-board-game/Game/Items/Box.cs index 95e5259..da8bae3 100644 --- a/pac-man-board-game/Game/Items/Box.cs +++ b/pac-man-board-game/Game/Items/Box.cs @@ -1,17 +1,26 @@ -using System.Collections; using pacMan.Game.Interfaces; namespace pacMan.Game.Items; -public class Box : IBox +public class Box { - private readonly IList _pellets = new List(); - - public int CountNormal => _pellets.Count(pellet => pellet.Get == PelletType.Normal); + public required List? Pellets { get; init; } = new(); + public required string Colour { get; init; } - public void Add(IPellet pellet) => _pellets.Add(pellet); + public int CountNormal => Pellets?.Count(pellet => !pellet.IsPowerPellet) ?? 0; - public IEnumerator GetEnumerator() => _pellets.GetEnumerator(); + public IEnumerator GetEnumerator() + { + return Pellets?.GetEnumerator() ?? new List.Enumerator(); + } - IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); + // IEnumerator IEnumerable.GetEnumerator() + // { + // return GetEnumerator(); + // } + + public void Add(Pellet pellet) + { + Pellets?.Add(pellet); + } } \ No newline at end of file diff --git a/pac-man-board-game/Game/Items/Pellet.cs b/pac-man-board-game/Game/Items/Pellet.cs index 6a80f60..8dc0812 100644 --- a/pac-man-board-game/Game/Items/Pellet.cs +++ b/pac-man-board-game/Game/Items/Pellet.cs @@ -4,5 +4,5 @@ namespace pacMan.Game.Items; public class Pellet : IPellet { - public PelletType Get { get; set; } + public bool IsPowerPellet { get; init; } } \ No newline at end of file diff --git a/pac-man-board-game/Game/Items/Player.cs b/pac-man-board-game/Game/Items/Player.cs index 7767065..d39490f 100644 --- a/pac-man-board-game/Game/Items/Player.cs +++ b/pac-man-board-game/Game/Items/Player.cs @@ -4,5 +4,8 @@ namespace pacMan.Game.Items; public class Player : IPlayer { - public required IBox Box { get; init; } + public required string Name { get; init; } + public required Character PacMan { get; init; } + public required string Colour { get; init; } + public required Box Box { get; init; } } \ No newline at end of file diff --git a/pac-man-board-game/Game/Positions.cs b/pac-man-board-game/Game/Positions.cs new file mode 100644 index 0000000..3a26b44 --- /dev/null +++ b/pac-man-board-game/Game/Positions.cs @@ -0,0 +1,28 @@ +namespace pacMan.Game; + +public class MovePath +{ + public Position[]? Path { get; set; } + public required Position End { get; set; } + public required Direction Direction { get; set; } +} + +public class Position +{ + public int X { get; set; } = 0; + public int Y { get; set; } = 0; +} + +public enum Direction +{ + Left, + Up, + Right, + Down +} + +public class DirectionalPosition +{ + public required Position At { get; set; } + public required Direction Direction { get; set; } +} \ No newline at end of file diff --git a/pac-man-board-game/Interfaces/IWebSocketService.cs b/pac-man-board-game/Interfaces/IWebSocketService.cs index 88b4507..d85c33e 100644 --- a/pac-man-board-game/Interfaces/IWebSocketService.cs +++ b/pac-man-board-game/Interfaces/IWebSocketService.cs @@ -1,4 +1,6 @@ using System.Net.WebSockets; +using pacMan.Game.Interfaces; +using pacMan.Services; namespace pacMan.Interfaces; @@ -10,4 +12,5 @@ public interface IWebSocketService Task Receive(WebSocket webSocket, byte[] buffer); Task Close(WebSocket webSocket, WebSocketCloseStatus closeStatus, string closeStatusDescription); int CountConnected(); + GameGroup AddPlayer(IPlayer player); } \ No newline at end of file diff --git a/pac-man-board-game/Services/GameGroup.cs b/pac-man-board-game/Services/GameGroup.cs new file mode 100644 index 0000000..4028192 --- /dev/null +++ b/pac-man-board-game/Services/GameGroup.cs @@ -0,0 +1,24 @@ +using pacMan.Game; +using pacMan.Game.Interfaces; + +namespace pacMan.Services; + +public class GameGroup +{ + public List Players { get; } = new(); + public event Func, Task>? Connections; + + public bool AddPlayer(IPlayer player) + { + if (Players.Count >= Rules.MaxPlayers) return false; + if (Players.Exists(p => p.Name == player.Name)) return false; + + Players.Add(player); + return true; + } + + public void SendToAll(ArraySegment segment) + { + Connections?.Invoke(segment); + } +} \ No newline at end of file diff --git a/pac-man-board-game/Services/WebSocketService.cs b/pac-man-board-game/Services/WebSocketService.cs index f6e9240..dc50b1b 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.Interfaces; using pacMan.Interfaces; using pacMan.Utils; @@ -7,7 +8,6 @@ namespace pacMan.Services; public class WebSocketService : IWebSocketService { private readonly ILogger _logger; - public event Func, Task>? Connections; // TODO separate connections into groups (1 event per game) public WebSocketService(ILogger logger) { @@ -15,6 +15,10 @@ public class WebSocketService : IWebSocketService logger.Log(LogLevel.Debug, "WebSocket Service created"); } + public SynchronizedCollection Games { get; } = new(); + + public event Func, Task>? Connections; + public async Task Send(WebSocket webSocket, ArraySegment segment) { await webSocket.SendAsync( @@ -26,7 +30,10 @@ public class WebSocketService : IWebSocketService _logger.Log(LogLevel.Trace, "Message sent to WebSocket"); } - public void SendToAll(ArraySegment segment) => Connections?.Invoke(segment); + public void SendToAll(ArraySegment segment) + { + Connections?.Invoke(segment); + } public async Task Receive(WebSocket webSocket, byte[] buffer) { @@ -47,5 +54,25 @@ public class WebSocketService : IWebSocketService _logger.Log(LogLevel.Information, "WebSocket connection closed"); } - public int CountConnected() => Connections?.GetInvocationList().Length ?? 0; + public int CountConnected() + { + return Connections?.GetInvocationList().Length ?? 0; + } + + public GameGroup AddPlayer(IPlayer player) + { + var index = 0; + try + { + while (!Games[index].AddPlayer(player)) index++; + } + catch (ArgumentOutOfRangeException) + { + var game = new GameGroup(); + game.AddPlayer(player); + Games.Add(game); + } + + return Games[index]; + } } \ No newline at end of file