From f0a701a2fd888ffea400b2f113e3baf7224d3f62 Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Thu, 20 Jul 2023 18:26:41 +0200 Subject: [PATCH 1/5] Added gameId to Game path --- pac-man-board-game/ClientApp/src/AppRoutes.tsx | 2 +- pac-man-board-game/ClientApp/src/pages/lobby.tsx | 9 +++++---- pac-man-board-game/Controllers/GameController.cs | 2 +- 3 files changed, 7 insertions(+), 6 deletions(-) diff --git a/pac-man-board-game/ClientApp/src/AppRoutes.tsx b/pac-man-board-game/ClientApp/src/AppRoutes.tsx index a1d7816..ba2695d 100644 --- a/pac-man-board-game/ClientApp/src/AppRoutes.tsx +++ b/pac-man-board-game/ClientApp/src/AppRoutes.tsx @@ -15,7 +15,7 @@ const AppRoutes = [ element: }, { - path: "/game", + path: "/game/:id", element: , }, { diff --git a/pac-man-board-game/ClientApp/src/pages/lobby.tsx b/pac-man-board-game/ClientApp/src/pages/lobby.tsx index 08701de..bc5fcaa 100644 --- a/pac-man-board-game/ClientApp/src/pages/lobby.tsx +++ b/pac-man-board-game/ClientApp/src/pages/lobby.tsx @@ -26,7 +26,7 @@ const LobbyPage: FC = () => { // TODO check if player is defined in storage, if if (response.ok) { const data = await response.json(); console.debug("Game created: ", data); - // TODO redirect to game page + navigate("/game/" + data.id) } else { const data = await response.text(); console.error("Error: ", data); @@ -51,6 +51,7 @@ const GameTable: FC = ({className}) => { const data = useAtomValue(fetchAtom); const thisPlayer = useAtomValue(thisPlayerAtom); + const navigate = useNavigate(); async function joinGame(gameId: string): Promise { if (thisPlayer === undefined) throw new Error("Player is undefined"); @@ -60,10 +61,10 @@ const GameTable: FC = ({className}) => { const result = await postData("/game/join/" + gameId, {body: thisPlayer}); if (result.ok) { - console.debug("Joined game " + gameId, await result.json()); - // TODO redirect to game page + console.debug("Joined game " + gameId, await result.text()); + navigate("/game/" + gameId); } else { - console.error("Failed to join game " + gameId, await result.json()); + console.error("Failed to join game " + gameId, await result.text()); // TODO show error message } } diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index 7a89127..80a96cb 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -10,7 +10,7 @@ namespace pacMan.Controllers; [ApiController] [Route("api/[controller]")] -public class GameController : GenericController // TODO reconnect using player id +public class GameController : GenericController { private readonly IActionService _actionService; private readonly GameService _gameService; From 7af502f570cc8e567e87c45dbd0d6802b36d19a1 Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Thu, 20 Jul 2023 22:56:35 +0200 Subject: [PATCH 2/5] fixed state not updating when joining from lobby --- BackendTests/Services/GameTests.cs | 10 +++++----- pac-man-board-game/ClientApp/src/utils/actions.ts | 1 + pac-man-board-game/Controllers/GameController.cs | 2 +- pac-man-board-game/GameStuff/Actions.cs | 3 ++- pac-man-board-game/Services/ActionService.cs | 6 +++--- pac-man-board-game/Services/Game.cs | 5 +++-- 6 files changed, 15 insertions(+), 12 deletions(-) diff --git a/BackendTests/Services/GameTests.cs b/BackendTests/Services/GameTests.cs index 0c6b448..0d7b17b 100644 --- a/BackendTests/Services/GameTests.cs +++ b/BackendTests/Services/GameTests.cs @@ -110,8 +110,8 @@ public class GameTests _game.AddPlayer(_redPlayer); _game.AddPlayer(_bluePlayer); - _game.SetReady(_redPlayer); - _game.SetReady(_bluePlayer); + _game.SetReady(_redPlayer.Username); + _game.SetReady(_bluePlayer.Username); _game.SetAllInGame(); Assert.That(_game.AddPlayer(_greenPlayer), Is.False); @@ -154,7 +154,7 @@ public class GameTests _game.AddPlayer(_redPlayer); _game.AddPlayer(_bluePlayer); - var players = _game.SetReady(_redPlayer).ToList(); + var players = _game.SetReady(_redPlayer.Username).ToList(); Assert.Multiple(() => { @@ -171,7 +171,7 @@ public class GameTests Assert.That(_redPlayer.State, Is.Not.EqualTo(State.Ready)); - _game.SetReady(_redPlayer); + _game.SetReady(_redPlayer.Username); Assert.That(_redPlayer.State, Is.EqualTo(State.Ready)); } @@ -179,7 +179,7 @@ public class GameTests [Test] public void SetReady_WhenPlayerIsNotInPlayers() { - Assert.Throws(() => _game.SetReady(_redPlayer)); + Assert.Throws(() => _game.SetReady(_redPlayer.Username)); } #endregion diff --git a/pac-man-board-game/ClientApp/src/utils/actions.ts b/pac-man-board-game/ClientApp/src/utils/actions.ts index d5f3bb7..7dac3f5 100644 --- a/pac-man-board-game/ClientApp/src/utils/actions.ts +++ b/pac-man-board-game/ClientApp/src/utils/actions.ts @@ -12,6 +12,7 @@ export enum GameAction { playerInfo, ready, nextPlayer, + disconnect, } const store = getDefaultStore(); diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index 80a96cb..b496466 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -82,7 +82,7 @@ public class GameController : GenericController protected override Task Echo() { - _gameService.Connections += WsServiceOnFire; + _gameService.Connections += WsServiceOnFire; // TODO move to ActionService // _actionService.Game.Connections += WsServiceOnFire; return base.Echo(); } diff --git a/pac-man-board-game/GameStuff/Actions.cs b/pac-man-board-game/GameStuff/Actions.cs index eeeb847..94d1e41 100644 --- a/pac-man-board-game/GameStuff/Actions.cs +++ b/pac-man-board-game/GameStuff/Actions.cs @@ -9,7 +9,8 @@ public enum GameAction MoveCharacter, PlayerInfo, Ready, - NextPlayer + NextPlayer, + Disconnect } public class ActionMessage diff --git a/pac-man-board-game/Services/ActionService.cs b/pac-man-board-game/Services/ActionService.cs index bd09d77..a4b2237 100644 --- a/pac-man-board-game/Services/ActionService.cs +++ b/pac-man-board-game/Services/ActionService.cs @@ -38,10 +38,10 @@ public class ActionService : IActionService message.Data = message.Action switch { GameAction.RollDice => RollDice(), + GameAction.MoveCharacter => HandleMoveCharacter(message.Data), GameAction.PlayerInfo => SetPlayerInfo(message.Data), GameAction.Ready => Ready(), GameAction.NextPlayer => FindNextPlayer(), - GameAction.MoveCharacter => HandleMoveCharacter(message.Data), _ => message.Data }; } @@ -55,7 +55,7 @@ public class ActionService : IActionService return rolls; } - public List SetPlayerInfo(JsonElement? jsonElement) + public List SetPlayerInfo(JsonElement? jsonElement) // TODO split up into two actions { var data = jsonElement?.Deserialize() ?? throw new NullReferenceException("Data is null"); Player = data.Player; @@ -83,7 +83,7 @@ public class ActionService : IActionService object data; if (Player != null && Game != null) { - var players = Game.SetReady(Player).ToArray(); + var players = Game.SetReady(Player.Username).ToArray(); // TODO roll to start Game.Shuffle(); var allReady = players.All(p => p.State == State.Ready); diff --git a/pac-man-board-game/Services/Game.cs b/pac-man-board-game/Services/Game.cs index a58ec20..25eda66 100644 --- a/pac-man-board-game/Services/Game.cs +++ b/pac-man-board-game/Services/Game.cs @@ -81,9 +81,10 @@ public class Game // TODO handle disconnects and reconnects public void SendToAll(ArraySegment segment) => Connections?.Invoke(segment); - public IEnumerable SetReady(Player player) + public IEnumerable SetReady(string username) { - if (!Players.Contains(player)) + var player = Players.FirstOrDefault(p => p.Username == username); + if (player is null) throw new PlayerNotFoundException("The player was not found in the game group."); player.State = State.Ready; return Players; From 554b8cff2a1658ca8c69a33fa5d24c033e9c267d Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Thu, 20 Jul 2023 23:54:20 +0200 Subject: [PATCH 3/5] Moved event to Game, added leave game button --- .../ClientApp/src/AppRoutes.tsx | 3 +- .../src/components/gameComponent.tsx | 18 +++++-- .../ClientApp/src/utils/actions.ts | 9 ++-- .../Controllers/GameController.cs | 26 +++------- pac-man-board-game/GameStuff/Character.cs | 2 +- pac-man-board-game/Services/ActionService.cs | 52 ++++++++++++++----- pac-man-board-game/Services/GameService.cs | 7 --- 7 files changed, 67 insertions(+), 50 deletions(-) diff --git a/pac-man-board-game/ClientApp/src/AppRoutes.tsx b/pac-man-board-game/ClientApp/src/AppRoutes.tsx index ba2695d..22c388c 100644 --- a/pac-man-board-game/ClientApp/src/AppRoutes.tsx +++ b/pac-man-board-game/ClientApp/src/AppRoutes.tsx @@ -1,6 +1,5 @@ import React from "react"; import {Counter} from "./pages/counter"; -import Home from "./pages/home"; import Game from "./pages/game"; import LobbyPage from "./pages/lobby"; import Login from "./pages/login"; @@ -8,7 +7,7 @@ import Login from "./pages/login"; const AppRoutes = [ { index: true, - element: + element: }, { path: "/counter", diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index 1b52fe4..ff37c8a 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -9,19 +9,21 @@ import PlayerStats from "../components/playerStats"; import {useAtom, useAtomValue, useSetAtom} from "jotai"; import {diceAtom, ghostsAtom, playersAtom, rollDiceButtonAtom, selectedDiceAtom} from "../utils/state"; import GameButton from "./gameButton"; +import {Button} from "./button"; +import {useNavigate} from "react-router-dom"; const wsService = new WebSocketService(import.meta.env.VITE_API_WS); -// TODO bug, when taking player on last dice, the currentPlayer changes and the wrong character get to steal -// TODO bug, first player can sometimes roll dice twice (maybe only on firefox) +// TODO bug, when taking player on last dice, the currentPlayer changes and the wrong character gets to steal +// TODO bug, first player can sometimes roll dice twice +// TODO bug, when refreshing page, the player is still in disconnected state // TODO bug, when refreshing page, the characters are reset for all players // TODO bug, when refreshing page, some data is missing until other clients make a move +// TODO bug, teleportation doesn't work // TODO store map in backend and save it in state on each client // TODO add debug menu on dev, for testing and cheating -// TODO join/new game lobby // TODO sign up player page -// TODO sign in page // TODO show box with collected pellets // TODO layout @@ -33,6 +35,8 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map const setActiveRollDiceButton = useSetAtom(rollDiceButtonAtom); const ghosts = useAtomValue(ghostsAtom); + const navigate = useNavigate(); + function rollDice(): void { if (!player.isTurn()) return; @@ -79,6 +83,11 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map wsService.send({action: GameAction.nextPlayer}); } + function leaveGame(): void { + wsService.send({action: GameAction.disconnect}); + navigate("/lobby"); + } + useEffect(() => { wsService.onReceive = doAction; wsService.open(); @@ -90,6 +99,7 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map return ( <> +
{players?.map(p => )}
diff --git a/pac-man-board-game/ClientApp/src/utils/actions.ts b/pac-man-board-game/ClientApp/src/utils/actions.ts index 7dac3f5..cbb869c 100644 --- a/pac-man-board-game/ClientApp/src/utils/actions.ts +++ b/pac-man-board-game/ClientApp/src/utils/actions.ts @@ -50,6 +50,9 @@ export const doAction: MessageEventFunction = (event): void => { // TODO case GameAction.nextPlayer: nextPlayer(message.data); break; + case GameAction.disconnect: + updatePlayers(message.data); + break; } }; @@ -61,14 +64,12 @@ type MoveCharacterData = { dice: number[], players: PlayerProps[], ghosts: Chara function moveCharacter(data?: MoveCharacterData): void { store.set(diceAtom, data?.dice); - updatePlayers(data); + updatePlayers(data?.players); updateGhosts(data); removeEatenPellets(data); } -function updatePlayers(data?: MoveCharacterData): void { - const updatedPlayers = data?.players; - +function updatePlayers(updatedPlayers?: PlayerProps[]): void { if (updatedPlayers) { const newList: Player[] = updatedPlayers.map(p => new Player(p)); store.set(playersAtom, newList); diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index b496466..199192a 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -66,6 +66,11 @@ public class GameController : GenericController } } + protected override Task Echo() + { + _actionService.WebSocket = WebSocket ?? throw new NullReferenceException("WebSocket is null"); + return base.Echo(); + } protected override ArraySegment Run(WebSocketReceiveResult result, byte[] data) { @@ -78,24 +83,7 @@ public class GameController : GenericController return action.ToArraySegment(); } - protected override void Send(ArraySegment segment) => _gameService.SendToAll(segment); + protected override void Send(ArraySegment segment) => _actionService.SendToAll(segment); - protected override Task Echo() - { - _gameService.Connections += WsServiceOnFire; // TODO move to ActionService - // _actionService.Game.Connections += WsServiceOnFire; - return base.Echo(); - } - - protected override void Disconnect() - { - _gameService.Connections -= WsServiceOnFire; - _actionService.Disconnect(); - } - - private async Task WsServiceOnFire(ArraySegment segment) - { - if (WebSocket == null) return; - await GameService.Send(WebSocket, segment); - } + protected override void Disconnect() => _actionService.Disconnect(); } diff --git a/pac-man-board-game/GameStuff/Character.cs b/pac-man-board-game/GameStuff/Character.cs index fc451c7..66211e2 100644 --- a/pac-man-board-game/GameStuff/Character.cs +++ b/pac-man-board-game/GameStuff/Character.cs @@ -31,7 +31,7 @@ public class Character : IEquatable return obj.GetType() == GetType() && Equals((Character)obj); } - public override int GetHashCode() => HashCode.Combine(Colour, Position, IsEatable, SpawnPosition, (int)Type); + public override int GetHashCode() => HashCode.Combine(Colour, Position, IsEatable, SpawnPosition, (int?)Type); } public enum CharacterType diff --git a/pac-man-board-game/Services/ActionService.cs b/pac-man-board-game/Services/ActionService.cs index a4b2237..54923d3 100644 --- a/pac-man-board-game/Services/ActionService.cs +++ b/pac-man-board-game/Services/ActionService.cs @@ -1,3 +1,4 @@ +using System.Net.WebSockets; using System.Text.Json; using System.Text.Json.Serialization; using pacMan.GameStuff; @@ -9,12 +10,15 @@ public interface IActionService { Player Player { set; } Game Game { set; } + WebSocket? WebSocket { set; } void DoAction(ActionMessage message); List RollDice(); List SetPlayerInfo(JsonElement? jsonElement); - object? HandleMoveCharacter(JsonElement? jsonElement); // TODO test + object? HandleMoveCharacter(JsonElement? jsonElement); object Ready(); string FindNextPlayer(); + List LeaveGame(); + void SendToAll(ArraySegment segment); void Disconnect(); } @@ -29,6 +33,8 @@ public class ActionService : IActionService _gameService = gameService; } + public WebSocket? WebSocket { private get; set; } + public Game? Game { get; set; } public Player? Player { get; set; } @@ -42,6 +48,7 @@ public class ActionService : IActionService GameAction.PlayerInfo => SetPlayerInfo(message.Data), GameAction.Ready => Ready(), GameAction.NextPlayer => FindNextPlayer(), + GameAction.Disconnect => LeaveGame(), _ => message.Data }; } @@ -55,19 +62,28 @@ public class ActionService : IActionService return rolls; } + public object? HandleMoveCharacter(JsonElement? jsonElement) + { + if (Game != null && jsonElement.HasValue) + Game.Ghosts = jsonElement.Value.GetProperty("ghosts").Deserialize>() ?? + throw new JsonException("Ghosts is null"); + + return jsonElement; + } + public List SetPlayerInfo(JsonElement? jsonElement) // TODO split up into two actions { var data = jsonElement?.Deserialize() ?? throw new NullReferenceException("Data is null"); Player = data.Player; - Game? group; + Game? game; Player? player; - if ((group = _gameService.FindGameByUsername(Player.Username)) != null && - (player = group.Players.Find(p => p.Username == Player.Username))?.State == State.Disconnected) + if ((game = _gameService.FindGameByUsername(Player.Username)) != null && + (player = game.Players.Find(p => p.Username == Player.Username))?.State == State.Disconnected) { - player.State = group.IsGameStarted ? State.InGame : State.WaitingForPlayers; + player.State = game.IsGameStarted ? State.InGame : State.WaitingForPlayers; // TODO doesn't work anymore Player = player; - Game = group; + Game = game; // TODO send missing data: Dices, CurrentPlayer, Ghosts } else @@ -75,6 +91,8 @@ public class ActionService : IActionService Game = _gameService.AddPlayer(Player, data.Spawns); } + Game.Connections += SendSegment; + return Game.Players; } @@ -100,18 +118,26 @@ public class ActionService : IActionService public string FindNextPlayer() => Game?.NextPlayer().Username ?? "Error: No group found"; - public void Disconnect() + public List LeaveGame() { - if (Player != null) Player.State = State.Disconnected; + if (Game == null || Player == null) throw new NullReferenceException("Game or Player is null"); + Game.RemovePlayer(Player.Username); + return Game.Players; } - public object? HandleMoveCharacter(JsonElement? jsonElement) + public void Disconnect() { - if (Game != null && jsonElement.HasValue) - Game.Ghosts = jsonElement.Value.GetProperty("ghosts").Deserialize>() ?? - throw new JsonException("Ghosts is null"); + if (Player == null) return; + Player.State = State.Disconnected; + if (Game != null) Game.Connections -= SendSegment; + } - return jsonElement; + public void SendToAll(ArraySegment segment) => Game?.SendToAll(segment); + + private async Task SendSegment(ArraySegment segment) + { + if (WebSocket != null) await _gameService.Send(WebSocket, segment); + else await Task.FromCanceled(new CancellationToken(true)); } } diff --git a/pac-man-board-game/Services/GameService.cs b/pac-man-board-game/Services/GameService.cs index c8296be..39b3e47 100644 --- a/pac-man-board-game/Services/GameService.cs +++ b/pac-man-board-game/Services/GameService.cs @@ -19,13 +19,6 @@ public class GameService : WebSocketService /// public SynchronizedCollection Games { get; } = new(); - public event Func, Task>? Connections; // TODO remove and use Game - - public void SendToAll(ArraySegment segment) - { - Connections?.Invoke(segment); - } - public Game AddPlayer(Player player, Queue spawns) { var index = 0; From cb9371b33b1e8e76325873c6762776a8ce66df12 Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Fri, 21 Jul 2023 13:54:44 +0200 Subject: [PATCH 4/5] Fixed bugs when disconnecting and reconnecting, disconnect message is sent before closing websocket connection --- .../src/components/gameComponent.tsx | 2 -- .../Controllers/GameController.cs | 12 +++++++---- .../Controllers/GenericController.cs | 21 +++++++++++-------- pac-man-board-game/GameStuff/Items/Player.cs | 4 ++-- pac-man-board-game/Services/ActionService.cs | 18 ++++++++++------ pac-man-board-game/Services/Game.cs | 15 ++++++++++++- 6 files changed, 48 insertions(+), 24 deletions(-) diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index ff37c8a..6039958 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -16,8 +16,6 @@ const wsService = new WebSocketService(import.meta.env.VITE_API_WS); // TODO bug, when taking player on last dice, the currentPlayer changes and the wrong character gets to steal // TODO bug, first player can sometimes roll dice twice -// TODO bug, when refreshing page, the player is still in disconnected state -// TODO bug, when refreshing page, the characters are reset for all players // TODO bug, when refreshing page, some data is missing until other clients make a move // TODO bug, teleportation doesn't work diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index 199192a..0bbd501 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -15,10 +15,10 @@ public class GameController : GenericController private readonly IActionService _actionService; private readonly GameService _gameService; - public GameController(ILogger logger, GameService gameService, IActionService actionService) : - base(logger, gameService) + public GameController(ILogger logger, GameService webSocketService, IActionService actionService) : + base(logger, webSocketService) { - _gameService = gameService; + _gameService = webSocketService; _actionService = actionService; } @@ -85,5 +85,9 @@ public class GameController : GenericController protected override void Send(ArraySegment segment) => _actionService.SendToAll(segment); - protected override void Disconnect() => _actionService.Disconnect(); + protected override ArraySegment? Disconnect() => + new ActionMessage { Action = GameAction.Disconnect, Data = _actionService.Disconnect() } + .ToArraySegment(); + + protected override void SendDisconnectMessage(ArraySegment segment) => _actionService.SendToAll(segment); } diff --git a/pac-man-board-game/Controllers/GenericController.cs b/pac-man-board-game/Controllers/GenericController.cs index 2c5f9e0..8e06533 100644 --- a/pac-man-board-game/Controllers/GenericController.cs +++ b/pac-man-board-game/Controllers/GenericController.cs @@ -7,14 +7,14 @@ namespace pacMan.Controllers; public abstract class GenericController : ControllerBase { private const int BufferSize = 1024 * 4; - protected readonly IWebSocketService GameService; + private readonly IWebSocketService _webSocketService; protected readonly ILogger Logger; protected WebSocket? WebSocket; - protected GenericController(ILogger logger, IWebSocketService gameService) + protected GenericController(ILogger logger, IWebSocketService webSocketService) { Logger = logger; - GameService = gameService; + _webSocketService = webSocketService; Logger.Log(LogLevel.Debug, "WebSocket Controller created"); } @@ -42,7 +42,7 @@ public abstract class GenericController : ControllerBase do { var buffer = new byte[BufferSize]; - result = await GameService.Receive(WebSocket, buffer); + result = await _webSocketService.Receive(WebSocket, buffer); if (result.CloseStatus.HasValue) break; @@ -51,23 +51,26 @@ public abstract class GenericController : ControllerBase Send(segment); } while (true); - await GameService.Close(WebSocket, result.CloseStatus.Value, result.CloseStatusDescription); + var disconnectSegment = Disconnect(); + if (disconnectSegment != null) SendDisconnectMessage((ArraySegment)disconnectSegment); + + await _webSocketService.Close(WebSocket, result.CloseStatus.Value, result.CloseStatusDescription); } catch (WebSocketException e) { Logger.Log(LogLevel.Error, "{}", e.Message); } - - Disconnect(); } protected virtual async void Send(ArraySegment segment) { if (WebSocket == null) return; - await GameService.Send(WebSocket, segment); + await _webSocketService.Send(WebSocket, segment); } protected abstract ArraySegment Run(WebSocketReceiveResult result, byte[] data); - protected virtual void Disconnect() { } + protected virtual ArraySegment? Disconnect() => null; + + protected virtual void SendDisconnectMessage(ArraySegment segment) { } } diff --git a/pac-man-board-game/GameStuff/Items/Player.cs b/pac-man-board-game/GameStuff/Items/Player.cs index 2139b80..640be81 100644 --- a/pac-man-board-game/GameStuff/Items/Player.cs +++ b/pac-man-board-game/GameStuff/Items/Player.cs @@ -15,11 +15,11 @@ public class Player : IEquatable { [JsonPropertyName("username")] public required string Username { get; init; } - [JsonPropertyName("pacMan")] public required Character PacMan { get; init; } + [JsonPropertyName("pacMan")] public required Character PacMan { get; set; } [JsonPropertyName("colour")] public required string Colour { get; init; } - [JsonPropertyName("box")] public Box? Box { get; init; } + [JsonPropertyName("box")] public Box? Box { get; set; } [JsonPropertyName("state")] public State State { get; set; } = State.WaitingForPlayers; diff --git a/pac-man-board-game/Services/ActionService.cs b/pac-man-board-game/Services/ActionService.cs index 54923d3..d8b586c 100644 --- a/pac-man-board-game/Services/ActionService.cs +++ b/pac-man-board-game/Services/ActionService.cs @@ -19,7 +19,7 @@ public interface IActionService string FindNextPlayer(); List LeaveGame(); void SendToAll(ArraySegment segment); - void Disconnect(); + List? Disconnect(); } public class ActionService : IActionService @@ -65,8 +65,12 @@ public class ActionService : IActionService public object? HandleMoveCharacter(JsonElement? jsonElement) { if (Game != null && jsonElement.HasValue) + { Game.Ghosts = jsonElement.Value.GetProperty("ghosts").Deserialize>() ?? throw new JsonException("Ghosts is null"); + Game.Players = jsonElement.Value.GetProperty("players").Deserialize>() ?? + throw new NullReferenceException("Players is null"); + } return jsonElement; } @@ -77,10 +81,11 @@ public class ActionService : IActionService Player = data.Player; Game? game; - Player? player; - if ((game = _gameService.FindGameByUsername(Player.Username)) != null && - (player = game.Players.Find(p => p.Username == Player.Username))?.State == State.Disconnected) + if ((game = _gameService.FindGameByUsername(Player.Username)) != null) { + var player = game.Players.Find(p => p.Username == Player.Username); + if (player is null) throw new NullReferenceException("Player is null"); + player.State = game.IsGameStarted ? State.InGame : State.WaitingForPlayers; // TODO doesn't work anymore Player = player; Game = game; @@ -125,11 +130,12 @@ public class ActionService : IActionService return Game.Players; } - public void Disconnect() + public List? Disconnect() { - if (Player == null) return; + if (Player == null) return null; Player.State = State.Disconnected; if (Game != null) Game.Connections -= SendSegment; + return Game?.Players; } public void SendToAll(ArraySegment segment) => Game?.SendToAll(segment); diff --git a/pac-man-board-game/Services/Game.cs b/pac-man-board-game/Services/Game.cs index 25eda66..9a9a49d 100644 --- a/pac-man-board-game/Services/Game.cs +++ b/pac-man-board-game/Services/Game.cs @@ -9,12 +9,25 @@ public class Game // TODO handle disconnects and reconnects { private readonly Random _random = new(); private int _currentPlayerIndex; + private List _players = new(); public Game(Queue spawns) => Spawns = spawns; [JsonInclude] public Guid Id { get; } = Guid.NewGuid(); - [JsonIgnore] public List Players { get; } = new(); + [JsonIgnore] + public List Players + { + get => _players; + set => + _players = _players.Select((player, index) => + { + player.State = value[index].State; + player.PacMan = value[index].PacMan; + player.Box = value[index].Box; + return player; + }).ToList(); + } [JsonIgnore] public List Ghosts { get; set; } = new(); From 71b24fedeee964961c1d460d1cc2214502a37dc5 Mon Sep 17 00:00:00 2001 From: Martin Berg Alstad <600878@stud.hvl.no> Date: Fri, 21 Jul 2023 16:49:56 +0200 Subject: [PATCH 5/5] Fixed typeError on vitest --- .../ClientApp/src/components/gameComponent.tsx | 1 + .../ClientApp/src/game/possibleMovesAlgorithm.ts | 16 ++++++++-------- pac-man-board-game/ClientApp/src/utils/api.ts | 1 + 3 files changed, 10 insertions(+), 8 deletions(-) diff --git a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx index 6039958..8b12e13 100644 --- a/pac-man-board-game/ClientApp/src/components/gameComponent.tsx +++ b/pac-man-board-game/ClientApp/src/components/gameComponent.tsx @@ -19,6 +19,7 @@ const wsService = new WebSocketService(import.meta.env.VITE_API_WS); // TODO bug, when refreshing page, some data is missing until other clients make a move // TODO bug, teleportation doesn't work +// TODO guest users // TODO store map in backend and save it in state on each client // TODO add debug menu on dev, for testing and cheating // TODO sign up player page diff --git a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts index 17cb44a..025402f 100644 --- a/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts +++ b/pac-man-board-game/ClientApp/src/game/possibleMovesAlgorithm.ts @@ -18,26 +18,26 @@ export default function findPossiblePositions(board: GameMap, character: Charact }; /** - * Uses recursion to move through the board and find all the possible positions - * @param board The board the character is on + * Uses recursion to move through the map and find all the possible positions + * @param map The map the character is on * @param currentPath The current path the character is on * @param steps The number of steps the character can move * @param character The current character * @param characters * @returns An array of paths the character can move to */ -function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number, character: Character, characters: Character[]): Path[] { +function findPossibleRecursive(map: GameMap, currentPath: Path, steps: number, character: Character, characters: Character[]): Path[] { const paths: Path[] = []; - if (isOutsideBoard(currentPath, board.length)) { // TODO not working on new map + if (isOutsideBoard(currentPath, map.length)) { // TODO not working on new map if (character.isPacMan()) { - return addTeleportationTiles(board, currentPath, steps, character, characters); + return addTeleportationTiles(map, currentPath, steps, character, characters); } - } else if (!isWall(board, currentPath)) { + } else if (!isWall(map, currentPath)) { if (!characterHitsAnotherCharacter(character, currentPath, characters)) { if (steps <= 0) { - if (!(isSpawn(board, currentPath) && !isOwnSpawn(currentPath, character))) { + if (!(isSpawn(map, currentPath) && !isOwnSpawn(currentPath, character))) { paths.push(currentPath); } @@ -47,7 +47,7 @@ function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number, steps--; for (const direction of getDirections()) { - paths.push(...tryMove(board, currentPath, direction, steps, character, characters)); + paths.push(...tryMove(map, currentPath, direction, steps, character, characters)); } } } else { diff --git a/pac-man-board-game/ClientApp/src/utils/api.ts b/pac-man-board-game/ClientApp/src/utils/api.ts index 32cfb36..b9ab872 100644 --- a/pac-man-board-game/ClientApp/src/utils/api.ts +++ b/pac-man-board-game/ClientApp/src/utils/api.ts @@ -1,4 +1,5 @@ export const getData: Api = async (path, {headers} = {}) => { + if (import.meta.env.MODE === "test") return Promise.resolve(new Response(JSON.stringify([]))); return await fetch(import.meta.env.VITE_API_HTTP + path, { method: "GET", headers: headers