From 58ab8d68ee6538ebdf35122d63fb94edabc38ea5 Mon Sep 17 00:00:00 2001 From: martin Date: Thu, 8 Jun 2023 19:12:10 +0200 Subject: [PATCH] Ghost can eat a pacman and send it back to spawn --- .../ClientApp/src/components/gameBoard.tsx | 16 +++++++- .../ClientApp/src/game/character.ts | 4 ++ .../src/game/possibleMovesAlgorithm.ts | 39 +++++++++++-------- .../tests/game/possibleMovesAlgorithm.test.ts | 8 +++- 4 files changed, 49 insertions(+), 18 deletions(-) diff --git a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx index 8667dda..c68339a 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -37,6 +37,11 @@ const Board: Component = ( function handleMoveCharacter(destination: Path): void { if (selectedCharacter) { setHoveredPosition(undefined); + + if (selectedCharacter.isGhost()) { + tryMovePacManToSpawn(destination); + } + selectedCharacter.follow(destination); const positions = pickUpPellets(destination); @@ -44,6 +49,15 @@ const Board: Component = ( setSelectedCharacter(undefined); } } + + function tryMovePacManToSpawn(destination: Path): void { + const takenChar = characters.find(c => c.isPacMan() && c.isAt(destination.end)); + if (takenChar) { + takenChar.moveToSpawn(); + // TODO send message to other client + // TODO steal from player + } + } function pickUpPellets(destination: Path): Position[] { const positions: Position[] = []; @@ -69,7 +83,7 @@ const Board: Component = ( useEffect(() => { if (selectedCharacter && selectedDice) { - const possiblePaths = findPossiblePositions(map, selectedCharacter, selectedDice.value); + const possiblePaths = findPossiblePositions(map, selectedCharacter, selectedDice.value, characters); setPossiblePositions(possiblePaths); } else { setPossiblePositions([]); diff --git a/pac-man-board-game/ClientApp/src/game/character.ts b/pac-man-board-game/ClientApp/src/game/character.ts index adc8922..d2f4cf9 100644 --- a/pac-man-board-game/ClientApp/src/game/character.ts +++ b/pac-man-board-game/ClientApp/src/game/character.ts @@ -24,6 +24,10 @@ export abstract class Character { return this instanceof PacMan; } + public isGhost(): this is Ghost { + return this instanceof Ghost; + } + public moveToSpawn(): void { this.follow({end: this.spawnPosition.at, direction: this.spawnPosition.direction}); } diff --git a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts index 049a240..d003d85 100644 --- a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts +++ b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts @@ -11,41 +11,45 @@ import {Direction, getDirections} from "./direction"; * @param board The board the character is on * @param character The current position of the character * @param steps The number of steps the character can move + * @param characters * @returns An array of paths the character can move to */ -export default function findPossiblePositions(board: GameMap, character: Character, steps: number): Path[] { - return findPossibleRecursive(board, character.position, steps, character); +export default function findPossiblePositions(board: GameMap, character: Character, steps: number, characters: Character[]): Path[] { + return findPossibleRecursive(board, character.position, steps, character, characters); } /** * Uses recursion to move through the board and find all the possible positions * @param board The board the character is on - * @param currentPos The current path the character is on + * @param currentPath The current path the character is on * @param steps The number of steps the character can move * @param character The current character + * @param characters * @returns An array of paths the character can move to */ -function findPossibleRecursive(board: GameMap, currentPos: Path, steps: number, character: Character): Path[] { +function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number, character: Character, characters: Character[]): Path[] { const paths: Path[] = []; - if (isOutsideBoard(currentPos, board.length)) { + if (isOutsideBoard(currentPath, board.length)) { if (character.isPacMan()) { - return addTeleportationTiles(board, currentPos, steps, character); + return addTeleportationTiles(board, currentPath, steps, character, characters); } - } else if (!isWall(board, currentPos)) { + } else if (!isWall(board, currentPath)) { if (steps <= 0) { - if (!(isSpawn(board, currentPos) && !isOwnSpawn(currentPos, character))) { - paths.push(currentPos); + if (!(isSpawn(board, currentPath) && !isOwnSpawn(currentPath, character))) { + paths.push(currentPath); } - + + } else if (character.isGhost() && characters.find(c => c.isPacMan() && c.isAt(currentPath.end))) { + paths.push(currentPath); } else { - addToPath(currentPos); + addToPath(currentPath); steps--; for (const direction of getDirections()) { - paths.push(...tryMove(board, currentPos, direction, steps, character)); + paths.push(...tryMove(board, currentPath, direction, steps, character, characters)); } } } @@ -72,9 +76,10 @@ function addToPath(currentPos: Path): void { * @param direction The direction to move in * @param steps The number of steps the character can move * @param character The current character + * @param characters * @returns An array of paths the character can move to */ -function tryMove(board: GameMap, path: Path, direction: Direction, steps: number, character: Character): Path[] { +function tryMove(board: GameMap, path: Path, direction: Direction, steps: number, character: Character, characters: Character[]): Path[] { function getNewPosition(): Position { switch (direction) { @@ -102,9 +107,10 @@ function tryMove(board: GameMap, path: Path, direction: Direction, steps: number } 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 - }, steps, character); + }, steps, character, characters); } return []; } @@ -115,8 +121,9 @@ function tryMove(board: GameMap, path: Path, direction: Direction, steps: number * @param currentPath The current path the character is on * @param steps The number of steps the character can move * @param character The current character + * @param characters */ -function addTeleportationTiles(board: GameMap, currentPath: Path, steps: number, character: Character): Path[] { +function addTeleportationTiles(board: GameMap, currentPath: Path, steps: number, character: Character, characters: Character[]): Path[] { const possiblePositions = findTeleportationTiles(board); const paths: Path[] = []; for (const pos of possiblePositions) { @@ -124,7 +131,7 @@ function addTeleportationTiles(board: GameMap, currentPath: Path, steps: number, pos.end.y !== interval(0, board.length - 1, currentPath.end.y)) { pos.path = currentPath.path; - paths.push(...findPossibleRecursive(board, pos, steps, character)); + paths.push(...findPossibleRecursive(board, pos, steps, character, characters)); } } return paths; 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 45a3b5e..29d53ca 100644 --- a/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts +++ b/pac-man-board-game/ClientApp/tests/game/possibleMovesAlgorithm.test.ts @@ -105,7 +105,13 @@ test("Pac-Man rolls six from start, should return six positions", () => { test("Pac-Man rolls four from position [5,1] (right), should return 11", () => { pacMan.follow({end: {x: 5, y: 1}, direction: Direction.right}); const result = possibleMovesAlgorithm(testMap, pacMan, 4); - expect(result.length).toBe(11); // TODO a character can't move to a different character's spawn + 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}); + const result = possibleMovesAlgorithm(testMap, pacMan, 4); + expect(result.length).toBe(12); }); test("Pac-Man rolls three from position [1,5] (left), should return 5", () => {