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 (
<>
+
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