Switch players after move, can only roll dice once, and other improvements
This commit is contained in:
parent
8d8a606fb8
commit
373f08609d
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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));
|
||||||
});
|
});
|
||||||
|
@ -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;
|
||||||
|
@ -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();
|
||||||
}, []);
|
}, []);
|
||||||
|
@ -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;
|
||||||
|
@ -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 {
|
||||||
|
@ -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);
|
||||||
|
}
|
||||||
|
@ -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);
|
||||||
|
@ -7,7 +7,8 @@ public enum GameAction
|
|||||||
RollDice,
|
RollDice,
|
||||||
MoveCharacter,
|
MoveCharacter,
|
||||||
PlayerInfo,
|
PlayerInfo,
|
||||||
Ready
|
Ready,
|
||||||
|
NextPlayer
|
||||||
}
|
}
|
||||||
|
|
||||||
public class ActionMessage<T>
|
public class ActionMessage<T>
|
||||||
|
@ -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
|
||||||
|
@ -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;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user