Players can rejoin when refreshing window, changed Name to UserName
This commit is contained in:
parent
01757f825e
commit
767189821d
@ -165,7 +165,7 @@ public class ActionServiceTests
|
|||||||
// If selected the state is changed to InGame
|
// If selected the state is changed to InGame
|
||||||
_whitePlayer.State = State.InGame;
|
_whitePlayer.State = State.InGame;
|
||||||
var players = result.GetType().GetProperty("Players")?.GetValue(result) as IEnumerable<IPlayer>;
|
var players = result.GetType().GetProperty("Players")?.GetValue(result) as IEnumerable<IPlayer>;
|
||||||
Assert.That(players?.First().Name, Is.EqualTo(_whitePlayer.Name));
|
Assert.That(players?.First().UserName, Is.EqualTo(_whitePlayer.UserName));
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
@ -185,7 +185,7 @@ public class ActionServiceTests
|
|||||||
result = _service.Ready();
|
result = _service.Ready();
|
||||||
|
|
||||||
var players = result.GetType().GetProperty("Players")?.GetValue(result) as IEnumerable<IPlayer>;
|
var players = result.GetType().GetProperty("Players")?.GetValue(result) as IEnumerable<IPlayer>;
|
||||||
Assert.That(players?.First().Name, Is.EqualTo(_blackPlayer.Name).Or.EqualTo(_whitePlayer.Name));
|
Assert.That(players?.First().UserName, Is.EqualTo(_blackPlayer.UserName).Or.EqualTo(_whitePlayer.UserName));
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@ -208,7 +208,7 @@ public class ActionServiceTests
|
|||||||
{ Players = { _whitePlayer } };
|
{ Players = { _whitePlayer } };
|
||||||
|
|
||||||
var name = _service.FindNextPlayer();
|
var name = _service.FindNextPlayer();
|
||||||
Assert.That(name, Is.EqualTo(_whitePlayer.Name));
|
Assert.That(name, Is.EqualTo(_whitePlayer.UserName));
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
@ -222,9 +222,9 @@ public class ActionServiceTests
|
|||||||
})) { Players = { _whitePlayer, _blackPlayer } };
|
})) { Players = { _whitePlayer, _blackPlayer } };
|
||||||
|
|
||||||
var first = _service.FindNextPlayer();
|
var first = _service.FindNextPlayer();
|
||||||
Assert.That(first, Is.EqualTo(_blackPlayer.Name));
|
Assert.That(first, Is.EqualTo(_blackPlayer.UserName));
|
||||||
var second = _service.FindNextPlayer();
|
var second = _service.FindNextPlayer();
|
||||||
Assert.That(second, Is.EqualTo(_whitePlayer.Name));
|
Assert.That(second, Is.EqualTo(_whitePlayer.UserName));
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
@ -8,7 +8,7 @@ internal static class Players
|
|||||||
internal static IPlayer Create(string colour) =>
|
internal static IPlayer Create(string colour) =>
|
||||||
new Player
|
new Player
|
||||||
{
|
{
|
||||||
Name = colour,
|
UserName = colour,
|
||||||
Colour = colour,
|
Colour = colour,
|
||||||
PacMan = CreatePacMan(colour),
|
PacMan = CreatePacMan(colour),
|
||||||
Box = CreateBox(colour)
|
Box = CreateBox(colour)
|
||||||
@ -33,7 +33,7 @@ internal static class Players
|
|||||||
{
|
{
|
||||||
Box = player.Box,
|
Box = player.Box,
|
||||||
Colour = player.Colour,
|
Colour = player.Colour,
|
||||||
Name = player.Name,
|
UserName = player.UserName,
|
||||||
PacMan = player.PacMan
|
PacMan = player.PacMan
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -181,8 +181,8 @@ const SelectPlayerModal: Component = () => {
|
|||||||
|
|
||||||
{
|
{
|
||||||
allPlayers.map(player =>
|
allPlayers.map(player =>
|
||||||
<div key={player.Name} className={"border-b pb-1"}>
|
<div key={player.UserName} className={"border-b pb-1"}>
|
||||||
<span className={"mx-2"}>{player.Name} has {player.Box.count} pellets</span>
|
<span className={"mx-2"}>{player.UserName} has {player.Box.count} pellets</span>
|
||||||
<button className={"text-blue-500 enabled:cursor-pointer disabled:text-gray-500"}
|
<button className={"text-blue-500 enabled:cursor-pointer disabled:text-gray-500"}
|
||||||
style={{background: "none"}}
|
style={{background: "none"}}
|
||||||
disabled={player.Box.count === 0}
|
disabled={player.Box.count === 0}
|
||||||
|
@ -15,9 +15,11 @@ 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, 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, first player can sometimes roll dice twice (maybe only on firefox)
|
||||||
|
// 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 add debug menu on dev, for testing and cheating
|
// TODO add debug menu on dev, for testing and cheating
|
||||||
// TODO join game lobby
|
// TODO join/new game lobby
|
||||||
// TODO sign up player page
|
// TODO sign up player page
|
||||||
// TODO sign in page
|
// TODO sign in page
|
||||||
// TODO show box with collected pellets
|
// TODO show box with collected pellets
|
||||||
@ -91,7 +93,7 @@ export const GameComponent: Component<{ player: Player, map: GameMap }> = ({play
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={"flex justify-center"}>
|
<div className={"flex justify-center"}>
|
||||||
{players?.map(p => <PlayerStats key={p.Name} player={p}/>)}
|
{players?.map(p => <PlayerStats key={p.UserName} player={p}/>)}
|
||||||
</div>
|
</div>
|
||||||
<div className={"flex-center"}>
|
<div className={"flex-center"}>
|
||||||
<GameButton onReadyClick={sendReady} onRollDiceClick={rollDice}/>
|
<GameButton onReadyClick={sendReady} onRollDiceClick={rollDice}/>
|
||||||
|
@ -11,10 +11,11 @@ const PlayerStats: Component<{ player: Player } & ComponentProps> = (
|
|||||||
}) => {
|
}) => {
|
||||||
const currentPlayerName = useAtomValue(currentPlayerNameAtom);
|
const currentPlayerName = useAtomValue(currentPlayerNameAtom);
|
||||||
return (
|
return (
|
||||||
<div key={player.Colour} className={`w-fit m-2 ${className}`} id={id}>
|
<div key={player.Colour}
|
||||||
<p className={player.Name === currentPlayerName ? "underline" : ""}>Player: {player.Name}</p>
|
className={`w-fit m-2 ${player.State === State.disconnected ? "text-gray-500" : ""} ${className}`} id={id}>
|
||||||
|
<p className={player.UserName === currentPlayerName ? "underline" : ""}>Player: {player.UserName}</p>
|
||||||
<p>Colour: {player.Colour}</p>
|
<p>Colour: {player.Colour}</p>
|
||||||
{player.State === State.inGame ?
|
{player.State === State.inGame || player.State === State.disconnected ?
|
||||||
<>
|
<>
|
||||||
<p>Pellets: {player.Box.count}</p>
|
<p>Pellets: {player.Box.count}</p>
|
||||||
<p>PowerPellets: {player.Box.countPowerPellets}</p>
|
<p>PowerPellets: {player.Box.countPowerPellets}</p>
|
||||||
|
@ -9,18 +9,19 @@ import rules from "./rules";
|
|||||||
export enum State {
|
export enum State {
|
||||||
waitingForPlayers,
|
waitingForPlayers,
|
||||||
ready,
|
ready,
|
||||||
inGame
|
inGame,
|
||||||
|
disconnected
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class Player {
|
export default class Player {
|
||||||
public readonly Name: string;
|
public readonly UserName: string;
|
||||||
public readonly PacMan: Character;
|
public readonly PacMan: Character;
|
||||||
public readonly Colour: Colour;
|
public readonly Colour: Colour;
|
||||||
public readonly Box: Box;
|
public readonly Box: Box;
|
||||||
public State: State;
|
public State: State;
|
||||||
|
|
||||||
constructor(props: PlayerProps) {
|
constructor(props: PlayerProps) {
|
||||||
this.Name = props.Name;
|
this.UserName = props.UserName;
|
||||||
this.Colour = props.Colour;
|
this.Colour = props.Colour;
|
||||||
this.Box = new Box(props.Box ?? {Colour: props.Colour});
|
this.Box = new Box(props.Box ?? {Colour: props.Colour});
|
||||||
this.PacMan = new Character(props.PacMan ?? {
|
this.PacMan = new Character(props.PacMan ?? {
|
||||||
@ -32,7 +33,7 @@ export default class Player {
|
|||||||
|
|
||||||
public isTurn(): boolean {
|
public isTurn(): boolean {
|
||||||
const store = getDefaultStore();
|
const store = getDefaultStore();
|
||||||
return store.get(currentPlayerNameAtom) === this.Name;
|
return store.get(currentPlayerNameAtom) === this.UserName;
|
||||||
}
|
}
|
||||||
|
|
||||||
public addPellet(pellet: Pellet): void {
|
public addPellet(pellet: Pellet): void {
|
||||||
|
@ -17,7 +17,7 @@ const Home: Component = () => {
|
|||||||
function formHandler(): void {
|
function formHandler(): void {
|
||||||
if (!input.current || !dropdown.current) return;
|
if (!input.current || !dropdown.current) return;
|
||||||
const player = new Player({
|
const player = new Player({
|
||||||
Name: input.current.value,
|
UserName: input.current.value,
|
||||||
Colour: dropdown.current.value as Colour,
|
Colour: dropdown.current.value as Colour,
|
||||||
});
|
});
|
||||||
setPlayer(player);
|
setPlayer(player);
|
||||||
|
@ -39,7 +39,7 @@ interface BoxProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
interface PlayerProps {
|
interface PlayerProps {
|
||||||
readonly Name: string,
|
readonly UserName: string,
|
||||||
readonly PacMan?: CharacterProps,
|
readonly PacMan?: CharacterProps,
|
||||||
readonly Colour: import("../game/colour").Colour,
|
readonly Colour: import("../game/colour").Colour,
|
||||||
readonly Box?: BoxProps,
|
readonly Box?: BoxProps,
|
||||||
|
@ -91,7 +91,7 @@ function removeEatenPellets(data?: MoveCharacterData): void {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function playerInfo(data?: PlayerProps[]): void {
|
function playerInfo(data?: PlayerProps[]): void { // TODO missing data when refreshing page
|
||||||
const playerProps = data ?? [];
|
const playerProps = data ?? [];
|
||||||
spawns = getCharacterSpawns(testMap).filter(spawn => spawn.type === CharacterType.pacMan);
|
spawns = getCharacterSpawns(testMap).filter(spawn => spawn.type === CharacterType.pacMan);
|
||||||
store.set(playersAtom, playerProps.map(p => new Player(p)));
|
store.set(playersAtom, playerProps.map(p => new Player(p)));
|
||||||
@ -104,7 +104,7 @@ function ready(data?: ReadyData): void {
|
|||||||
const players = data.Players.map(p => new Player(p));
|
const players = data.Players.map(p => new Player(p));
|
||||||
store.set(playersAtom, players);
|
store.set(playersAtom, players);
|
||||||
if (data.AllReady) {
|
if (data.AllReady) {
|
||||||
store.set(currentPlayerNameAtom, data.Players[0].Name);
|
store.set(currentPlayerNameAtom, data.Players[0].UserName);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
console.error("Error:", data);
|
console.error("Error:", data);
|
||||||
|
@ -47,7 +47,7 @@ export const currentPlayerNameAtom = atom<string | undefined>(undefined);
|
|||||||
*/
|
*/
|
||||||
export const currentPlayerAtom = atom<Player | undefined>(get => {
|
export const currentPlayerAtom = atom<Player | undefined>(get => {
|
||||||
const currentPlayerName = get(currentPlayerNameAtom);
|
const currentPlayerName = get(currentPlayerNameAtom);
|
||||||
return get(playersAtom).find(player => player.Name === currentPlayerName);
|
return get(playersAtom).find(player => player.UserName === currentPlayerName);
|
||||||
});
|
});
|
||||||
/**
|
/**
|
||||||
* Whether the roll dice button should be enabled.
|
* Whether the roll dice button should be enabled.
|
||||||
|
@ -2,7 +2,7 @@ namespace pacMan.GameStuff.Items;
|
|||||||
|
|
||||||
public interface IPlayer
|
public interface IPlayer
|
||||||
{
|
{
|
||||||
string Name { get; init; }
|
string UserName { get; init; }
|
||||||
Character PacMan { get; init; }
|
Character PacMan { get; init; }
|
||||||
string Colour { get; init; }
|
string Colour { get; init; }
|
||||||
Box? Box { get; init; }
|
Box? Box { get; init; }
|
||||||
@ -23,11 +23,10 @@ public class Player : IPlayer, IEquatable<Player>
|
|||||||
{
|
{
|
||||||
if (ReferenceEquals(null, other)) return false;
|
if (ReferenceEquals(null, other)) return false;
|
||||||
if (ReferenceEquals(this, other)) return true;
|
if (ReferenceEquals(this, other)) return true;
|
||||||
return Name == other.Name && PacMan.Equals(other.PacMan) && Colour == other.Colour && Box.Equals(other.Box) &&
|
return UserName == other.UserName;
|
||||||
State == other.State;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public required string Name { get; init; }
|
public required string UserName { get; init; }
|
||||||
public required Character PacMan { get; init; }
|
public required Character PacMan { get; init; }
|
||||||
public required string Colour { get; init; }
|
public required string Colour { get; init; }
|
||||||
public Box? Box { get; init; }
|
public Box? Box { get; init; }
|
||||||
@ -40,5 +39,5 @@ public class Player : IPlayer, IEquatable<Player>
|
|||||||
return obj.GetType() == GetType() && Equals((Player)obj);
|
return obj.GetType() == GetType() && Equals((Player)obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override int GetHashCode() => HashCode.Combine(Name, PacMan, Colour, Box, (int)State);
|
public override int GetHashCode() => UserName.GetHashCode();
|
||||||
}
|
}
|
||||||
|
@ -62,7 +62,20 @@ public class ActionService : IActionService
|
|||||||
PlayerInfoData data = JsonSerializer.Deserialize<PlayerInfoData>(message.Data);
|
PlayerInfoData data = JsonSerializer.Deserialize<PlayerInfoData>(message.Data);
|
||||||
Player = data.Player;
|
Player = data.Player;
|
||||||
|
|
||||||
Group = _gameService.AddPlayer(Player, data.Spawns);
|
Game? group;
|
||||||
|
IPlayer? player;
|
||||||
|
if ((group = _gameService.FindGameByUsername(Player.UserName)) != null &&
|
||||||
|
(player = group.Players.Find(p => p.UserName == Player.UserName))?.State == State.Disconnected)
|
||||||
|
{
|
||||||
|
player.State = group.IsGameStarted ? State.InGame : State.WaitingForPlayers;
|
||||||
|
Player = player;
|
||||||
|
Group = group;
|
||||||
|
// TODO send missing data: Dices, CurrentPlayer, Ghosts
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Group = _gameService.AddPlayer(Player, data.Spawns);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (RuntimeBinderException e)
|
catch (RuntimeBinderException e)
|
||||||
{
|
{
|
||||||
@ -95,7 +108,7 @@ public class ActionService : IActionService
|
|||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
public string FindNextPlayer() => Group?.NextPlayer().Name ?? "Error: No group found";
|
public string FindNextPlayer() => Group?.NextPlayer().UserName ?? "Error: No group found";
|
||||||
|
|
||||||
public void Disconnect()
|
public void Disconnect()
|
||||||
{
|
{
|
||||||
|
@ -52,7 +52,7 @@ public class Game // TODO handle disconnects and reconnects
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
player.State = State.WaitingForPlayers;
|
player.State = State.WaitingForPlayers;
|
||||||
if (Players.Exists(p => p.Name == player.Name)) return true; // TODO change to false
|
if (Players.Exists(p => p.UserName == player.UserName)) return true; // TODO change to false
|
||||||
Players.Add(player);
|
Players.Add(player);
|
||||||
if (player.PacMan.SpawnPosition is null) SetSpawn(player);
|
if (player.PacMan.SpawnPosition is null) SetSpawn(player);
|
||||||
return true;
|
return true;
|
||||||
@ -70,7 +70,7 @@ public class Game // TODO handle disconnects and reconnects
|
|||||||
|
|
||||||
public IEnumerable<IPlayer> SetReady(IPlayer player)
|
public IEnumerable<IPlayer> SetReady(IPlayer player)
|
||||||
{
|
{
|
||||||
if (!Players.Contains(player)) // TODO throws exception after game has started and refresh
|
if (!Players.Contains(player))
|
||||||
throw new PlayerNotFoundException("The player was not found in the game group.");
|
throw new PlayerNotFoundException("The player was not found in the game group.");
|
||||||
player.State = State.Ready;
|
player.State = State.Ready;
|
||||||
return Players;
|
return Players;
|
||||||
|
@ -5,15 +5,17 @@ using pacMan.GameStuff.Items;
|
|||||||
namespace pacMan.Services;
|
namespace pacMan.Services;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The GameService class provides functionality for managing games in a WebSocket environment. It inherits from the WebSocketService class.
|
/// The GameService class provides functionality for managing games in a WebSocket environment. It inherits from the
|
||||||
|
/// WebSocketService class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class GameService : WebSocketService
|
public class GameService : WebSocketService
|
||||||
{
|
{
|
||||||
public GameService(ILogger<GameService> logger) : base(logger) { }
|
public GameService(ILogger<GameService> logger) : base(logger) { }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// A thread-safe collection (SynchronizedCollection) of "Game" objects. Utilized for managing multiple game instances simultaneously.
|
/// A thread-safe collection (SynchronizedCollection) of "Game" objects. Utilized for managing multiple game instances
|
||||||
/// It represents all the current games being managed by GameService.
|
/// simultaneously.
|
||||||
|
/// It represents all the current games being managed by GameService.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public SynchronizedCollection<Game> Games { get; } = new();
|
public SynchronizedCollection<Game> Games { get; } = new();
|
||||||
|
|
||||||
@ -42,7 +44,7 @@ public class GameService : WebSocketService
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// This method tries to find a game with the specified id, add a player to it and return the updated game.
|
/// This method tries to find a game with the specified id, add a player to it and return the updated game.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="id">The unique id of the game the player wants to join</param>
|
/// <param name="id">The unique id of the game the player wants to join</param>
|
||||||
/// <param name="player">The player instance that wants to join the game</param>
|
/// <param name="player">The player instance that wants to join the game</param>
|
||||||
@ -58,12 +60,15 @@ public class GameService : WebSocketService
|
|||||||
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates a new game and adds a player to it.
|
/// Creates a new game and adds a player to it.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="player">The player instance that is going to join the new game.</param>
|
/// <param name="player">The player instance that is going to join the new game.</param>
|
||||||
/// <param name="spawns">A collection of spawn points arranged in a directional queue.</param>
|
/// <param name="spawns">A collection of spawn points arranged in a directional queue.</param>
|
||||||
/// <returns>Returns the newly created Game object, with the player added to it.</returns>
|
/// <returns>Returns the newly created Game object, with the player added to it.</returns>
|
||||||
/// <exception cref="ArgumentException">Thrown if the number of spawns is not equal to the maximum number of players set by the Rules.</exception>
|
/// <exception cref="ArgumentException">
|
||||||
|
/// Thrown if the number of spawns is not equal to the maximum number of players set by
|
||||||
|
/// the Rules.
|
||||||
|
/// </exception>
|
||||||
public Game CreateAndJoin(IPlayer player, Queue<DirectionalPosition> spawns)
|
public Game CreateAndJoin(IPlayer player, Queue<DirectionalPosition> spawns)
|
||||||
{
|
{
|
||||||
if (spawns.Count != Rules.MaxPlayers)
|
if (spawns.Count != Rules.MaxPlayers)
|
||||||
@ -75,4 +80,9 @@ public class GameService : WebSocketService
|
|||||||
|
|
||||||
return game;
|
return game;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Game? FindGameByUsername(string username)
|
||||||
|
{
|
||||||
|
return Games.FirstOrDefault(game => game.Players.Exists(player => player.UserName == username));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user