diff --git a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx index a9bee66..6f474e6 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -3,11 +3,13 @@ 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 {NormalPellet, PowerPellet} from "../game/pellet"; interface BoardProps extends ComponentProps { characters: Character[], selectedDice?: SelectedDice, - onMove?: (character: Character) => void, + onMove?: Action, map: GameMap } @@ -32,15 +34,34 @@ const Board: Component = ( setHoveredPosition(path); } - function handleMoveCharacter(path: Path): void { + function handleMoveCharacter(destination: Path): void { if (selectedCharacter) { setHoveredPosition(undefined); - selectedCharacter.follow(path); + selectedCharacter.follow(destination); + + pickUpPellets(destination); onMove?.(selectedCharacter); setSelectedCharacter(undefined); } } + function pickUpPellets(destination: Path): void { + if (selectedCharacter instanceof PacMan) { + const pacMan = selectedCharacter as PacMan; + + for (const tile of [...destination.path ?? [], destination.end]) { + const currentTile = map[tile.y][tile.x]; + if (currentTile === TileType.pellet) { + pacMan.box.addPellet(new NormalPellet()); + map[tile.y][tile.x] = TileType.empty; + } else if (currentTile === TileType.powerPellet) { + pacMan.box.addPellet(new PowerPellet()); + map[tile.y][tile.x] = TileType.empty; + } + } + } + } + useEffect(() => { if (selectedCharacter && selectedDice) { const possiblePaths = findPossiblePositions(map, selectedCharacter, selectedDice.value); diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index 63bd1ca..6457c05 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -1,16 +1,21 @@ -import React, {useEffect, useRef, useState} from "react"; +import React, {useEffect, useState} from "react"; import {AllDice} from "./dice"; -import {Action} from "../websockets/actions"; +import {GameAction} from "../websockets/actions"; import GameBoard from "./gameBoard"; import {Character, Ghost, PacMan} from "../game/character"; import WebSocketService from "../websockets/WebSocketService"; import {testMap} from "../game/map"; +import {Direction} from "../game/direction"; +import Box from "../game/box"; const wsService = new WebSocketService("wss://localhost:3000/api/game"); export const GameComponent: Component = () => { - // Better for testing than outside of the component - const characters = useRef([new PacMan("yellow"), new Ghost("purple")]); + // TODO find spawn points + const [characters, setCharacters] = useState([ + new PacMan("yellow", {at: {x: 3, y: 3}, direction: Direction.up}), + new Ghost("purple", {at: {x: 8, y: 3}, direction: Direction.up}) + ]); const [dice, setDice] = useState(); const [selectedDice, setSelectedDice] = useState(); @@ -20,7 +25,7 @@ export const GameComponent: Component = () => { } function rollDice(): void { - wsService.send({Action: Action.rollDice}); + wsService.send({Action: GameAction.rollDice}); } function startGameLoop(): void { @@ -36,13 +41,23 @@ export const GameComponent: Component = () => { const parsed: ActionMessage = JSON.parse(message.data); switch (parsed.Action) { - case Action.rollDice: + case GameAction.rollDice: setDice(parsed.Data as number[]); break; - case Action.moveCharacter: + case GameAction.moveCharacter: setDice(parsed.Data?.dice as number[]); - const character = parsed.Data?.character as Character; - characters.current.find(c => c.color === character.color)?.follow(character.position); + const character = parsed.Data?.character satisfies Ghost | PacMan; + const currentCharacter = characters.find(c => c.color === character.color); + + if (currentCharacter) { + currentCharacter.position = character.position; + } + + // TODO update pellets on other clients (character and on map) + // if (character satisfies PacMan) { + // (characters[currentCharacter] as PacMan).box = new Box(character.box.colour, character.box.pellets); + // console.log(characters[currentCharacter]); + // } break; } } @@ -53,7 +68,7 @@ export const GameComponent: Component = () => { } setSelectedDice(undefined); const data: ActionMessage = { - Action: Action.moveCharacter, + Action: GameAction.moveCharacter, Data: { dice: dice?.length ?? 0 > 0 ? dice : null, character: character @@ -73,11 +88,19 @@ export const GameComponent: Component = () => { return (

Pac-Man The Board Game

-
+
- c instanceof PacMan) as PacMan[]).map(c => +
+

Pac-Man: {c.color}

+

Pellets: {c.box.count}

+

PowerPellets: {c.box.countPowerPellets}

+
) + } +
); diff --git a/pac-man-board-game/ClientApp/src/components/gameTile.tsx b/pac-man-board-game/ClientApp/src/components/gameTile.tsx index e1029cd..5d8052b 100644 --- a/pac-man-board-game/ClientApp/src/components/gameTile.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameTile.tsx @@ -2,15 +2,16 @@ import React, {useEffect, useState} from "react"; import {TileType} from "../game/tileType"; import {Character, Dummy} from "../game/character"; import {Direction} from "../game/direction"; +import {getCSSColour} from "../utils/colours"; interface TileWithCharacterProps extends ComponentProps { possiblePath?: Path, character?: Character, type?: TileType, - handleMoveCharacter?: (path: Path) => void, - handleSelectCharacter?: (character: Character) => void, - handleStartShowPath?: (path: Path) => void, - handleStopShowPath?: () => void, + handleMoveCharacter?: Action, + handleSelectCharacter?: Action, + handleStartShowPath?: Action, + handleStopShowPath?: VoidFunction, isSelected?: boolean, showPath?: boolean } @@ -26,32 +27,34 @@ export const GameTile: Component = ( handleStopShowPath, isSelected = false, showPath = false - }) => { - return ( - handleMoveCharacter?.(possiblePath) : undefined} - onMouseEnter={possiblePath ? () => handleStartShowPath?.(possiblePath) : undefined} - onMouseLeave={handleStopShowPath}> - <> - {character && -
- -
- } - {showPath && } - - -
- ); -}; + }) => ( + handleMoveCharacter?.(possiblePath) : undefined} + onMouseEnter={possiblePath ? () => handleStartShowPath?.(possiblePath) : undefined} + onMouseLeave={handleStopShowPath}> + <> + {character && +
+ +
+ } + {showPath && } + + +
+); -const PathSymbol: Component = () => ( // TODO sometimes shows up when it shouldn't +interface CircleProps extends ComponentProps { + colour?: Colour, +} + +const Circle: Component = ({colour = "white"}) => (
-
+
); @@ -79,18 +82,14 @@ const Tile: Component = ( function setColor(): string { switch (type) { - case TileType.empty: - return "bg-black"; case TileType.wall: return "bg-blue-500"; - case TileType.pellet: - return "bg-yellow-500"; - case TileType.powerPellet: - return "bg-orange-500"; case TileType.ghostSpawn: return "bg-red-500"; case TileType.pacmanSpawn: return "bg-green-500"; + default: + return "bg-black"; } } @@ -113,6 +112,8 @@ const Tile: Component = ( onClick={onClick} onMouseEnter={onMouseEnter} onMouseLeave={onMouseLeave}> + {type === TileType.pellet && } + {type === TileType.powerPellet && } {children}
); @@ -126,7 +127,7 @@ const AddDummy: Component = ({path}) => ( <> {path &&
- +
} diff --git a/pac-man-board-game/ClientApp/src/game/box.ts b/pac-man-board-game/ClientApp/src/game/box.ts new file mode 100644 index 0000000..56ca24e --- /dev/null +++ b/pac-man-board-game/ClientApp/src/game/box.ts @@ -0,0 +1,28 @@ +import {NormalPellet, Pellet, PowerPellet} from "./pellet"; + +export default class Box { + private pellets: Pellet[] = []; + public readonly colour: Colour; + + public constructor(colour: Colour, pellets: Pellet[] = []) { + this.colour = colour; + this.pellets = pellets; + } + + public addPellet(pellet: Pellet): void { + this.pellets.push(pellet); + } + + get powerPellet(): Pellet | undefined { + return this.pellets.find(pellet => pellet instanceof PowerPellet); + } + + get count(): number { + return this.pellets.filter(pellet => pellet instanceof NormalPellet).length; + } + + get countPowerPellets(): number { + return this.pellets.filter(pellet => pellet instanceof PowerPellet).length; + } + +} diff --git a/pac-man-board-game/ClientApp/src/game/character.ts b/pac-man-board-game/ClientApp/src/game/character.ts index c41a8d8..c38a7c2 100644 --- a/pac-man-board-game/ClientApp/src/game/character.ts +++ b/pac-man-board-game/ClientApp/src/game/character.ts @@ -1,20 +1,15 @@ -import {Direction} from "./direction"; - -type CharacterColor = "red" | "blue" | "yellow" | "green" | "purple" | "grey"; - -const defaultDirection: Path = { - end: {x: 0, y: 0}, - direction: Direction.up -}; +import Box from "./box"; export abstract class Character { - public color: CharacterColor; + public color: Colour; public position: Path; - public isEatable: boolean = false; + public isEatable = false; + public readonly spawnPosition: DirectionalPosition; - protected constructor(color: CharacterColor, startPosition = defaultDirection) { + protected constructor(color: Colour, spawnPosition: DirectionalPosition) { this.color = color; - this.position = startPosition; + this.position = {end: spawnPosition.at, direction: spawnPosition.direction}; + this.spawnPosition = spawnPosition; } public follow(path: Path): void { @@ -30,24 +25,27 @@ export abstract class Character { export class PacMan extends Character { - constructor(color: CharacterColor, startPosition = defaultDirection) { - super(color, startPosition); + public box: Box; + + public constructor(color: Colour, spawnPosition: DirectionalPosition) { + super(color, spawnPosition); this.isEatable = true; + this.box = new Box(color); } } export class Ghost extends Character { - constructor(color: CharacterColor, startPosition = defaultDirection) { - super(color, startPosition); + public constructor(color: Colour, spawnPosition: DirectionalPosition) { + super(color, spawnPosition); } } export class Dummy extends Character { - constructor(path: Path) { - super("grey", path); + public constructor(position: DirectionalPosition) { + super("grey", position); } } diff --git a/pac-man-board-game/ClientApp/src/game/pellet.ts b/pac-man-board-game/ClientApp/src/game/pellet.ts new file mode 100644 index 0000000..920bbd0 --- /dev/null +++ b/pac-man-board-game/ClientApp/src/game/pellet.ts @@ -0,0 +1,20 @@ +export abstract class Pellet { + public readonly colour: Colour; + + protected constructor(colour: Colour) { + this.colour = colour; + } + +} + +export class NormalPellet extends Pellet { + public constructor() { + super("white"); + } +} + +export class PowerPellet extends Pellet { + public constructor() { + super("yellow"); + } +} 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 dfaa6a2..a52d42d 100644 --- a/pac-man-board-game/ClientApp/src/types/types.d.ts +++ b/pac-man-board-game/ClientApp/src/types/types.d.ts @@ -5,10 +5,12 @@ type Setter = React.Dispatch>; type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView; type ActionMessage = { - Action: import("../websockets/actions").Action, + Action: import("../websockets/actions").GameAction, Data?: T } +type Action = (obj: T) => void; + type SelectedDice = { value: number, index: number @@ -28,3 +30,5 @@ type Path = { 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 new file mode 100644 index 0000000..3f64e60 --- /dev/null +++ b/pac-man-board-game/ClientApp/src/utils/colours.ts @@ -0,0 +1,27 @@ +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; +} diff --git a/pac-man-board-game/ClientApp/src/websockets/actions.ts b/pac-man-board-game/ClientApp/src/websockets/actions.ts index 22e5723..40c919f 100644 --- a/pac-man-board-game/ClientApp/src/websockets/actions.ts +++ b/pac-man-board-game/ClientApp/src/websockets/actions.ts @@ -1,4 +1,4 @@ -export enum Action { +export enum GameAction { rollDice, moveCharacter, } \ No newline at end of file