From 89df373d45a2f577c7f63b55741e12d63b661535 Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Sat, 20 May 2023 23:57:20 +0200 Subject: [PATCH] Can click a player to select it. Started implementing algorithm to find paths --- .../ClientApp/src/components/gameBoard.tsx | 73 +++++++++++++++---- .../src/components/gameComponent.tsx | 7 +- .../ClientApp/src/game/character.ts | 4 + .../ClientApp/src/utils/game.ts | 46 ++++++++++++ 4 files changed, 111 insertions(+), 19 deletions(-) create mode 100644 pac-man-board-game/ClientApp/src/utils/game.ts diff --git a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx index ed6150e..dbe2579 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -1,5 +1,6 @@ import React, {useEffect, useState} from "react"; import {Character, PacMan} from "../game/character"; +import {findPossiblePositions} from "../utils/game"; /** * 0 = empty @@ -23,7 +24,7 @@ const map: number[][] = [ [1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1], ]; -enum TileType { +export enum TileType { empty, wall, pellet, @@ -33,15 +34,34 @@ enum TileType { } interface BoardProps extends ComponentProps { - characters: Character[]; + characters: Character[], + selectedDice?: SelectedDice, } -const Board: Component = ({className, characters}) => { +const Board: Component = ( + { + className, + characters, + selectedDice + }) => { - const [tileSize, setTileSize] = useState(2); + const [tileSize, setTileSize] = useState(2); + const [selectedCharacter, setSelectedCharacter] = useState(); + const [possiblePositions, setPossiblePositions] = useState([]); + + function handleSelectCharacter(character: Character) { + setSelectedCharacter(character); + } useEffect(() => { - + if (selectedCharacter && selectedDice) { + const possiblePositions = findPossiblePositions(map, selectedCharacter.position, selectedDice.value); + setPossiblePositions(possiblePositions); + } + }, [selectedCharacter, selectedDice]); + + useEffect(() => { + for (const character of characters) { // TODO make more dynamic if (character instanceof PacMan) { character.position = {x: 3, y: 3}; @@ -68,10 +88,14 @@ const Board: Component = ({className, characters}) => {
{ row.map((tile, colIndex) => - p.x === colIndex && p.y === rowIndex) ? "rounded-full" : ""}`} + characterClass={`${selectedCharacter?.isAt({x: colIndex, y: rowIndex}) ? "animate-bounce" : ""}`} + key={colIndex + rowIndex * colIndex} type={tile} size={tileSize} - character={characters.find(c => c.position.x === colIndex && c.position.y === rowIndex)}/> + character={characters.find(c => c.isAt({x: colIndex, y: rowIndex}))} + onClick={handleSelectCharacter} + /> ) }
) @@ -86,9 +110,19 @@ interface TileProps extends ComponentProps { size: number, type?: TileType, character?: Character, + onClick?: (character: Character) => void, + characterClass?: string, } -const Tile: Component = ({size, type = TileType.empty, character}) => { +const Tile: Component = ( + { + size, + type = TileType.empty, + character, + onClick, + className, + characterClass, + }) => { function setColor(): string { switch (type) { @@ -108,23 +142,30 @@ const Tile: Component = ({size, type = TileType.empty, character}) => } return ( -
+
{character &&
- +
} +
); }; interface CharacterComponentProps extends ComponentProps { character: Character, + onClick?: (character: Character) => void, } -const CharacterComponent: Component = ({character}) => { - return ( -
- ); -}; +const CharacterComponent: Component = ( + { + character, + onClick, + className + }) => ( +
onClick(character) : undefined}/> +); diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index f839d7a..73fb0be 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -1,4 +1,4 @@ -import React, {useState} from "react"; +import React, {useEffect, useState} from "react"; import Game from "../game/game"; import {AllDice} from "./dice"; import {Action} from "../websockets/actions"; @@ -20,6 +20,7 @@ export const GameComponent: Component = () => { } function startGameLoop(): void { + setSelectedDice(undefined); if (!game.isConnected()) { setTimeout(startGameLoop, 50); return; @@ -39,7 +40,7 @@ export const GameComponent: Component = () => { }; } - React.useEffect(() => { + useEffect(() => { game = new Game(); updateState(); @@ -55,7 +56,7 @@ export const GameComponent: Component = () => {
- +
); }; diff --git a/pac-man-board-game/ClientApp/src/game/character.ts b/pac-man-board-game/ClientApp/src/game/character.ts index fe3c261..75d4950 100644 --- a/pac-man-board-game/ClientApp/src/game/character.ts +++ b/pac-man-board-game/ClientApp/src/game/character.ts @@ -10,6 +10,10 @@ export abstract class Character { } public abstract moveTo(position: CharacterPosition): void; + + public isAt(position: CharacterPosition): boolean { + return this.position.x === position.x && this.position.y === position.y; + } } export class PacMan extends Character { diff --git a/pac-man-board-game/ClientApp/src/utils/game.ts b/pac-man-board-game/ClientApp/src/utils/game.ts new file mode 100644 index 0000000..e51f53d --- /dev/null +++ b/pac-man-board-game/ClientApp/src/utils/game.ts @@ -0,0 +1,46 @@ +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; +} +