Switch players after move, can only roll dice once, and other improvements

This commit is contained in:
Martin Berg Alstad 2023-07-16 12:10:53 +02:00
parent 8d8a606fb8
commit 373f08609d
12 changed files with 124 additions and 46 deletions

View File

@ -182,4 +182,20 @@ public class ActionServiceTests
} }
#endregion #endregion
#region FindNextPlayer()
[Test]
public void FindNexPlayer_OnePlayer()
{
Assert.Fail();
}
[Test]
public void FindNexPlayer_TwoPlayers()
{
Assert.Fail();
}
#endregion
} }

View File

@ -196,4 +196,23 @@ public class GameGroupTests
} }
#endregion #endregion
#region IsGameStarted()
[Test]
public void IsGameStarted_AllWaiting()
{
AddFullParty();
Assert.That(_gameGroup.IsGameStarted, Is.False);
}
[Test]
public void IsGameStarted_AllInGame()
{
AddFullParty();
_gameGroup.Players.ForEach(player => player.State = State.InGame);
Assert.That(_gameGroup.IsGameStarted, Is.True);
}
#endregion
} }

View File

@ -153,7 +153,7 @@ public class WebSocketServiceTests
Assert.Multiple(() => Assert.Multiple(() =>
{ {
Assert.That(group.Players, Has.Count.EqualTo(1)); Assert.That(group.Players, Has.Count.EqualTo(1));
Assert.That(group.RandomPlayer, Is.EqualTo(player)); Assert.That(group.NextPlayer, Is.EqualTo(player));
Assert.That(_service.Games, Has.Count.EqualTo(1)); Assert.That(_service.Games, Has.Count.EqualTo(1));
}); });
} }
@ -174,7 +174,7 @@ public class WebSocketServiceTests
Assert.Multiple(() => Assert.Multiple(() =>
{ {
Assert.That(group.Players, Has.Count.EqualTo(1)); Assert.That(group.Players, Has.Count.EqualTo(1));
Assert.That(group.RandomPlayer, Is.EqualTo(player5)); Assert.That(group.NextPlayer, Is.EqualTo(player5));
Assert.That(_service.Games, Has.Count.EqualTo(2)); Assert.That(_service.Games, Has.Count.EqualTo(2));
Assert.That(_service.Games.First(), Has.Count.EqualTo(Rules.MaxPlayers)); Assert.That(_service.Games.First(), Has.Count.EqualTo(Rules.MaxPlayers));
}); });

View File

@ -1,6 +1,6 @@
import React, {MouseEventHandler} from "react"; import React, {MouseEventHandler} from "react";
import {State} from "../game/player"; import {State} from "../game/player";
import {currentPlayerAtom, thisPlayerAtom} from "../utils/state"; import {currentPlayerAtom, rollDiceButtonAtom, thisPlayerAtom} from "../utils/state";
import {useAtomValue} from "jotai"; import {useAtomValue} from "jotai";
interface GameButtonProps extends ComponentProps { interface GameButtonProps extends ComponentProps {
@ -13,16 +13,18 @@ const GameButton: Component<GameButtonProps> = (
onReadyClick, onReadyClick,
onRollDiceClick, onRollDiceClick,
}) => { }) => {
const currentPlayer = useAtomValue(currentPlayerAtom); const currentPlayer = useAtomValue(currentPlayerAtom);
const thisPlayer = useAtomValue(thisPlayerAtom); const thisPlayer = useAtomValue(thisPlayerAtom);
const activeRollDiceButton = useAtomValue(rollDiceButtonAtom);
if (currentPlayer === undefined || currentPlayer.State === State.waitingForPlayers) { if (currentPlayer === undefined || currentPlayer.State === State.waitingForPlayers) {
return <button onClick={onReadyClick}>Ready</button>; return <button onClick={onReadyClick}>Ready</button>;
} }
if (!thisPlayer?.isTurn()) { if (!thisPlayer?.isTurn()) { // TODO also show when waiting for other players
return <button disabled>Please wait</button>; return <button disabled>Please wait</button>;
} }
return <button onClick={onRollDiceClick}>Roll dice</button>; return <button onClick={onRollDiceClick} disabled={!activeRollDiceButton}>Roll dice</button>;
}; };
export default GameButton; export default GameButton;

View File

@ -6,27 +6,32 @@ import WebSocketService from "../websockets/WebSocketService";
import {getCharacterSpawns, testMap} from "../game/map"; import {getCharacterSpawns, testMap} from "../game/map";
import Player from "../game/player"; import Player from "../game/player";
import PlayerStats from "../components/playerStats"; import PlayerStats from "../components/playerStats";
import {getDefaultStore, useAtom, useAtomValue} from "jotai"; import {getDefaultStore, useAtom, useAtomValue, useSetAtom} from "jotai";
import {diceAtom, ghostsAtom, playersAtom, selectedDiceAtom} from "../utils/state"; import {diceAtom, ghostsAtom, playersAtom, rollDiceButtonAtom, selectedDiceAtom} from "../utils/state";
import {CharacterType} from "../game/character"; import {CharacterType} from "../game/character";
import GameButton from "./gameButton"; import GameButton from "./gameButton";
const wsService = new WebSocketService(import.meta.env.VITE_API); const wsService = new WebSocketService(import.meta.env.VITE_API);
// TODO do not allow players to roll dice multiple times // TODO don't start game until at least 2 players have joined
// TODO fix tailwind colours from getBgCssColour // TODO join game lobby
// TODO steal from other players
// TODO show box with collected pellets
// TODO layout
export const GameComponent: Component<{ player: Player }> = ({player}) => { export const GameComponent: Component<{ player: Player }> = ({player}) => {
const players = useAtomValue(playersAtom); const players = useAtomValue(playersAtom);
const dice = useAtomValue(diceAtom); const dice = useAtomValue(diceAtom);
const [selectedDice, setSelectedDice] = useAtom(selectedDiceAtom); const [selectedDice, setSelectedDice] = useAtom(selectedDiceAtom);
const setActiveRollDiceButton = useSetAtom(rollDiceButtonAtom);
function rollDice(): void { function rollDice(): void {
if (!player.isTurn()) return; if (!player.isTurn()) return;
setSelectedDice(undefined); setSelectedDice(undefined);
wsService.send({Action: GameAction.rollDice}); wsService.send({Action: GameAction.rollDice});
setActiveRollDiceButton(false);
} }
function onCharacterMove(eatenPellets: Position[]): void { function onCharacterMove(eatenPellets: Position[]): void {
@ -44,9 +49,13 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => {
} }
}; };
wsService.send(data); wsService.send(data);
if (dice?.length === 0) {
endTurn();
}
} }
async function sendPlayer(): Promise<void> { function sendPlayer(): void {
wsService.send({ wsService.send({
Action: GameAction.playerInfo, Action: GameAction.playerInfo,
Data: { Data: {
@ -61,11 +70,15 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => {
wsService.send({Action: GameAction.ready}); wsService.send({Action: GameAction.ready});
} }
function endTurn() {
wsService.send({Action: GameAction.nextPlayer});
}
useEffect(() => { useEffect(() => {
wsService.onReceive = doAction; wsService.onReceive = doAction;
wsService.open(); wsService.open();
wsService.waitForOpen().then(() => void sendPlayer()); wsService.waitForOpen().then(() => sendPlayer());
return () => wsService.close(); return () => wsService.close();
}, []); }, []);

View File

@ -1,22 +1,27 @@
import React from "react"; import React from "react";
import Player, {State} from "../game/player"; import Player, {State} from "../game/player";
import {useAtomValue} from "jotai";
import {currentPlayerNameAtom} from "../utils/state";
const PlayerStats: Component<{ player: Player } & ComponentProps> = ( const PlayerStats: Component<{ player: Player } & ComponentProps> = (
{ {
player, player,
className, className,
id id
}) => ( }) => {
<div key={player.Colour} className={`w-fit m-2 ${className}`} id={id}> const currentPlayerName = useAtomValue(currentPlayerNameAtom);
<p className={player.isTurn() ? "underline" : ""}>Player: {player.Name}</p> return (
<p>Colour: {player.Colour}</p> <div key={player.Colour} className={`w-fit m-2 ${className}`} id={id}>
{player.State === State.inGame ? <p className={player.Name === currentPlayerName ? "underline" : ""}>Player: {player.Name}</p>
<> <p>Colour: {player.Colour}</p>
<p>Pellets: {player.Box.count}</p> {player.State === State.inGame ?
<p>PowerPellets: {player.Box.countPowerPellets}</p> <>
</> : <p>Pellets: {player.Box.count}</p>
<p>{player.State === State.waitingForPlayers ? "Waiting" : "Ready"}</p>} <p>PowerPellets: {player.Box.countPowerPellets}</p>
</div> </> :
); <p>{player.State === State.waitingForPlayers ? "Waiting" : "Ready"}</p>}
</div>
);
};
export default PlayerStats; export default PlayerStats;

View File

@ -2,7 +2,7 @@ import {Character, CharacterType} from "./character";
import Box from "./box"; import Box from "./box";
import {Colour} from "./colour"; import {Colour} from "./colour";
import {getDefaultStore} from "jotai"; import {getDefaultStore} from "jotai";
import {currentPlayerAtom} from "../utils/state"; import {currentPlayerNameAtom} from "../utils/state";
import Pellet from "./pellet"; import Pellet from "./pellet";
export enum State { export enum State {
@ -31,7 +31,7 @@ export default class Player {
public isTurn(): boolean { public isTurn(): boolean {
const store = getDefaultStore(); const store = getDefaultStore();
return store.get(currentPlayerAtom)?.Name === this.Name; return store.get(currentPlayerNameAtom) === this.Name;
} }
public addPellet(pellet: Pellet): void { public addPellet(pellet: Pellet): void {

View File

@ -3,7 +3,7 @@ import {CharacterType, Ghost} from "../game/character";
import {getCharacterSpawns, testMap} from "../game/map"; import {getCharacterSpawns, testMap} from "../game/map";
import {TileType} from "../game/tileType"; import {TileType} from "../game/tileType";
import {getDefaultStore} from "jotai"; import {getDefaultStore} from "jotai";
import {currentPlayerNameAtom, diceAtom, ghostsAtom, playersAtom} from "./state"; import {currentPlayerNameAtom, diceAtom, ghostsAtom, playersAtom, rollDiceButtonAtom} from "./state";
import {Colour} from "../game/colour"; import {Colour} from "../game/colour";
export enum GameAction { export enum GameAction {
@ -11,6 +11,7 @@ export enum GameAction {
moveCharacter, moveCharacter,
playerInfo, playerInfo,
ready, ready,
nextPlayer,
} }
const store = getDefaultStore(); const store = getDefaultStore();
@ -45,6 +46,9 @@ export const doAction: MessageEventFunction<string> = (event): void => { // TODO
case GameAction.ready: case GameAction.ready:
ready(message.Data); ready(message.Data);
break; break;
case GameAction.nextPlayer:
nextPlayer(message.Data);
break;
} }
}; };
@ -93,17 +97,21 @@ function playerInfo(data?: PlayerProps[]): void {
store.set(playersAtom, playerProps.map(p => new Player(p))); store.set(playersAtom, playerProps.map(p => new Player(p)));
} }
type ReadyData = type ReadyData = { AllReady: boolean, Players: PlayerProps[] } | string;
| { AllReady: true, Starter: string, Players: PlayerProps[] }
| { AllReady: false, Players: PlayerProps[] }
| string;
function ready(data?: ReadyData): void { function ready(data?: ReadyData): void {
if (data && typeof data !== "string") { if (data && typeof data !== "string") {
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.Starter); store.set(currentPlayerNameAtom, data.Players[0].Name);
} }
} else {
console.error("Error:", data);
} }
} }
function nextPlayer(currentPlayerName?: string): void {
store.set(currentPlayerNameAtom, currentPlayerName);
store.set(rollDiceButtonAtom, true);
}

View File

@ -49,3 +49,7 @@ 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.Name === currentPlayerName);
}); });
/**
* Whether the roll dice button should be enabled.
*/
export const rollDiceButtonAtom = atom(true);

View File

@ -7,7 +7,8 @@ public enum GameAction
RollDice, RollDice,
MoveCharacter, MoveCharacter,
PlayerInfo, PlayerInfo,
Ready Ready,
NextPlayer
} }
public class ActionMessage<T> public class ActionMessage<T>

View File

@ -40,6 +40,7 @@ public class ActionService : IActionService
GameAction.RollDice => RollDice(), GameAction.RollDice => RollDice(),
GameAction.PlayerInfo => SetPlayerInfo(message), GameAction.PlayerInfo => SetPlayerInfo(message),
GameAction.Ready => Ready(), GameAction.Ready => Ready(),
GameAction.NextPlayer => FindNextPlayer(),
_ => message.Data _ => message.Data
}; };
} }
@ -79,16 +80,11 @@ public class ActionService : IActionService
if (Player != null && Group != null) if (Player != null && Group != null)
{ {
var players = Group.SetReady(Player).ToArray(); var players = Group.SetReady(Player).ToArray();
if (players.All(p => p.State == State.Ready)) // TODO roll to start
{ Group.Shuffle();
// TODO roll to start var allReady = players.All(p => p.State == State.Ready);
Group.SetAllInGame(); if (allReady) Group.SetAllInGame();
data = new { AllReady = true, Players = players, Starter = Group.RandomPlayer.Name }; data = new { AllReady = allReady, Players = players };
}
else
{
data = new { AllReady = false, Players = players };
}
} }
else else
{ {
@ -97,6 +93,8 @@ public class ActionService : IActionService
return data; return data;
} }
public string FindNextPlayer() => Group?.NextPlayer.Name ?? "Error: No group found";
} }
public class PlayerInfoData public class PlayerInfoData

View File

@ -8,25 +8,37 @@ namespace pacMan.Services;
public class GameGroup : IEnumerable<IPlayer> // TODO handle disconnects and reconnects public class GameGroup : IEnumerable<IPlayer> // TODO handle disconnects and reconnects
{ {
private readonly Random _random = new(); private readonly Random _random = new();
private int _currentPlayerIndex;
public GameGroup(Queue<DirectionalPosition> spawns) => Spawns = spawns; public GameGroup(Queue<DirectionalPosition> spawns) => Spawns = spawns;
public List<IPlayer> Players { get; } = new(); public List<IPlayer> Players { get; } = new();
private Queue<DirectionalPosition> Spawns { get; } private Queue<DirectionalPosition> Spawns { get; }
public IPlayer RandomPlayer => Players[_random.Next(Count)];
public int Count => Players.Count; public int Count => Players.Count;
public IPlayer NextPlayer
{
get
{
_currentPlayerIndex = (_currentPlayerIndex + 1) % Count;
return Players[_currentPlayerIndex];
}
}
public bool IsGameStarted => Count > 0 && Players.All(player => player.State is State.InGame or State.Disconnected);
public IEnumerator<IPlayer> GetEnumerator() => Players.GetEnumerator(); public IEnumerator<IPlayer> GetEnumerator() => Players.GetEnumerator();
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
public void Shuffle() => Players.Sort((_, _) => _random.Next(-1, 2));
public event Func<ArraySegment<byte>, Task>? Connections; public event Func<ArraySegment<byte>, Task>? Connections;
public bool AddPlayer(IPlayer player) // TODO if name exists, use that player instead public bool AddPlayer(IPlayer player)
{ {
if (Players.Count >= Rules.MaxPlayers) return false; if (Players.Count >= Rules.MaxPlayers || IsGameStarted) return false;
player.State = State.WaitingForPlayers; player.State = State.WaitingForPlayers;
if (Players.Exists(p => p.Name == player.Name)) return true; if (Players.Exists(p => p.Name == player.Name)) return true;