diff --git a/pac-man-board-game/ClientApp/src/classes/WebSocketService.ts b/pac-man-board-game/ClientApp/src/classes/WebSocketService.ts index cef8e11..22f118a 100644 --- a/pac-man-board-game/ClientApp/src/classes/WebSocketService.ts +++ b/pac-man-board-game/ClientApp/src/classes/WebSocketService.ts @@ -33,29 +33,34 @@ export default class WebSocketService { if (this._onError) this.ws.onerror = this._onError; } - public send(data: WebSocketData): void { + public send(data: ActionRequest | string): void { + if (typeof data !== "string") { + data = JSON.stringify(data); + } this.ws?.send(data); } - - public async sendAndReceive(data: WebSocketData): Promise { + + public async sendAndReceive(data: ActionRequest): Promise { if (!this.isOpen()) return Promise.reject("WebSocket is not open"); - let result: T | undefined; - this.onReceive = (event: MessageEvent) => { - result = JSON.parse(event.data) as T; + let result: R | undefined; + this.ws!.onmessage = (event: MessageEvent) => { + result = JSON.parse(event.data) as R; }; this.send(data); - return new Promise((resolve) => { - function f() { + return new Promise((resolve) => { + const f = () => { if (result === undefined) { setTimeout(f, 50); return; } - } + const resolved = resolve(result); + if (this._onReceive) this.onReceive = this._onReceive; + return resolved; + }; f(); - return resolve(result!); }); } @@ -90,4 +95,4 @@ export default class WebSocketService { if (!this.ws) return; this.ws.onerror = onError; } -} \ No newline at end of file +} diff --git a/pac-man-board-game/ClientApp/src/classes/actions.ts b/pac-man-board-game/ClientApp/src/classes/actions.ts new file mode 100644 index 0000000..f076022 --- /dev/null +++ b/pac-man-board-game/ClientApp/src/classes/actions.ts @@ -0,0 +1,4 @@ +export enum Action { + rollDice, + pickDice, +} \ No newline at end of file diff --git a/pac-man-board-game/ClientApp/src/components/dice.tsx b/pac-man-board-game/ClientApp/src/components/dice.tsx index f65a0ba..1a3ef22 100644 --- a/pac-man-board-game/ClientApp/src/components/dice.tsx +++ b/pac-man-board-game/ClientApp/src/components/dice.tsx @@ -1,14 +1,14 @@ import React from "react"; interface AllDiceProps extends ComponentProps { - values: number[], + values?: number[], } export const AllDice: Component = ({className, values}) => { return ( - <> - {values?.map((value, index) => )} - +
+ {values?.map((value, index) => )} +
); }; @@ -16,8 +16,6 @@ interface DiceProps extends ComponentProps { value?: number, } -export const Dice: Component = ({className, value}) => { - return ( -
{value?.toString()}
- ); -}; +export const Dice: Component = ({className, value}) => ( +

{value?.toString()}

+); diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index b21daca..02751f3 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -3,33 +3,42 @@ import GameCanvas from "../components/gameCanvas"; import Game from "../game/game"; import {AllDice} from "./dice"; +let game: Game; export const GameComponent: Component = () => { - const [dice, setDice] = React.useState([0, 0]); + const [dice, setDice] = React.useState(); + + function startGameLoop() { + if (!game.isConnected()) { + setTimeout(startGameLoop, 50); + return; + } + void game.gameLoop(setDice); + } + + function updateState() { + game.wsService.onReceive = (message) => { + const parsed = JSON.parse(message.data); + if (parsed instanceof Array) { + setDice(parsed); + } + }; + } React.useEffect(() => { - let game: Game = new Game(); + game = new Game(); + updateState(); + game.connectToServer(); - function f() { - if (!game.isConnected()) { - setTimeout(f, 50); - return; - } - game.gameLoop(setDice); - } - f(); - // TODO only call gameLoop after the previous one has finished - // const id = setInterval(() => game.gameLoop(), 5000); - // return () => clearInterval(id); + startGameLoop(); }, []); - - React.useEffect(() => { - console.log(dice); - }, [dice]); return (
-

Pac-Man

+

Pac-Man The Board Game

+
+ +
diff --git a/pac-man-board-game/ClientApp/src/game/game.ts b/pac-man-board-game/ClientApp/src/game/game.ts index aeeecc7..ee9fc7b 100644 --- a/pac-man-board-game/ClientApp/src/game/game.ts +++ b/pac-man-board-game/ClientApp/src/game/game.ts @@ -1,11 +1,12 @@ import WebSocketService from "../classes/WebSocketService"; +import {Action} from "../classes/actions"; export default class Game { - private wsService: WebSocketService; + private _wsService: WebSocketService; constructor() { - this.wsService = new WebSocketService("wss://localhost:3000/api/game"); + this._wsService = new WebSocketService("wss://localhost:3000/api/game"); // Connect to the server // Create players @@ -15,12 +16,10 @@ export default class Game { // Roll to start } - public gameLoop(setDice: Setter): void { + public async gameLoop(setDice: Setter): Promise { // Throw the dices - this.rollDice().then((dices) => { - console.log(dices); - setDice(dices); - }); + const result = await this.rollDice(); + setDice(result); // Choose a dice and move pac-man or a ghost @@ -32,12 +31,12 @@ export default class Game { } public connectToServer(): void { - this.wsService.open(); - this.wsService.registerEvents(); + this._wsService.open(); + this._wsService.registerEvents(); } public isConnected(): boolean { - return this.wsService.isOpen(); + return this._wsService.isOpen(); } private createPlayers(): void { @@ -54,7 +53,7 @@ export default class Game { private async rollDice(): Promise { let result: number[]; - result = await this.wsService.sendAndReceive("roll"); + result = await this._wsService.sendAndReceive({action: Action.rollDice}); return result; } @@ -78,4 +77,8 @@ export default class Game { throw new Error("Not implemented"); } + get wsService(): WebSocketService { + return this._wsService; + } + } \ No newline at end of file diff --git a/pac-man-board-game/ClientApp/src/index.css b/pac-man-board-game/ClientApp/src/index.css index a8af36a..49ad03f 100644 --- a/pac-man-board-game/ClientApp/src/index.css +++ b/pac-man-board-game/ClientApp/src/index.css @@ -10,3 +10,7 @@ h1 { @apply text-4xl; } + +button { + @apply bg-blue-500 hover:bg-blue-600 text-white font-bold py-2 px-4 rounded; +} 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 559ac8f..b624056 100644 --- a/pac-man-board-game/ClientApp/src/types/types.d.ts +++ b/pac-man-board-game/ClientApp/src/types/types.d.ts @@ -3,3 +3,8 @@ type MessageEventFunction = (data: MessageEvent) => void; type Setter = React.Dispatch>; type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView; + +type ActionRequest = { + action: import("../classes/actions").Action, + data?: object +} diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index 532a5cc..188f2b9 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -1,7 +1,8 @@ using System.Net.WebSockets; -using System.Text; using System.Text.Json; +using System.Text.RegularExpressions; using Microsoft.AspNetCore.Mvc; +using pacMan.Game; using pacMan.Game.Interfaces; using pacMan.Game.Items; using pacMan.Interfaces; @@ -14,6 +15,7 @@ namespace pacMan.Controllers; public class GameController : GenericController { private readonly IDiceCup _diceCup; + public GameController(ILogger logger, IWebSocketService wsService) : base(logger, wsService) { _diceCup = new DiceCup(); @@ -25,11 +27,21 @@ public class GameController : GenericController protected override ArraySegment Run(WebSocketReceiveResult result, byte[] data) { var stringResult = data.GetString(data.Length); - Logger.Log(LogLevel.Information, "Received: {}", stringResult); - - var rolls = _diceCup.Roll(); - Logger.Log(LogLevel.Information, "Rolled {}", string.Join(", ", rolls)); + // Removes invalid characters from the string + stringResult = Regex.Replace(stringResult, @"\p{C}+", ""); - return rolls.ToArraySegment(); + Logger.Log(LogLevel.Information, "Received: {}", stringResult); + var action = JsonSerializer.Deserialize(stringResult); + + switch (action?.Action) + { + case GameAction.RollDice: + var rolls = _diceCup.Roll(); + Logger.Log(LogLevel.Information, "Rolled {}", string.Join(", ", rolls)); + + return rolls.ToArraySegment(); + default: + return new ArraySegment("Invalid action"u8.ToArray()); + } } } \ No newline at end of file diff --git a/pac-man-board-game/Game/Action.cs b/pac-man-board-game/Game/Action.cs new file mode 100644 index 0000000..d6bebed --- /dev/null +++ b/pac-man-board-game/Game/Action.cs @@ -0,0 +1,6 @@ +namespace pacMan.Game; + +public enum GameAction +{ + RollDice +} \ No newline at end of file diff --git a/pac-man-board-game/Game/ActionRequest.cs b/pac-man-board-game/Game/ActionRequest.cs new file mode 100644 index 0000000..d95b570 --- /dev/null +++ b/pac-man-board-game/Game/ActionRequest.cs @@ -0,0 +1,7 @@ +namespace pacMan.Game; + +public class ActionRequest +{ + public GameAction Action { get; set; } + public object? Data { get; set; } +} \ No newline at end of file