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();