From a16de14d54b1c5520f58ca51e7bee15abb0ab9ac Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Sun, 21 May 2023 21:42:21 +0200 Subject: [PATCH] Refactored --- .../ClientApp/src/components/gameBoard.tsx | 5 +- .../src/components/gameComponent.tsx | 52 ++++++------ .../ClientApp/src/game/character.ts | 12 +-- pac-man-board-game/ClientApp/src/game/game.ts | 79 ------------------- .../src/game/possibleMovesAlgorithm.ts | 26 +++--- .../ClientApp/src/pages/counter.tsx | 5 +- .../ClientApp/src/types/types.d.ts | 4 +- .../src/websockets/WebSocketService.ts | 4 - .../Controllers/GameController.cs | 2 +- .../Controllers/GenericController.cs | 4 +- pac-man-board-game/pac-man-board-game.csproj | 1 + 11 files changed, 54 insertions(+), 140 deletions(-) delete mode 100644 pac-man-board-game/ClientApp/src/game/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 4dc653c..b087736 100644 --- a/pac-man-board-game/ClientApp/src/components/gameBoard.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameBoard.tsx @@ -41,13 +41,14 @@ const Board: Component = ( const [tileSize, setTileSize] = useState(2); const [selectedCharacter, setSelectedCharacter] = useState(); - const [possiblePositions, setPossiblePositions] = useState([]); + // TODO show the paths to the positions when hovering over a possible position (type Path = CharacterPosition[]) + const [possiblePositions, setPossiblePositions] = useState([]); function handleSelectCharacter(character: Character): void { setSelectedCharacter(character); } - function handleMoveCharacter(position: CharacterPosition): void { + function handleMoveCharacter(position: Position): void { if (selectedCharacter) { selectedCharacter.moveTo(position); onMove?.(selectedCharacter); diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index 0b078c8..6c135cd 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -1,11 +1,11 @@ import React, {useEffect, useRef, useState} from "react"; -import Game from "../game/game"; import {AllDice} from "./dice"; import {Action} from "../websockets/actions"; import GameBoard from "./gameBoard"; import {Character, Ghost, PacMan} from "../game/character"; +import WebSocketService from "../websockets/WebSocketService"; -let game: Game; +const wsService = new WebSocketService("wss://localhost:3000/api/game"); export const GameComponent: Component = () => { // Better for testing than outside of the component @@ -16,39 +16,38 @@ export const GameComponent: Component = () => { function handleDiceClick(selected: SelectedDice): void { setSelectedDice(selected); - game.selectedDice = selected; + } + + function rollDice(): void { + wsService.send({Action: Action.rollDice}); } function startGameLoop(): void { setSelectedDice(undefined); - if (!game.isConnected()) { + if (!wsService.isOpen()) { setTimeout(startGameLoop, 50); return; } - void game.gameLoop(setDice); + rollDice(); } - function updateState(): void { - game.wsService.onReceive = (message) => { - const parsed: ActionMessage = JSON.parse(message.data); + function doAction(message: MessageEvent): void { + const parsed: ActionMessage = JSON.parse(message.data); - switch (parsed.Action) { - case Action.rollDice: - setDice(parsed.Data as number[]); // Updates the state of other players - break; - case Action.moveCharacter: - setDice(parsed.Data?.dice as number[]); - const character = parsed.Data?.character as Character; - characters.current.find(c => c.color === character.color)?.moveTo(character.position); - break; - } - }; + switch (parsed.Action) { + case Action.rollDice: + setDice(parsed.Data as number[]); + break; + case Action.moveCharacter: + setDice(parsed.Data?.dice as number[]); + const character = parsed.Data?.character as Character; + characters.current.find(c => c.color === character.color)?.moveTo(character.position); + break; + } } function onCharacterMove(character: Character): void { if (dice && selectedDice) { - // Remove the dice that was used from the list of dice - dice.splice(selectedDice.index, 1); setDice([...dice]); } @@ -60,16 +59,15 @@ export const GameComponent: Component = () => { character: character } }; - game.wsService.send(data); + wsService.send(data); } useEffect(() => { - game = new Game(); - updateState(); - - game.connectToServer(); + wsService.onReceive = doAction; + wsService.open(); + startGameLoop(); - return () => game.disconnect(); + return () => wsService.close(); }, []); return ( diff --git a/pac-man-board-game/ClientApp/src/game/character.ts b/pac-man-board-game/ClientApp/src/game/character.ts index b89f4a2..cdfa64c 100644 --- a/pac-man-board-game/ClientApp/src/game/character.ts +++ b/pac-man-board-game/ClientApp/src/game/character.ts @@ -2,29 +2,29 @@ type CharacterColor = "red" | "blue" | "yellow" | "green" | "purple"; export abstract class Character { public color: CharacterColor; - public position: CharacterPosition; + public position: Position; - public constructor(color: CharacterColor, startPosition: CharacterPosition = {x: 0, y: 0}) { + public constructor(color: CharacterColor, startPosition: Position = {x: 0, y: 0}) { this.color = color; this.position = startPosition; } - public abstract moveTo(position: CharacterPosition): void; + public abstract moveTo(position: Position): void; - public isAt(position: CharacterPosition): boolean { + public isAt(position: Position): boolean { return this.position.x === position.x && this.position.y === position.y; } } export class PacMan extends Character { - moveTo(position: CharacterPosition): void { + moveTo(position: Position): void { this.position = position; } } export class Ghost extends Character { - moveTo(position: CharacterPosition): void { + moveTo(position: Position): void { this.position = position; } diff --git a/pac-man-board-game/ClientApp/src/game/game.ts b/pac-man-board-game/ClientApp/src/game/game.ts deleted file mode 100644 index 298cf25..0000000 --- a/pac-man-board-game/ClientApp/src/game/game.ts +++ /dev/null @@ -1,79 +0,0 @@ -import WebSocketService from "../websockets/WebSocketService"; -import {Action} from "../websockets/actions"; - -export default class Game { - - private readonly _wsService: WebSocketService; - public selectedDice?: SelectedDice; - - constructor() { - this._wsService = new WebSocketService("wss://localhost:3000/api/game"); - // Connect to the server - - // Create players - - // Pick player pieces - - // Roll to start - } - - public async gameLoop(setDice: Setter): Promise { - // Throw the dice - const result = await this.rollDice(); - const dice = result.Data; - setDice(dice); // Updates the state of the current player - - // Use the remaining dice to move pac-man if the player moved a ghost or vice versa - - // Check if the game is over - - // If not, next player - } - - public connectToServer(): void { - this._wsService.open(); - this._wsService.registerEvents(); - } - - public isConnected(): boolean { - return this._wsService.isOpen(); - } - - public disconnect(): void { - this._wsService.close(); - } - - private createPlayers(): void { - throw new Error("Not implemented"); - } - - private pickPlayerPieces(): void { - throw new Error("Not implemented"); - } - - private rollToStart(): void { - throw new Error("Not implemented"); - } - - private async rollDice(): Promise> { - let result: ActionMessage; - result = await this._wsService.sendAndReceive>({Action: Action.rollDice}); - return result; - } - - private isGameOver(): boolean { - throw new Error("Not implemented"); - } - - private nextPlayer(): void { - throw new Error("Not implemented"); - } - - private chooseCharacter(): void { - throw new Error("Method not implemented."); - } - - get wsService(): WebSocketService { - return this._wsService; - } -} \ No newline at end of file diff --git a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts index e51ef19..e412107 100644 --- a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts +++ b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts @@ -6,14 +6,14 @@ import {TileType} from "./tileType"; * @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[] = []; +export default function findPossiblePositions(board: number[][], currentPos: Position, steps: number): Position[] { + const possiblePositions: Position[] = []; findPossibleRecursive(board, currentPos, steps, possiblePositions, []); return possiblePositions; } -function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, steps: number, - possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): CharacterPosition | null { +function findPossibleRecursive(board: number[][], currentPos: Position, steps: number, + possibleList: Position[], visitedTiles: Position[]): Position | null { if (isOutsideBoard(currentPos, board.length)) { addTeleportationTiles(board, currentPos, steps, possibleList, visitedTiles); } else if (visitedTiles.find(tile => tile.x === currentPos.x && tile.y === currentPos.y)) { @@ -36,9 +36,9 @@ function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, return null; } -function addTeleportationTiles(board: number[][], currentPos: CharacterPosition, steps: number, - possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): void { - const newPositons: (CharacterPosition | null)[] = []; +function addTeleportationTiles(board: number[][], currentPos: Position, steps: number, + possibleList: Position[], visitedTiles: Position[]): void { + const newPositons: (Position | 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)) { @@ -48,7 +48,7 @@ function addTeleportationTiles(board: number[][], currentPos: CharacterPosition, pushToList(board, possibleList, newPositons); } -function pushToList(board: number[][], list: CharacterPosition[], newEntries: (CharacterPosition | null)[]): void { +function pushToList(board: number[][], list: Position[], newEntries: (Position | null)[]): void { for (const entry of newEntries) { if (entry !== null && !list.find(p => p.x === entry.x && p.y === entry.y) && !isOutsideBoard(entry, board.length) && !isSpawn(board, entry)) { list.push(entry); @@ -56,8 +56,8 @@ function pushToList(board: number[][], list: CharacterPosition[], newEntries: (C } } -function findTeleportationTiles(board: number[][]): CharacterPosition[] { - const possiblePositions: CharacterPosition[] = []; +function findTeleportationTiles(board: number[][]): Position[] { + const possiblePositions: Position[] = []; const edge = [0, board.length - 1]; for (const e of edge) { @@ -75,15 +75,15 @@ function findTeleportationTiles(board: number[][]): CharacterPosition[] { return possiblePositions; } -function isOutsideBoard(currentPos: CharacterPosition, boardSize: number): boolean { +function isOutsideBoard(currentPos: Position, boardSize: number): boolean { return currentPos.x < 0 || currentPos.x >= boardSize || currentPos.y < 0 || currentPos.y >= boardSize; } -function isWall(board: number[][], currentPos: CharacterPosition): boolean { +function isWall(board: number[][], currentPos: Position): boolean { return board[currentPos.y][currentPos.x] === TileType.wall; // TODO shouldn't work, but it does } -function isSpawn(board: number[][], currentPos: CharacterPosition): boolean { +function isSpawn(board: number[][], currentPos: Position): 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/pages/counter.tsx b/pac-man-board-game/ClientApp/src/pages/counter.tsx index 552e77a..d6efc59 100644 --- a/pac-man-board-game/ClientApp/src/pages/counter.tsx +++ b/pac-man-board-game/ClientApp/src/pages/counter.tsx @@ -22,10 +22,7 @@ export const Counter: Component = () => { React.useEffect(() => { ws.onReceive = receiveMessage; ws.open(); - ws.registerEvents(); - return () => { - ws.close(); - }; + return () => ws.close(); }, []); return ( 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 1b7acb4..dbd1fa9 100644 --- a/pac-man-board-game/ClientApp/src/types/types.d.ts +++ b/pac-man-board-game/ClientApp/src/types/types.d.ts @@ -1,4 +1,4 @@ -type MessageEventFunction = (data: MessageEvent) => void; +type MessageEventFunction = (data: MessageEvent) => void; type Setter = React.Dispatch>; @@ -14,4 +14,4 @@ type SelectedDice = { index: number }; -type CharacterPosition = { x: number, y: number }; +type Position = { x: number, y: number }; diff --git a/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts b/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts index 7c425ec..39fea08 100644 --- a/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts +++ b/pac-man-board-game/ClientApp/src/websockets/WebSocketService.ts @@ -23,10 +23,6 @@ export default class WebSocketService { public open(): void { this.ws = new WebSocket(this._url); - } - - public registerEvents(): void { - if (!this.ws) return; if (this._onOpen) this.ws.onopen = this._onOpen; if (this._onReceive) this.ws.onmessage = this._onReceive; if (this._onClose) this.ws.onclose = this._onClose; diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index d08c603..d1b468d 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -44,7 +44,7 @@ public class GameController : GenericController message.Data = rolls; break; default: - Logger.Log(LogLevel.Information, "Sending message to all clients"); + Logger.Log(LogLevel.Information, "Forwarding message to all clients"); break; } } diff --git a/pac-man-board-game/Controllers/GenericController.cs b/pac-man-board-game/Controllers/GenericController.cs index 3bc7e17..cb33ad9 100644 --- a/pac-man-board-game/Controllers/GenericController.cs +++ b/pac-man-board-game/Controllers/GenericController.cs @@ -36,12 +36,12 @@ public abstract class GenericController : ControllerBase { try { - var buffer = new byte[BufferSize]; WebSocketReceiveResult? result; do { + var buffer = new byte[BufferSize]; result = await _wsService.Receive(webSocket, buffer); - + if (result.CloseStatus.HasValue) break; var segment = Run(result, buffer); diff --git a/pac-man-board-game/pac-man-board-game.csproj b/pac-man-board-game/pac-man-board-game.csproj index a1679ee..e62b20b 100644 --- a/pac-man-board-game/pac-man-board-game.csproj +++ b/pac-man-board-game/pac-man-board-game.csproj @@ -38,6 +38,7 @@ +