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
#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
#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.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));
});
}
@ -174,7 +174,7 @@ public class WebSocketServiceTests
Assert.Multiple(() =>
{
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.First(), Has.Count.EqualTo(Rules.MaxPlayers));
});

View File

@ -1,6 +1,6 @@
import React, {MouseEventHandler} from "react";
import {State} from "../game/player";
import {currentPlayerAtom, thisPlayerAtom} from "../utils/state";
import {currentPlayerAtom, rollDiceButtonAtom, thisPlayerAtom} from "../utils/state";
import {useAtomValue} from "jotai";
interface GameButtonProps extends ComponentProps {
@ -13,16 +13,18 @@ const GameButton: Component<GameButtonProps> = (
onReadyClick,
onRollDiceClick,
}) => {
const currentPlayer = useAtomValue(currentPlayerAtom);
const thisPlayer = useAtomValue(thisPlayerAtom);
const activeRollDiceButton = useAtomValue(rollDiceButtonAtom);
if (currentPlayer === undefined || currentPlayer.State === State.waitingForPlayers) {
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 onClick={onRollDiceClick}>Roll dice</button>;
return <button onClick={onRollDiceClick} disabled={!activeRollDiceButton}>Roll dice</button>;
};
export default GameButton;

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@ -8,25 +8,37 @@ namespace pacMan.Services;
public class GameGroup : IEnumerable<IPlayer> // TODO handle disconnects and reconnects
{
private readonly Random _random = new();
private int _currentPlayerIndex;
public GameGroup(Queue<DirectionalPosition> spawns) => Spawns = spawns;
public List<IPlayer> Players { get; } = new();
private Queue<DirectionalPosition> Spawns { get; }
public IPlayer RandomPlayer => Players[_random.Next(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();
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
public void Shuffle() => Players.Sort((_, _) => _random.Next(-1, 2));
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;
if (Players.Exists(p => p.Name == player.Name)) return true;