From c5d8ecc3620c27615bf9406efed6e1e6c18ac072 Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Sun, 21 May 2023 11:38:16 +0200 Subject: [PATCH] Finished the possibleMovesAlgorithm --- .../.idea/jsLibraryMappings.xml | 6 ++ .../ClientApp/src/components/gameBoard.tsx | 19 ++-- .../src/game/possibleMovesAlgorithm.ts | 92 +++++++++++++++++++ .../ClientApp/src/game/tileType.ts | 8 ++ .../ClientApp/src/utils/game.ts | 46 ---------- 5 files changed, 113 insertions(+), 58 deletions(-) create mode 100644 .idea/.idea.pac-man-board-game/.idea/jsLibraryMappings.xml create mode 100644 pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts create mode 100644 pac-man-board-game/ClientApp/src/game/tileType.ts delete mode 100644 pac-man-board-game/ClientApp/src/utils/game.ts diff --git a/.idea/.idea.pac-man-board-game/.idea/jsLibraryMappings.xml b/.idea/.idea.pac-man-board-game/.idea/jsLibraryMappings.xml new file mode 100644 index 0000000..d23208f --- /dev/null +++ b/.idea/.idea.pac-man-board-game/.idea/jsLibraryMappings.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx index dbe2579..dec0813 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -1,6 +1,7 @@ import React, {useEffect, useState} from "react"; import {Character, PacMan} from "../game/character"; -import {findPossiblePositions} from "../utils/game"; +import findPossiblePositions from "../game/possibleMovesAlgorithm"; +import {TileType} from "../game/tileType"; /** * 0 = empty @@ -24,15 +25,6 @@ const map: number[][] = [ [1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1], ]; -export enum TileType { - empty, - wall, - pellet, - powerPellet, - ghostSpawn, - pacmanSpawn, -} - interface BoardProps extends ComponentProps { characters: Character[], selectedDice?: SelectedDice, @@ -49,7 +41,7 @@ const Board: Component = ( const [selectedCharacter, setSelectedCharacter] = useState(); const [possiblePositions, setPossiblePositions] = useState([]); - function handleSelectCharacter(character: Character) { + function handleSelectCharacter(character: Character): void { setSelectedCharacter(character); } @@ -57,6 +49,8 @@ const Board: Component = ( if (selectedCharacter && selectedDice) { const possiblePositions = findPossiblePositions(map, selectedCharacter.position, selectedDice.value); setPossiblePositions(possiblePositions); + } else { + setPossiblePositions([]); } }, [selectedCharacter, selectedDice]); @@ -88,7 +82,8 @@ const Board: Component = (
{ row.map((tile, colIndex) => - p.x === colIndex && p.y === rowIndex) ? "rounded-full" : ""}`} + p.x === colIndex && p.y === rowIndex) ? + "border-4 border-white" : ""}`} characterClass={`${selectedCharacter?.isAt({x: colIndex, y: rowIndex}) ? "animate-bounce" : ""}`} key={colIndex + rowIndex * colIndex} type={tile} diff --git a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts new file mode 100644 index 0000000..9822f0e --- /dev/null +++ b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts @@ -0,0 +1,92 @@ +import {TileType} from "./tileType"; + +/** + * Finds all the possible positions for the character to move to + * @param board The board the character is on + * @param currentPos The current position of the character + * @param steps The number of steps the character can move + */ +export default function findPossiblePositions(board: number[][], currentPos: CharacterPosition, steps: number): CharacterPosition[] { + const possiblePositions: CharacterPosition[] = []; + findPossibleRecursive(board, currentPos, steps, possiblePositions, []); + return possiblePositions; +} + +function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, steps: number, + possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): CharacterPosition | null { + if (isOutsideBoard(currentPos)) { + addTeleportationTiles(board, currentPos, steps, possibleList, visitedTiles); + } else if (visitedTiles.find(tile => tile.x === currentPos.x && tile.y === currentPos.y)) { + return null; + } else { + if (isWall(board, currentPos)) { + return null; + } + } + visitedTiles.push(currentPos); + if (steps === 0) return currentPos; + + const nextStep = steps - 1; + const result = { + up: findPossibleRecursive(board, {x: currentPos.x, y: currentPos.y + 1}, nextStep, possibleList, visitedTiles), + right: findPossibleRecursive(board, {x: currentPos.x + 1, y: currentPos.y}, nextStep, possibleList, visitedTiles), + down: findPossibleRecursive(board, {x: currentPos.x, y: currentPos.y - 1}, nextStep, possibleList, visitedTiles), + left: findPossibleRecursive(board, {x: currentPos.x - 1, y: currentPos.y}, nextStep, possibleList, visitedTiles), + }; + + pushToList(board, possibleList, Object.values(result)); + return null; +} + +function addTeleportationTiles(board: number[][], currentPos: CharacterPosition, steps: number, + possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): void { + const newPositons: (CharacterPosition | null)[] = []; + const possiblePositions = findTeleportationTiles(board); + for (const pos of possiblePositions) { + if (pos.x !== Math.max(currentPos.x, 0) || pos.y !== Math.max(currentPos.y, 0)) { + newPositons.push(findPossibleRecursive(board, pos, steps, possibleList, visitedTiles)); + } + } + pushToList(board, possibleList, newPositons); +} + +function pushToList(board: number[][], list: CharacterPosition[], newEntries: (CharacterPosition | null)[]): void { + for (const entry of newEntries) { + if (entry !== null && !list.find(p => p.x === entry.x && p.y === entry.y) && !isSpawn(board, entry)) { + list.push(entry); + } + } +} + +function findTeleportationTiles(board: number[][]): CharacterPosition[] { + const possiblePositions: CharacterPosition[] = []; + const edge = [0, board.length - 1]; + + for (const e of edge) { + for (let i = 0; i < board[e].length; i++) { + + if (board[e][i] !== TileType.wall) { + possiblePositions.push({x: i, y: e}); + } + if (board[i][e] !== TileType.wall) { + possiblePositions.push({x: e, y: i}); + } + } + } + + return possiblePositions; +} + +function isOutsideBoard(currentPos: CharacterPosition): boolean { + return currentPos.x < 0 || currentPos.y < 0; +} + +function isWall(board: number[][], currentPos: CharacterPosition): boolean { + return board[currentPos.y][currentPos.x] === TileType.wall; // TODO shouldn't work, but it does +} + +function isSpawn(board: number[][], currentPos: CharacterPosition): boolean { + return board[currentPos.x][currentPos.y] === TileType.pacmanSpawn || + board[currentPos.x][currentPos.y] === TileType.ghostSpawn; +} + diff --git a/pac-man-board-game/ClientApp/src/game/tileType.ts b/pac-man-board-game/ClientApp/src/game/tileType.ts new file mode 100644 index 0000000..2ea8ed0 --- /dev/null +++ b/pac-man-board-game/ClientApp/src/game/tileType.ts @@ -0,0 +1,8 @@ +export enum TileType { + empty, + wall, + pellet, + powerPellet, + ghostSpawn, + pacmanSpawn, +} \ No newline at end of file diff --git a/pac-man-board-game/ClientApp/src/utils/game.ts b/pac-man-board-game/ClientApp/src/utils/game.ts deleted file mode 100644 index e51f53d..0000000 --- a/pac-man-board-game/ClientApp/src/utils/game.ts +++ /dev/null @@ -1,46 +0,0 @@ -import {TileType} from "../components/gameBoard"; - -/** - * Finds all the possible positions for the character to move to - * @param board The board the character is on - * @param currentPos The current position of the character - * @param steps The number of steps the character can move - */ -export function findPossiblePositions(board: number[][], currentPos: CharacterPosition, steps: number): CharacterPosition[] { - const possiblePositions: CharacterPosition[] = []; - findPossibleRecursive(board, currentPos, steps, possiblePositions); - return possiblePositions; -} - -// TODO character phasing through wall next to spawn -function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, steps: number, - possibleList: CharacterPosition[]): CharacterPosition | null { - if (isWall(board, currentPos)) return null; - // TODO handle teleportation - if (steps === 0) return currentPos; - - const result = { - up: findPossibleRecursive(board, {x: currentPos.x, y: currentPos.y + 1}, steps - 1, possibleList), - right: findPossibleRecursive(board, {x: currentPos.x + 1, y: currentPos.y}, steps - 1, possibleList), - down: findPossibleRecursive(board, {x: currentPos.x, y: currentPos.y - 1}, steps - 1, possibleList), - left: findPossibleRecursive(board, {x: currentPos.x - 1, y: currentPos.y}, steps - 1, possibleList), - }; - - for (const [_, value] of Object.entries(result)) { - if (value !== null && !possibleList.find(p => p.x === value.x && p.y === value.y) && !isSpawn(board, value)) { - possibleList.push(value); - } - } - - return null; -} - -function isWall(board: number[][], currentPos: CharacterPosition): boolean { - return board[currentPos.x][currentPos.y] === TileType.wall; -} - -function isSpawn(board: number[][], currentPos: CharacterPosition): boolean { - return board[currentPos.x][currentPos.y] === TileType.pacmanSpawn || - board[currentPos.x][currentPos.y] === TileType.ghostSpawn; -} -