Json is being handled with camelCase

This commit is contained in:
Martin Berg Alstad 2023-07-20 18:06:30 +02:00
parent 0c9ba333ea
commit 1db515d796
33 changed files with 409 additions and 375 deletions

View File

@ -10,7 +10,7 @@ namespace BackendTests.Services;
public class ActionServiceTests public class ActionServiceTests
{ {
private readonly Player _blackPlayer = (Player)Players.Create("black"); private readonly Player _blackPlayer = Players.Create("black");
private readonly Player _redPlayer = (Player)Players.Create("red"); private readonly Player _redPlayer = (Player)Players.Create("red");
private readonly Player _whitePlayer = (Player)Players.Create("white"); private readonly Player _whitePlayer = (Player)Players.Create("white");
@ -109,7 +109,7 @@ public class ActionServiceTests
var pos = _spawns.Dequeue(); var pos = _spawns.Dequeue();
_whitePlayer.PacMan.Position = pos; _whitePlayer.PacMan.Position = pos;
_whitePlayer.PacMan.SpawnPosition = pos; _whitePlayer.PacMan.SpawnPosition = pos;
Assert.That(new List<IPlayer> { _whitePlayer }, Is.EqualTo(players)); Assert.That(new List<Player> { _whitePlayer }, Is.EqualTo(players));
} }
#endregion #endregion
@ -173,7 +173,7 @@ public class ActionServiceTests
var result = _service.Ready(); var result = _service.Ready();
// 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<Player>;
Assert.That(players?.First().Username, Is.EqualTo(_whitePlayer.Username)); Assert.That(players?.First().Username, Is.EqualTo(_whitePlayer.Username));
} }
@ -193,7 +193,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<Player>;
Assert.That(players?.First().Username, Is.EqualTo(_blackPlayer.Username).Or.EqualTo(_whitePlayer.Username)); Assert.That(players?.First().Username, Is.EqualTo(_blackPlayer.Username).Or.EqualTo(_whitePlayer.Username));
} }

View File

@ -20,14 +20,14 @@ public class GameTests
private readonly DirectionalPosition _spawn7By7Right = new() private readonly DirectionalPosition _spawn7By7Right = new()
{ At = new Position { X = 7, Y = 7 }, Direction = Direction.Right }; { At = new Position { X = 7, Y = 7 }, Direction = Direction.Right };
private IPlayer _bluePlayer = null!; private Player _bluePlayer = null!;
private pacMan.Services.Game _game = null!; private pacMan.Services.Game _game = null!;
private IPlayer _greenPlayer = null!; private Player _greenPlayer = null!;
private IPlayer _purplePlayer = null!; private Player _purplePlayer = null!;
private IPlayer _redPlayer = null!; private Player _redPlayer = null!;
private Queue<DirectionalPosition> _spawns = null!; private Queue<DirectionalPosition> _spawns = null!;
private IPlayer _yellowPlayer = null!; private Player _yellowPlayer = null!;
[SetUp] [SetUp]
public void Setup() public void Setup()
@ -61,7 +61,7 @@ public class GameTests
#endregion #endregion
#region AddPlayer(IPlayer player) #region AddPlayer(Player player)
[Test] [Test]
public void AddPlayer_WhenEmpty() public void AddPlayer_WhenEmpty()
@ -82,7 +82,7 @@ public class GameTests
[Test] [Test]
public void AddPlayer_WhenNameExists() public void AddPlayer_WhenNameExists()
{ {
var redClone = Players.Clone(_redPlayer); var redClone = _redPlayer.Clone();
_game.AddPlayer(_redPlayer); _game.AddPlayer(_redPlayer);
var added = _game.AddPlayer(redClone); var added = _game.AddPlayer(redClone);
Assert.That(added, Is.True); Assert.That(added, Is.True);
@ -146,7 +146,7 @@ public class GameTests
#endregion #endregion
#region SetReady(IPlayer player) #region SetReady(Player player)
[Test] [Test]
public void SetReady_ReturnsAllPlayers() public void SetReady_ReturnsAllPlayers()
@ -191,14 +191,14 @@ public class GameTests
{ {
AddFullParty(); AddFullParty();
_game.Players.ForEach(player => player.State = State.Ready); _game.Players.ForEach(player => player.State = State.Ready);
Assert.That(_game.Players, Has.All.Property(nameof(IPlayer.State)).EqualTo(State.Ready)); Assert.That(_game.Players, Has.All.Property(nameof(Player.State)).EqualTo(State.Ready));
var allInGame = _game.SetAllInGame(); var allInGame = _game.SetAllInGame();
Assert.Multiple(() => Assert.Multiple(() =>
{ {
Assert.That(allInGame, Is.True); Assert.That(allInGame, Is.True);
Assert.That(_game.Players, Has.All.Property(nameof(IPlayer.State)).EqualTo(State.InGame)); Assert.That(_game.Players, Has.All.Property(nameof(Player.State)).EqualTo(State.InGame));
}); });
} }

View File

@ -5,8 +5,8 @@ namespace BackendTests.TestUtils;
internal static class Players internal static class Players
{ {
internal static IPlayer Create(string colour) => internal static Player Create(string colour) =>
new Player new()
{ {
Username = colour, Username = colour,
Colour = colour, Colour = colour,
@ -28,8 +28,8 @@ internal static class Players
Pellets = new List<Pellet>() Pellets = new List<Pellet>()
}; };
internal static IPlayer Clone(IPlayer player) => internal static Player Clone(this Player player) =>
new Player new()
{ {
Box = player.Box, Box = player.Box,
Colour = player.Colour, Colour = player.Colour,

View File

@ -31,7 +31,7 @@ const Board: FC<BoardProps> = (
const setModalOpen = useSetAtom(modalOpenAtom); const setModalOpen = useSetAtom(modalOpenAtom);
function handleSelectCharacter(character: Character): void { function handleSelectCharacter(character: Character): void {
if (character.isPacMan() && currentPlayer?.PacMan.Colour !== character.Colour) { if (character.isPacMan() && currentPlayer?.pacMan.colour !== character.colour) {
return; return;
} }
setSelectedCharacter(character); setSelectedCharacter(character);
@ -58,7 +58,7 @@ const Board: FC<BoardProps> = (
} }
function tryMovePacManToSpawn(destination: Path): void { function tryMovePacManToSpawn(destination: Path): void {
const takenChar = characters.find(c => c.isPacMan() && c.isAt(destination.End)); const takenChar = characters.find(c => c.isPacMan() && c.isAt(destination.end));
if (takenChar) { if (takenChar) {
takenChar.moveToSpawn(); takenChar.moveToSpawn();
stealFromPlayer(); stealFromPlayer();
@ -73,12 +73,12 @@ const Board: FC<BoardProps> = (
const positions: Position[] = []; const positions: Position[] = [];
if (selectedCharacter?.isPacMan()) { if (selectedCharacter?.isPacMan()) {
for (const tile of [...destination.Path ?? [], destination.End]) { for (const tile of [...destination.path ?? [], destination.end]) {
const currentTile = map[tile.Y][tile.X]; const currentTile = map[tile.y][tile.x];
function updateTileAndPlayerBox(isPowerPellet = false): void { function updateTileAndPlayerBox(isPowerPellet = false): void {
currentPlayer?.addPellet(new Pellet(isPowerPellet)); currentPlayer?.addPellet(new Pellet(isPowerPellet));
map[tile.Y][tile.X] = TileType.empty; map[tile.y][tile.x] = TileType.empty;
positions.push(tile); positions.push(tile);
} }
@ -112,10 +112,10 @@ const Board: FC<BoardProps> = (
<GameTile <GameTile
key={colIndex + rowIndex * colIndex} key={colIndex + rowIndex * colIndex}
type={tile} type={tile}
possiblePath={possiblePositions.find(p => p.End.X === colIndex && p.End.Y === rowIndex)} possiblePath={possiblePositions.find(p => p.end.x === colIndex && p.end.y === rowIndex)}
character={characters.find(c => c.isAt({X: colIndex, Y: rowIndex}))} character={characters.find(c => c.isAt({x: colIndex, y: rowIndex}))}
isSelected={selectedCharacter?.isAt({X: colIndex, Y: rowIndex})} isSelected={selectedCharacter?.isAt({x: colIndex, y: rowIndex})}
showPath={hoveredPosition?.Path?.find(pos => pos.X === colIndex && pos.Y === rowIndex) !== undefined} showPath={hoveredPosition?.path?.find(pos => pos.x === colIndex && pos.y === rowIndex) !== undefined}
handleMoveCharacter={handleMoveCharacter} handleMoveCharacter={handleMoveCharacter}
handleSelectCharacter={handleSelectCharacter} handleSelectCharacter={handleSelectCharacter}
handleStartShowPath={handleShowPath} handleStartShowPath={handleShowPath}
@ -181,11 +181,11 @@ const SelectPlayerModal: FC = () => {
{ {
allPlayers.map(player => allPlayers.map(player =>
<div key={player.Username} className={"border-b pb-1"}> <div key={player.username} className={"border-b pb-1"}>
<span className={"mx-2"}>{player.Username} 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}
onClick={() => { onClick={() => {
currentPlayer?.stealFrom(player); currentPlayer?.stealFrom(player);
close(); close();

View File

@ -21,7 +21,7 @@ const GameButton: FC<GameButtonProps> = (
const players = useAtomValue(playersAtom); const players = useAtomValue(playersAtom);
const activeRollDiceButton = useAtomValue(rollDiceButtonAtom); const activeRollDiceButton = useAtomValue(rollDiceButtonAtom);
if (players.length >= rules.minPlayers && (currentPlayer === undefined || currentPlayer.State === State.waitingForPlayers)) { if (players.length >= rules.minPlayers && (currentPlayer === undefined || currentPlayer.state === State.waitingForPlayers)) {
return <Button onClick={onReadyClick}>Ready</Button>; return <Button onClick={onReadyClick}>Ready</Button>;
} }
if (!thisPlayer?.isTurn()) { // TODO also show when waiting for other players if (!thisPlayer?.isTurn()) { // TODO also show when waiting for other players

View File

@ -36,7 +36,7 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map
if (!player.isTurn()) return; if (!player.isTurn()) return;
setSelectedDice(undefined); setSelectedDice(undefined);
wsService.send({Action: GameAction.rollDice}); wsService.send({action: GameAction.rollDice});
setActiveRollDiceButton(false); setActiveRollDiceButton(false);
} }
@ -46,12 +46,12 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map
} }
setSelectedDice(undefined); setSelectedDice(undefined);
const data: ActionMessage = { const data: ActionMessage = {
Action: GameAction.moveCharacter, action: GameAction.moveCharacter,
Data: { data: {
Dice: dice?.length ?? 0 > 0 ? dice : null, dice: dice?.length ?? 0 > 0 ? dice : null,
Players: players, players: players,
Ghosts: ghosts, ghosts: ghosts,
EatenPellets: eatenPellets eatenPellets: eatenPellets
} }
}; };
wsService.send(data); wsService.send(data);
@ -63,19 +63,19 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map
function sendPlayer(): void { function sendPlayer(): void {
wsService.send({ wsService.send({
Action: GameAction.playerInfo, action: GameAction.playerInfo,
Data: { data: {
Player: player, Spawns: getPacManSpawns(map) player: player, spawns: getPacManSpawns(map)
} as PlayerInfoData } as PlayerInfoData
}); });
} }
function sendReady(): void { function sendReady(): void {
wsService.send({Action: GameAction.ready}); wsService.send({action: GameAction.ready});
} }
function endTurn(): void { function endTurn(): void {
wsService.send({Action: GameAction.nextPlayer}); wsService.send({action: GameAction.nextPlayer});
} }
useEffect(() => { useEffect(() => {
@ -90,7 +90,7 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map
return ( return (
<> <>
<div className={"flex justify-center"}> <div className={"flex justify-center"}>
{players?.map(p => <PlayerStats key={p.Username} 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}/>

View File

@ -28,7 +28,7 @@ export const GameTile: FC<TileWithCharacterProps> = (
isSelected = false, isSelected = false,
showPath = false showPath = false
}) => ( }) => (
<Tile className={`${possiblePath?.End ? "border-4 border-white" : ""}`} <Tile className={`${possiblePath?.end ? "border-4 border-white" : ""}`}
type={type} type={type}
onClick={possiblePath ? () => handleMoveCharacter?.(possiblePath) : undefined} onClick={possiblePath ? () => handleMoveCharacter?.(possiblePath) : undefined}
onMouseEnter={possiblePath ? () => handleStartShowPath?.(possiblePath) : undefined} onMouseEnter={possiblePath ? () => handleStartShowPath?.(possiblePath) : undefined}
@ -48,7 +48,7 @@ export const GameTile: FC<TileWithCharacterProps> = (
</Tile> </Tile>
); );
const Circle: FC<{ colour?: Colour } & ComponentProps> = ({colour = Colour.White, className}) => ( const Circle: FC<{ colour?: Colour } & ComponentProps> = ({colour = Colour.white, className}) => (
<div className={`flex-center w-full h-full ${className}`}> <div className={`flex-center w-full h-full ${className}`}>
<div className={`w-1/2 h-1/2 rounded-full`} <div className={`w-1/2 h-1/2 rounded-full`}
style={{backgroundColor: colour}}/> style={{backgroundColor: colour}}/>
@ -93,7 +93,7 @@ const Tile: FC<TileProps> = (
useEffect(() => { useEffect(() => {
function handleResize(): void { function handleResize(): void {
const newSize = Math.floor(window.innerWidth / 12); const newSize = Math.floor(window.innerWidth / 16);
setTileSize(newSize); setTileSize(newSize);
} }
@ -110,8 +110,8 @@ const Tile: FC<TileProps> = (
onMouseEnter={onMouseEnter} onMouseEnter={onMouseEnter}
onMouseLeave={onMouseLeave}> onMouseLeave={onMouseLeave}>
{children} {children}
{type === TileType.pellet && <Circle colour={Colour.Yellow}/>} {type === TileType.pellet && <Circle colour={Colour.yellow}/>}
{type === TileType.powerPellet && <Circle colour={Colour.Red}/>} {type === TileType.powerPellet && <Circle colour={Colour.red}/>}
</div> </div>
); );
}; };
@ -139,7 +139,7 @@ const CharacterComponent: FC<CharacterComponentProps> = (
}) => { }) => {
function getSide() { function getSide() {
switch (character?.Position?.Direction) { switch (character?.position?.direction) {
case Direction.up: case Direction.up:
return "right-1/4 top-0"; return "right-1/4 top-0";
case Direction.down: case Direction.down:
@ -155,7 +155,7 @@ const CharacterComponent: FC<CharacterComponentProps> = (
return ( return (
<div className={`rounded-full w-4/5 h-4/5 cursor-pointer hover:border border-black relative ${className}`} <div className={`rounded-full w-4/5 h-4/5 cursor-pointer hover:border border-black relative ${className}`}
style={{backgroundColor: `${character.Colour}`}} style={{backgroundColor: `${character.colour}`}}
onClick={() => onClick?.(character)}> onClick={() => onClick?.(character)}>
<div> <div>
<div className={`absolute ${getSide()} w-1/2 h-1/2 rounded-full bg-black`}/> <div className={`absolute ${getSide()} w-1/2 h-1/2 rounded-full bg-black`}/>

View File

@ -11,16 +11,16 @@ const PlayerStats: FC<{ player: Player } & ComponentProps> = (
}) => { }) => {
const currentPlayerName = useAtomValue(currentPlayerNameAtom); const currentPlayerName = useAtomValue(currentPlayerNameAtom);
return ( return (
<div key={player.Colour} <div key={player.colour}
className={`w-fit m-2 ${player.State === State.disconnected ? "text-gray-500" : ""} ${className}`} id={id}> 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 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.disconnected ? {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>
</> : </> :
<p>{player.State === State.waitingForPlayers ? "Waiting" : "Ready"}</p>} <p>{player.state === State.waitingForPlayers ? "Waiting" : "Ready"}</p>}
</div> </div>
); );
}; };

View File

@ -2,28 +2,28 @@ import Pellet from "./pellet";
import {Colour} from "./colour"; import {Colour} from "./colour";
export default class Box { export default class Box {
public Pellets: Pellet[]; public pellets: Pellet[];
public readonly Colour: Colour; public readonly colour: Colour;
public constructor({Colour, Pellets = []}: BoxProps) { public constructor({colour, pellets = []}: BoxProps) {
this.Colour = Colour; this.colour = colour;
this.Pellets = Pellets; this.pellets = pellets;
} }
get powerPellet(): Pellet | undefined { get powerPellet(): Pellet | undefined {
return this.Pellets.find(pellet => pellet.IsPowerPellet); return this.pellets.find(pellet => pellet.isPowerPellet);
} }
get count(): number { get count(): number {
return this.Pellets.filter(pellet => !pellet.IsPowerPellet).length; return this.pellets.filter(pellet => !pellet.isPowerPellet).length;
} }
get countPowerPellets(): number { get countPowerPellets(): number {
return this.Pellets.filter(pellet => pellet.IsPowerPellet).length; return this.pellets.filter(pellet => pellet.isPowerPellet).length;
} }
public addPellet(pellet: Pellet): void { public addPellet(pellet: Pellet): void {
this.Pellets.push(pellet); this.pellets.push(pellet);
} }
} }

View File

@ -8,76 +8,76 @@ export enum CharacterType {
} }
export class Character { export class Character {
public readonly Colour: Colour; public readonly colour: Colour;
public Position: Path | null; public position: Path | null;
public IsEatable: boolean; public isEatable: boolean;
public readonly SpawnPosition: DirectionalPosition | null; public readonly spawnPosition: DirectionalPosition | null;
public readonly Type: CharacterType; public readonly type: CharacterType;
public constructor( public constructor(
{ {
Colour, colour,
Position = null, position = null,
Type = CharacterType.dummy, type = CharacterType.dummy,
IsEatable = Type === CharacterType.pacMan, isEatable = type === CharacterType.pacMan,
SpawnPosition = null spawnPosition = null
}: CharacterProps) { }: CharacterProps) {
this.Colour = Colour; this.colour = colour;
this.IsEatable = IsEatable; this.isEatable = isEatable;
this.SpawnPosition = SpawnPosition; this.spawnPosition = spawnPosition;
if (Position) { if (position) {
this.Position = Position; this.position = position;
} else { } else {
this.Position = SpawnPosition ? { this.position = spawnPosition ? {
End: SpawnPosition!.At, end: spawnPosition!.at,
Direction: SpawnPosition!.Direction direction: spawnPosition!.direction
} : null; } : null;
} }
this.Type = Type; this.type = type;
} }
public follow(path: Path): void { public follow(path: Path): void {
if (!this.Position) { if (!this.position) {
this.Position = path; this.position = path;
} else { } else {
this.Position.End = path.End; this.position.end = path.end;
this.Position.Direction = path.Direction; this.position.direction = path.direction;
this.Position.Path = undefined; this.position.path = undefined;
} }
} }
public isPacMan(): boolean { public isPacMan(): boolean {
return this.Type === CharacterType.pacMan; return this.type === CharacterType.pacMan;
} }
public isGhost(): boolean { public isGhost(): boolean {
return this.Type === CharacterType.ghost; return this.type === CharacterType.ghost;
} }
public moveToSpawn(): void { public moveToSpawn(): void {
if (!this.SpawnPosition) return; if (!this.spawnPosition) return;
this.follow({End: this.SpawnPosition.At, Direction: this.SpawnPosition.Direction}); this.follow({end: this.spawnPosition.at, direction: this.spawnPosition.direction});
} }
public isAt(position: Position): boolean { public isAt(position: Position): boolean {
return this.Position !== null && this.Position.End.X === position.X && this.Position.End.Y === position.Y; return this.position !== null && this.position.end.x === position.x && this.position.end.y === position.y;
} }
} }
export class PacMan extends Character { export class PacMan extends Character {
public constructor({Colour, Position, IsEatable = true, SpawnPosition, Type = CharacterType.pacMan}: CharacterProps) { public constructor({colour, position, isEatable = true, spawnPosition, type = CharacterType.pacMan}: CharacterProps) {
super({Colour: Colour, Position: Position, IsEatable: IsEatable, SpawnPosition: SpawnPosition, Type: Type}); super({colour: colour, position: position, isEatable: isEatable, spawnPosition: spawnPosition, type: type});
} }
} }
export class Ghost extends Character { export class Ghost extends Character {
public constructor({Colour, Position, IsEatable, SpawnPosition, Type = CharacterType.ghost}: CharacterProps) { public constructor({colour, position, isEatable, spawnPosition, type = CharacterType.ghost}: CharacterProps) {
super({Colour: Colour, Position: Position, IsEatable: IsEatable, SpawnPosition: SpawnPosition, Type: Type}); super({colour: colour, position: position, isEatable: isEatable, spawnPosition: spawnPosition, type: type});
} }
} }
@ -85,11 +85,11 @@ export class Dummy extends Character {
public constructor(position: Path) { // TODO see-through public constructor(position: Path) { // TODO see-through
super({ super({
Colour: Colour.Grey, colour: Colour.grey,
Position: position, position: position,
IsEatable: false, isEatable: false,
SpawnPosition: {At: {X: 0, Y: 0}, Direction: Direction.up}, spawnPosition: {at: {x: 0, y: 0}, direction: Direction.up},
Type: CharacterType.dummy, type: CharacterType.dummy,
}); });
} }

View File

@ -1,11 +1,11 @@
export enum Colour { export enum Colour {
White = "white", white = "white",
Red = "red", red = "red",
Blue = "blue", blue = "blue",
Yellow = "yellow", yellow = "yellow",
Green = "green", green = "green",
Purple = "purple", purple = "purple",
Grey = "grey", grey = "grey",
} }
export const getColours = (): Colour[] => Object.values(Colour); export const getColours = (): Colour[] => Object.values(Colour);

View File

@ -9,18 +9,22 @@ import {Direction} from "./direction";
* 4 = ghost spawn * 4 = ghost spawn
* 5 = pacman spawn * 5 = pacman spawn
*/ */
export const testMap: GameMap = [ // TODO create map class object using tile type enum export const customMap: GameMap = [
[1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1], [1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1],
[1, 2, 0, 0, 0, 2, 0, 0, 0, 2, 1], [1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 1],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1], [1, 2, 1, 0, 1, 2, 1, 0, 1, 2, 1, 0, 1, 2, 1],
[1, 0, 1, 5, 1, 0, 1, 4, 1, 0, 1], [0, 2, 1, 5, 1, 2, 1, 0, 1, 2, 1, 5, 1, 2, 0],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1], [1, 2, 1, 0, 0, 2, 0, 3, 0, 2, 0, 0, 1, 2, 1],
[0, 2, 0, 0, 0, 3, 0, 0, 0, 2, 0], [1, 2, 1, 1, 1, 2, 1, 0, 1, 2, 1, 1, 1, 2, 1],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1], [1, 2, 2, 2, 2, 2, 1, 4, 1, 2, 2, 2, 2, 2, 1],
[1, 0, 1, 4, 1, 0, 1, 5, 1, 0, 1], [1, 3, 1, 1, 1, 2, 1, 0, 1, 2, 1, 1, 1, 3, 1],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1], [1, 2, 2, 2, 2, 2, 1, 4, 1, 2, 2, 2, 2, 2, 1],
[1, 2, 0, 0, 0, 2, 0, 0, 0, 2, 1], [1, 2, 1, 1, 1, 2, 1, 0, 1, 2, 1, 1, 1, 2, 1],
[1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1], [1, 2, 1, 0, 0, 2, 0, 3, 0, 2, 0, 0, 1, 2, 1],
[0, 2, 1, 5, 1, 2, 1, 0, 1, 2, 1, 5, 1, 2, 0],
[1, 2, 1, 0, 1, 2, 1, 0, 1, 2, 1, 0, 1, 2, 1],
[1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 1],
[1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1],
]; ];
export function getCharacterSpawns(map: GameMap): { type: CharacterType, position: DirectionalPosition }[] { export function getCharacterSpawns(map: GameMap): { type: CharacterType, position: DirectionalPosition }[] {
@ -30,10 +34,10 @@ export function getCharacterSpawns(map: GameMap): { type: CharacterType, positio
for (let col = 0; col < map.length; col++) { for (let col = 0; col < map.length; col++) {
// TODO find direction // TODO find direction
if (map[row][col] === 4) { if (map[row][col] === 4) {
result.push({type: CharacterType.ghost, position: {At: {X: col, Y: row}, Direction: Direction.up}}); result.push({type: CharacterType.ghost, position: {at: {x: col, y: row}, direction: Direction.up}});
} else if (map[row][col] === 5) { } else if (map[row][col] === 5) {
result.push({ result.push({
type: CharacterType.pacMan, position: {At: {X: col, Y: row}, Direction: Direction.up} type: CharacterType.pacMan, position: {at: {x: col, y: row}, direction: Direction.up}
}); });
} }
} }

View File

@ -1,7 +1,7 @@
export default class Pellet { export default class Pellet {
public readonly IsPowerPellet: boolean; public readonly isPowerPellet: boolean;
public constructor(isPowerPellet = false) { public constructor(isPowerPellet = false) {
this.IsPowerPellet = isPowerPellet; this.isPowerPellet = isPowerPellet;
} }
} }

View File

@ -14,37 +14,37 @@ export enum State {
} }
export default class Player { export default class Player {
public readonly Username: 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.Username = props.Username; 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 ?? {
Colour: props.Colour, colour: props.colour,
Type: CharacterType.pacMan type: CharacterType.pacMan
}); });
this.State = props.State ?? State.waitingForPlayers; this.state = props.state ?? State.waitingForPlayers;
} }
public isTurn(): boolean { public isTurn(): boolean {
const store = getDefaultStore(); const store = getDefaultStore();
return store.get(currentPlayerNameAtom) === this.Username; return store.get(currentPlayerNameAtom) === this.username;
} }
public addPellet(pellet: Pellet): void { public addPellet(pellet: Pellet): void {
this.Box.addPellet(pellet); this.box.addPellet(pellet);
} }
public stealFrom(other: Player): void { public stealFrom(other: Player): void {
for (let i = 0; i < rules.maxStealPellets; i++) { for (let i = 0; i < rules.maxStealPellets; i++) {
const pellet = other.Box.Pellets.pop(); const pellet = other.box.pellets.pop();
if (pellet) if (pellet)
this.Box.addPellet(pellet); this.box.addPellet(pellet);
} }
const store = getDefaultStore(); const store = getDefaultStore();
store.set(playersAtom, store.get(playersAtom).map(player => player)); store.set(playersAtom, store.get(playersAtom).map(player => player));

View File

@ -13,8 +13,8 @@ import {Direction, getDirections} from "./direction";
* @returns An array of paths the character can move to * @returns An array of paths the character can move to
*/ */
export default function findPossiblePositions(board: GameMap, character: Character, steps: number, characters: Character[]): Path[] { export default function findPossiblePositions(board: GameMap, character: Character, steps: number, characters: Character[]): Path[] {
if (!character.Position || !character.SpawnPosition) throw new Error("Character has no position or spawn position"); if (!character.position || !character.spawnPosition) throw new Error("Character has no position or spawn position");
return findPossibleRecursive(board, character.Position, steps, character, characters); return findPossibleRecursive(board, character.position, steps, character, characters);
}; };
/** /**
@ -29,7 +29,7 @@ export default function findPossiblePositions(board: GameMap, character: Charact
function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number, character: Character, characters: Character[]): Path[] { function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number, character: Character, characters: Character[]): Path[] {
const paths: Path[] = []; const paths: Path[] = [];
if (isOutsideBoard(currentPath, board.length)) { if (isOutsideBoard(currentPath, board.length)) { // TODO not working on new map
if (character.isPacMan()) { if (character.isPacMan()) {
return addTeleportationTiles(board, currentPath, steps, character, characters); return addTeleportationTiles(board, currentPath, steps, character, characters);
} }
@ -61,7 +61,7 @@ function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number,
} }
function isCharactersSpawn(currentPath: Path, character: Character): boolean { function isCharactersSpawn(currentPath: Path, character: Character): boolean {
return character.SpawnPosition?.At.X === currentPath.End.X && character.SpawnPosition.At.Y === currentPath.End.Y; return character.spawnPosition?.at.x === currentPath.end.x && character.spawnPosition.at.y === currentPath.end.y;
} }
/** /**
@ -72,7 +72,7 @@ function isCharactersSpawn(currentPath: Path, character: Character): boolean {
* @returns True if the character is a ghost and hits Pac-Man * @returns True if the character is a ghost and hits Pac-Man
*/ */
function ghostHitsPacMan(character: Character, currentPath: Path, characters: Character[]): Character | undefined | false { function ghostHitsPacMan(character: Character, currentPath: Path, characters: Character[]): Character | undefined | false {
return character.isGhost() && characters.find(c => c.isPacMan() && c.isAt(currentPath.End)); return character.isGhost() && characters.find(c => c.isPacMan() && c.isAt(currentPath.end));
} }
/** /**
@ -83,7 +83,7 @@ function ghostHitsPacMan(character: Character, currentPath: Path, characters: Ch
* @returns True if the character hits another character * @returns True if the character hits another character
*/ */
function characterHitsAnotherCharacter(character: Character, currentPath: Path, characters: Character[]): boolean { function characterHitsAnotherCharacter(character: Character, currentPath: Path, characters: Character[]): boolean {
return characters.find(c => c !== character && c.isAt(currentPath.End)) !== undefined; return characters.find(c => c !== character && c.isAt(currentPath.end)) !== undefined;
} }
/** /**
@ -91,10 +91,10 @@ function characterHitsAnotherCharacter(character: Character, currentPath: Path,
* @param currentPos The current path the character is on * @param currentPos The current path the character is on
*/ */
function addToPath(currentPos: Path): void { function addToPath(currentPos: Path): void {
if (!currentPos.Path) { if (!currentPos.path) {
currentPos.Path = []; currentPos.path = [];
} else if (!currentPos.Path.includes(currentPos.End)) { } else if (!currentPos.path.includes(currentPos.end)) {
currentPos.Path = [...currentPos.Path, currentPos.End]; currentPos.path = [...currentPos.path, currentPos.end];
} }
} }
@ -115,31 +115,31 @@ function tryMove(board: GameMap, path: Path, direction: Direction, steps: number
switch (direction) { switch (direction) {
case Direction.left: case Direction.left:
return { return {
X: path.End.X - 1, x: path.end.x - 1,
Y: path.End.Y y: path.end.y
}; };
case Direction.up: case Direction.up:
return { return {
X: path.End.X, x: path.end.x,
Y: path.End.Y - 1 y: path.end.y - 1
}; };
case Direction.right: case Direction.right:
return { return {
X: path.End.X + 1, x: path.end.x + 1,
Y: path.End.Y y: path.end.y
}; };
case Direction.down: case Direction.down:
return { return {
X: path.End.X, x: path.end.x,
Y: path.End.Y + 1 y: path.end.y + 1
}; };
} }
} }
if (path.Direction !== (direction + 2) % 4) { if (path.direction !== (direction + 2) % 4) {
// TODO getNewPosition() and check if a character is on the new position // TODO getNewPosition() and check if a character is on the new position
return findPossibleRecursive(board, { return findPossibleRecursive(board, {
End: getNewPosition(), Direction: direction, Path: path.Path end: getNewPosition(), direction: direction, path: path.path
}, steps, character, characters); }, steps, character, characters);
} }
return []; return [];
@ -157,10 +157,10 @@ function addTeleportationTiles(board: GameMap, currentPath: Path, steps: number,
const possiblePositions = findTeleportationTiles(board); const possiblePositions = findTeleportationTiles(board);
const paths: Path[] = []; const paths: Path[] = [];
for (const pos of possiblePositions) { for (const pos of possiblePositions) {
if (pos.End.X !== interval(0, board.length - 1, currentPath.End.X) || if (pos.end.x !== interval(0, board.length - 1, currentPath.end.x) ||
pos.End.Y !== interval(0, board.length - 1, currentPath.End.Y)) { pos.end.y !== interval(0, board.length - 1, currentPath.end.y)) {
pos.Path = currentPath.Path; pos.path = currentPath.path;
paths.push(...findPossibleRecursive(board, pos, steps, character, characters)); paths.push(...findPossibleRecursive(board, pos, steps, character, characters));
} }
} }
@ -172,19 +172,19 @@ function interval(lower: number, upper: number, value: number): number {
} }
/** /**
* Finds all the teleportation tiles on the board * Finds all the teleportation tiles on the map
* @param board The board the character is on * @param map The map the character is on
* @returns An array of paths containing the teleportation tiles * @returns An array of paths containing the teleportation tiles
*/ */
function findTeleportationTiles(board: GameMap): Path[] { function findTeleportationTiles(map: GameMap): Path[] {
const possiblePositions: Path[] = []; const possiblePositions: Path[] = [];
const edge = [0, board.length - 1]; const edge = [0, map.length - 1];
for (const e of edge) { for (const e of edge) {
for (let i = 0; i < board[e].length; i++) { for (let i = 0; i < map[e].length; i++) {
pushPath(board, possiblePositions, i, e); pushPath(map, possiblePositions, i, e);
pushPath(board, possiblePositions, e, i); pushPath(map, possiblePositions, e, i);
} }
} }
@ -199,8 +199,8 @@ function findTeleportationTiles(board: GameMap): Path[] {
* @param y The y position of the path * @param y The y position of the path
*/ */
function pushPath(board: GameMap, possiblePositions: Path[], x: number, y: number): void { function pushPath(board: GameMap, possiblePositions: Path[], x: number, y: number): void {
if (board[x][y] !== TileType.wall) { if (board[x] && board[x][y] !== TileType.wall) {
possiblePositions.push({End: {X: x, Y: y}, Direction: findDirection(x, y, board.length)}); possiblePositions.push({end: {x: x, y: y}, direction: findDirection(x, y, board.length)});
} }
} }
@ -230,8 +230,8 @@ function findDirection(x: number, y: number, boardSize: number): Direction {
* @param boardSize The size of the board * @param boardSize The size of the board
*/ */
function isOutsideBoard(currentPos: Path, boardSize: number): boolean { function isOutsideBoard(currentPos: Path, boardSize: number): boolean {
const pos = currentPos.End; const pos = currentPos.end;
return pos.X < 0 || pos.X >= boardSize || pos.Y < 0 || pos.Y >= boardSize; return pos.x < 0 || pos.x >= boardSize || pos.y < 0 || pos.y >= boardSize;
} }
/** /**
@ -240,8 +240,8 @@ function isOutsideBoard(currentPos: Path, boardSize: number): boolean {
* @param currentPos The current position of the character * @param currentPos The current position of the character
*/ */
function isWall(board: GameMap, currentPos: Path): boolean { function isWall(board: GameMap, currentPos: Path): boolean {
const pos = currentPos.End; const pos = currentPos.end;
return board[pos.Y][pos.X] === TileType.wall; // Shouldn't work, but it does return board[pos.y][pos.x] === TileType.wall; // Shouldn't work, but it does
} }
/** /**
@ -250,8 +250,8 @@ function isWall(board: GameMap, currentPos: Path): boolean {
* @param currentPos The current position of the character * @param currentPos The current position of the character
*/ */
function isSpawn(board: GameMap, currentPos: Path) { function isSpawn(board: GameMap, currentPos: Path) {
const pos = currentPos.End; const pos = currentPos.end;
return board[pos.Y][pos.X] === TileType.pacmanSpawn || board[pos.Y][pos.X] === TileType.ghostSpawn; return board[pos.y][pos.x] === TileType.pacmanSpawn || board[pos.y][pos.x] === TileType.ghostSpawn;
} }
/** /**
@ -260,8 +260,8 @@ function isSpawn(board: GameMap, currentPos: Path) {
* @param character The current character * @param character The current character
*/ */
function isOwnSpawn(currentPos: Path, character: Character): boolean { function isOwnSpawn(currentPos: Path, character: Character): boolean {
const pos = currentPos.End; const pos = currentPos.end;
const charPos = character.SpawnPosition!.At; const charPos = character.spawnPosition!.at;
return charPos.X === pos.X && charPos.Y === pos.Y; return charPos.x === pos.x && charPos.y === pos.y;
} }

View File

@ -2,7 +2,7 @@ import React, {FC, useEffect} from "react";
import {GameComponent} from "../components/gameComponent"; import {GameComponent} from "../components/gameComponent";
import {useAtomValue} from "jotai"; import {useAtomValue} from "jotai";
import {thisPlayerAtom} from "../utils/state"; import {thisPlayerAtom} from "../utils/state";
import {testMap} from "../game/map"; import {customMap} from "../game/map";
const Game: FC = () => { // TODO gameId in path const Game: FC = () => { // TODO gameId in path
const player = useAtomValue(thisPlayerAtom); const player = useAtomValue(thisPlayerAtom);
@ -16,7 +16,7 @@ const Game: FC = () => { // TODO gameId in path
}, [player]); }, [player]);
if (player) { if (player) {
return <GameComponent player={player} map={testMap}/>; return <GameComponent player={player} map={customMap}/>;
} else { } else {
return null; return null;
} }

View File

@ -17,8 +17,8 @@ const Home: FC = () => {
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({
Username: input.current.value, username: input.current.value,
Colour: dropdown.current.value as Colour, colour: dropdown.current.value as Colour,
}); });
setPlayer(player); setPlayer(player);
navigate("/game"); navigate("/game");

View File

@ -3,7 +3,7 @@ import {atom, useAtomValue} from "jotai";
import {Button} from "../components/button"; import {Button} from "../components/button";
import {thisPlayerAtom} from "../utils/state"; import {thisPlayerAtom} from "../utils/state";
import {getData, postData} from "../utils/api"; import {getData, postData} from "../utils/api";
import {getPacManSpawns, testMap} from "../game/map"; import {customMap, getPacManSpawns} from "../game/map";
import {useNavigate} from "react-router-dom"; import {useNavigate} from "react-router-dom";
const fetchAtom = atom(async () => { const fetchAtom = atom(async () => {
@ -20,15 +20,15 @@ const LobbyPage: FC = () => { // TODO check if player is defined in storage, if
async function createGame(): Promise<void> { async function createGame(): Promise<void> {
const response = await postData("/game/create", { const response = await postData("/game/create", {
body: {Player: thisPlayer, Spawns: getPacManSpawns(testMap)} as PlayerInfoData body: {player: thisPlayer, spawns: getPacManSpawns(customMap)} as PlayerInfoData
}); });
const data = await response.json();
if (response.ok) { if (response.ok) {
const data = await response.json();
console.debug("Game created: ", data); console.debug("Game created: ", data);
// TODO redirect to game page // TODO redirect to game page
} else { } else {
const data = await response.text();
console.error("Error: ", data); console.error("Error: ", data);
// TODO display error // TODO display error
} }

View File

@ -28,13 +28,14 @@ const Login = () => {
body: {username: user.username, password: user.password} as User body: {username: user.username, password: user.password} as User
}) })
const data = await response.json();
if (response.ok) { if (response.ok) {
const data = await response.json();
console.debug("Login successful: ", data); console.debug("Login successful: ", data);
setThisPlayer(new Player(data as PlayerProps)); setThisPlayer(new Player(data as PlayerProps));
navigate("/lobby"); navigate("/lobby");
} else { } else {
const data = await response.text();
console.error("Error: ", data); console.error("Error: ", data);
// TODO display error // TODO display error
} }

View File

@ -25,22 +25,22 @@ interface InputProps extends ComponentProps {
} }
interface CharacterProps { interface CharacterProps {
Colour: import("../game/colour").Colour, colour: import("../game/colour").Colour,
Position?: Path | null, position?: Path | null,
IsEatable?: boolean, isEatable?: boolean,
SpawnPosition?: DirectionalPosition | null, spawnPosition?: DirectionalPosition | null,
Type?: import("../game/character").CharacterType, type?: import("../game/character").CharacterType,
} }
interface BoxProps { interface BoxProps {
Pellets?: import("../game/pellet").default[], pellets?: import("../game/pellet").default[],
readonly Colour: import("../game/colour").Colour, readonly colour: import("../game/colour").Colour,
} }
interface PlayerProps { interface PlayerProps {
readonly Username: 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,
State?: import("../game/player").State, state?: import("../game/player").State,
} }

View File

@ -5,8 +5,8 @@ type Setter<T> = React.Dispatch<React.SetStateAction<T>>;
type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView; type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView;
type ActionMessage<T = any> = { type ActionMessage<T = any> = {
readonly Action: import("../utils/actions").GameAction, readonly action: import("../utils/actions").GameAction,
readonly Data?: T readonly data?: T
} }
type Action<T> = (obj: T) => void; type Action<T> = (obj: T) => void;
@ -20,19 +20,19 @@ type SelectedDice = {
index: number index: number
}; };
type Position = { X: number, Y: number }; type Position = { x: number, y: number };
type GameMap = number[][]; type GameMap = number[][];
type DirectionalPosition = { type DirectionalPosition = {
At: Position, at: Position,
Direction: import("../game/direction").Direction direction: import("../game/direction").Direction
} }
type Path = { type Path = {
Path?: Position[] | null, path?: Position[] | null,
End: Position, end: Position,
Direction: import("../game/direction").Direction direction: import("../game/direction").Direction
} }
type Game = { type Game = {
@ -55,6 +55,6 @@ type ApiRequest = {
} }
type PlayerInfoData = { type PlayerInfoData = {
readonly Player: PlayerProps, readonly player: PlayerProps,
readonly Spawns: DirectionalPosition[], readonly spawns: DirectionalPosition[],
} }

View File

@ -1,6 +1,6 @@
import Player from "../game/player"; import Player from "../game/player";
import {CharacterType, Ghost} from "../game/character"; import {CharacterType, Ghost} from "../game/character";
import {getCharacterSpawns, testMap} from "../game/map"; import {customMap, getCharacterSpawns} 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, rollDiceButtonAtom} from "./state"; import {currentPlayerNameAtom, diceAtom, ghostsAtom, playersAtom, rollDiceButtonAtom} from "./state";
@ -17,12 +17,12 @@ export enum GameAction {
const store = getDefaultStore(); const store = getDefaultStore();
const ghostsProps: CharacterProps[] = [ const ghostsProps: CharacterProps[] = [
{Colour: Colour.Purple}, {colour: Colour.purple},
{Colour: Colour.Purple}, {colour: Colour.purple},
]; ];
let spawns = getCharacterSpawns(testMap).filter(spawn => spawn.type === CharacterType.ghost); let spawns = getCharacterSpawns(customMap).filter(spawn => spawn.type === CharacterType.ghost);
ghostsProps.forEach(ghost => { ghostsProps.forEach(ghost => {
ghost.SpawnPosition = spawns.pop()?.position; ghost.spawnPosition = spawns.pop()?.position;
}); });
const ghosts = ghostsProps.map(props => new Ghost(props)); const ghosts = ghostsProps.map(props => new Ghost(props));
@ -33,21 +33,21 @@ export const doAction: MessageEventFunction<string> = (event): void => { // TODO
const message: ActionMessage = JSON.parse(event.data); const message: ActionMessage = JSON.parse(event.data);
console.debug("Received message:", message); console.debug("Received message:", message);
switch (message.Action as GameAction) { switch (message.action as GameAction) {
case GameAction.rollDice: case GameAction.rollDice:
setDice(message.Data); setDice(message.data);
break; break;
case GameAction.moveCharacter: case GameAction.moveCharacter:
moveCharacter(message.Data); moveCharacter(message.data);
break; break;
case GameAction.playerInfo: case GameAction.playerInfo:
playerInfo(message.Data); playerInfo(message.data);
break; break;
case GameAction.ready: case GameAction.ready:
ready(message.Data); ready(message.data);
break; break;
case GameAction.nextPlayer: case GameAction.nextPlayer:
nextPlayer(message.Data); nextPlayer(message.data);
break; break;
} }
}; };
@ -56,17 +56,17 @@ function setDice(data?: number[]): void {
store.set(diceAtom, data); store.set(diceAtom, data);
} }
type MoveCharacterData = { Dice: number[], Players: PlayerProps[], Ghosts: CharacterProps[], EatenPellets: Position[] }; type MoveCharacterData = { dice: number[], players: PlayerProps[], ghosts: CharacterProps[], eatenPellets: Position[] };
function moveCharacter(data?: MoveCharacterData): void { function moveCharacter(data?: MoveCharacterData): void {
store.set(diceAtom, data?.Dice); store.set(diceAtom, data?.dice);
updatePlayers(data); updatePlayers(data);
updateGhosts(data); updateGhosts(data);
removeEatenPellets(data); removeEatenPellets(data);
} }
function updatePlayers(data?: MoveCharacterData): void { function updatePlayers(data?: MoveCharacterData): void {
const updatedPlayers = data?.Players; const updatedPlayers = data?.players;
if (updatedPlayers) { if (updatedPlayers) {
const newList: Player[] = updatedPlayers.map(p => new Player(p)); const newList: Player[] = updatedPlayers.map(p => new Player(p));
@ -75,7 +75,7 @@ function updatePlayers(data?: MoveCharacterData): void {
} }
function updateGhosts(data?: MoveCharacterData): void { function updateGhosts(data?: MoveCharacterData): void {
const updatedGhosts = data?.Ghosts; const updatedGhosts = data?.ghosts;
if (updatedGhosts) { if (updatedGhosts) {
const newList: Ghost[] = updatedGhosts.map(g => new Ghost(g)); const newList: Ghost[] = updatedGhosts.map(g => new Ghost(g));
@ -84,27 +84,27 @@ function updateGhosts(data?: MoveCharacterData): void {
} }
function removeEatenPellets(data?: MoveCharacterData): void { function removeEatenPellets(data?: MoveCharacterData): void {
const pellets = data?.EatenPellets; const pellets = data?.eatenPellets;
for (const pellet of pellets ?? []) { for (const pellet of pellets ?? []) {
testMap[pellet.Y][pellet.X] = TileType.empty; customMap[pellet.y][pellet.x] = TileType.empty;
} }
} }
function playerInfo(data?: PlayerProps[]): void { // TODO missing data when refreshing page 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(customMap).filter(spawn => spawn.type === CharacterType.pacMan);
store.set(playersAtom, playerProps.map(p => new Player(p))); store.set(playersAtom, playerProps.map(p => new Player(p)));
} }
type ReadyData = { AllReady: boolean, Players: PlayerProps[] } | string; type ReadyData = { allReady: boolean, 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.Players[0].Username); store.set(currentPlayerNameAtom, data.players[0].username);
} }
} else { } else {
console.error("Error:", data); console.error("Error:", data);

View File

@ -11,7 +11,7 @@ export const playersAtom = atom<Player[]>([]);
/** /**
* All player characters (Pac-Man) in the game. * All player characters (Pac-Man) in the game.
*/ */
export const playerCharactersAtom = atom(get => get(playersAtom).map(player => player.PacMan)); export const playerCharactersAtom = atom(get => get(playersAtom).map(player => player.pacMan));
/** /**
* All ghosts in the game. * All ghosts in the game.
*/ */
@ -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.Username === 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.

View File

@ -1,6 +1,5 @@
import {beforeEach, expect, test} from "vitest"; import {beforeEach, expect, test} from "vitest";
import possibleMovesAlgorithm from "../../src/game/possibleMovesAlgorithm"; import possibleMovesAlgorithm from "../../src/game/possibleMovesAlgorithm";
import {testMap} from "../../src/game/map";
import {Ghost, PacMan} from "../../src/game/character"; import {Ghost, PacMan} from "../../src/game/character";
import {Direction} from "../../src/game/direction"; import {Direction} from "../../src/game/direction";
import {Colour} from "../../src/game/colour"; import {Colour} from "../../src/game/colour";
@ -8,47 +7,61 @@ import {Colour} from "../../src/game/colour";
let pacMan: PacMan; let pacMan: PacMan;
let ghost: Ghost; let ghost: Ghost;
const testMap: GameMap = [
[1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1],
[1, 2, 0, 0, 0, 2, 0, 0, 0, 2, 1],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1],
[1, 0, 1, 5, 1, 0, 1, 4, 1, 0, 1],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1],
[0, 2, 0, 0, 0, 3, 0, 0, 0, 2, 0],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1],
[1, 0, 1, 4, 1, 0, 1, 5, 1, 0, 1],
[1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1],
[1, 2, 0, 0, 0, 2, 0, 0, 0, 2, 1],
[1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1],
];
beforeEach(() => { beforeEach(() => {
pacMan = new PacMan({ pacMan = new PacMan({
Colour: Colour.Yellow, SpawnPosition: {At: {X: 3, Y: 3}, Direction: Direction.up} colour: Colour.yellow, spawnPosition: {at: {x: 3, y: 3}, direction: Direction.up}
}); });
ghost = new Ghost({ ghost = new Ghost({
Colour: Colour.Red, SpawnPosition: {At: {X: 3, Y: 3}, Direction: Direction.up} colour: Colour.red, spawnPosition: {at: {x: 3, y: 3}, direction: Direction.up}
}); });
}); });
test("Pac-Man rolls one from start, should return one position", () => { test("Pac-Man rolls one from start, should return one position", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 1, []); const result = possibleMovesAlgorithm(testMap, pacMan, 1, []);
expect(result.length).toBe(1); expect(result.length).toBe(1);
expect(result[0].Path?.length).toBe(0); expect(result[0].path?.length).toBe(0);
expect(result).toEqual([{End: {X: 3, Y: 2}, Direction: Direction.up, Path: []}] as Path[]); expect(result).toEqual([{end: {x: 3, y: 2}, direction: Direction.up, path: []}] as Path[]);
}); });
test("Pac-Man rolls two from start, should return one position", () => { test("Pac-Man rolls two from start, should return one position", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 2, []); const result = possibleMovesAlgorithm(testMap, pacMan, 2, []);
expect(result.length).toBe(1); expect(result.length).toBe(1);
expect(result[0].Path?.length).toBe(1); expect(result[0].path?.length).toBe(1);
expect(result).toEqual([{End: {X: 3, Y: 1}, Direction: Direction.up, Path: [{X: 3, Y: 2}]}] as Path[]); expect(result).toEqual([{end: {x: 3, y: 1}, direction: Direction.up, path: [{x: 3, y: 2}]}] as Path[]);
}); });
test("Pac-Man rolls three from start, should return two positions", () => { test("Pac-Man rolls three from start, should return two positions", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 3, []); const result = possibleMovesAlgorithm(testMap, pacMan, 3, []);
expect(result.length).toBe(2); expect(result.length).toBe(2);
arrayEquals(result, [{End: {X: 2, Y: 1}, Direction: Direction.left, Path: [{X: 3, Y: 2}, {X: 3, Y: 1}]}, arrayEquals(result, [{end: {x: 2, y: 1}, direction: Direction.left, path: [{x: 3, y: 2}, {x: 3, y: 1}]},
{End: {X: 4, Y: 1}, Direction: Direction.right, Path: [{X: 3, Y: 2}, {X: 3, Y: 1}]}]); {end: {x: 4, y: 1}, direction: Direction.right, path: [{x: 3, y: 2}, {x: 3, y: 1}]}]);
}); });
test("Pac-Man rolls four from start, should return two positions", () => { test("Pac-Man rolls four from start, should return two positions", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 4, []); const result = possibleMovesAlgorithm(testMap, pacMan, 4, []);
expect(result.length).toBe(2); expect(result.length).toBe(2);
arrayEquals(result, [{ arrayEquals(result, [{
End: {X: 1, Y: 1}, end: {x: 1, y: 1},
Direction: Direction.left, direction: Direction.left,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 2, Y: 1}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}]
}, { }, {
End: {X: 5, Y: 1}, end: {x: 5, y: 1},
Direction: Direction.right, direction: Direction.right,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}]
}]); }]);
}); });
@ -56,21 +69,21 @@ test("Pac-Man rolls five from start, should return four positions", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 5, []); const result = possibleMovesAlgorithm(testMap, pacMan, 5, []);
expect(result.length).toBe(4); expect(result.length).toBe(4);
arrayEquals(result, [{ arrayEquals(result, [{
End: {X: 5, Y: 0}, end: {x: 5, y: 0},
Direction: Direction.up, direction: Direction.up,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}]
}, { }, {
End: {X: 6, Y: 1}, end: {x: 6, y: 1},
Direction: Direction.right, direction: Direction.right,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}]
}, { }, {
End: {X: 1, Y: 2}, end: {x: 1, y: 2},
Direction: Direction.down, direction: Direction.down,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 2, Y: 1}, {X: 1, Y: 1}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}, {x: 1, y: 1}]
}, { }, {
End: {X: 5, Y: 2}, end: {x: 5, y: 2},
Direction: Direction.down, direction: Direction.down,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}]
} }
]); ]);
}); });
@ -80,108 +93,108 @@ test("Pac-Man rolls six from start, should return six positions", () => {
expect(result.length).toBe(6); expect(result.length).toBe(6);
arrayEquals(result, [ arrayEquals(result, [
{ {
End: {X: 1, Y: 3}, end: {x: 1, y: 3},
Direction: Direction.down, direction: Direction.down,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 2, Y: 1}, {X: 1, Y: 1}, {X: 1, Y: 2}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 2, y: 1}, {x: 1, y: 1}, {x: 1, y: 2}]
}, { }, {
End: {X: 0, Y: 5}, end: {x: 0, y: 5},
Direction: Direction.right, direction: Direction.right,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 0}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 0}]
}, { }, {
End: {X: 5, Y: 3}, end: {x: 5, y: 3},
Direction: Direction.down, direction: Direction.down,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 2}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 2}]
}, { }, {
End: {X: 7, Y: 1}, end: {x: 7, y: 1},
Direction: Direction.right, direction: Direction.right,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 6, Y: 1}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 6, y: 1}]
}, { }, {
End: {X: 10, Y: 5}, end: {x: 10, y: 5},
Direction: Direction.left, direction: Direction.left,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 0}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 0}]
}, { }, {
End: {X: 5, Y: 10}, end: {x: 5, y: 10},
Direction: Direction.up, direction: Direction.up,
Path: [{X: 3, Y: 2}, {X: 3, Y: 1}, {X: 4, Y: 1}, {X: 5, Y: 1}, {X: 5, Y: 0}] path: [{x: 3, y: 2}, {x: 3, y: 1}, {x: 4, y: 1}, {x: 5, y: 1}, {x: 5, y: 0}]
} }
]); ]);
}); });
test("Pac-Man rolls four from position [5,1] (right), should return 11", () => { test("Pac-Man rolls four from position [5,1] (right), should return 11", () => {
pacMan.follow({End: {X: 5, Y: 1}, Direction: Direction.right}); pacMan.follow({end: {x: 5, y: 1}, direction: Direction.right});
const result = possibleMovesAlgorithm(testMap, pacMan, 4, []); const result = possibleMovesAlgorithm(testMap, pacMan, 4, []);
expect(result.length).toBe(11); expect(result.length).toBe(11);
}); });
test("Pac-Man rolls four from position [5,1] (left), should return 12", () => { test("Pac-Man rolls four from position [5,1] (left), should return 12", () => {
pacMan.follow({End: {X: 5, Y: 1}, Direction: Direction.left}); pacMan.follow({end: {x: 5, y: 1}, direction: Direction.left});
const result = possibleMovesAlgorithm(testMap, pacMan, 4, []); const result = possibleMovesAlgorithm(testMap, pacMan, 4, []);
expect(result.length).toBe(12); expect(result.length).toBe(12);
}); });
test("Pac-Man rolls three from position [1,5] (left), should return 5", () => { test("Pac-Man rolls three from position [1,5] (left), should return 5", () => {
pacMan.follow({End: {X: 1, Y: 5}, Direction: Direction.left}); pacMan.follow({end: {x: 1, y: 5}, direction: Direction.left});
const result = possibleMovesAlgorithm(testMap, pacMan, 3, []); const result = possibleMovesAlgorithm(testMap, pacMan, 3, []);
arrayEquals(result, [ arrayEquals(result, [
{End: {X: 1, Y: 2}, Direction: Direction.up, Path: [{X: 1, Y: 4}, {X: 1, Y: 3}]}, {end: {x: 1, y: 2}, direction: Direction.up, path: [{x: 1, y: 4}, {x: 1, y: 3}]},
{End: {X: 1, Y: 8}, Direction: Direction.down, Path: [{X: 1, Y: 6}, {X: 1, Y: 7}]}, {end: {x: 1, y: 8}, direction: Direction.down, path: [{x: 1, y: 6}, {x: 1, y: 7}]},
{End: {X: 5, Y: 1}, Direction: Direction.down, Path: [{X: 0, Y: 5}, {X: 5, Y: 0}]}, {end: {x: 5, y: 1}, direction: Direction.down, path: [{x: 0, y: 5}, {x: 5, y: 0}]},
{End: {X: 9, Y: 5}, Direction: Direction.left, Path: [{X: 0, Y: 5}, {X: 10, Y: 5}]}, {end: {x: 9, y: 5}, direction: Direction.left, path: [{x: 0, y: 5}, {x: 10, y: 5}]},
{End: {X: 5, Y: 9}, Direction: Direction.up, Path: [{X: 0, Y: 5}, {X: 5, Y: 10}]}, {end: {x: 5, y: 9}, direction: Direction.up, path: [{x: 0, y: 5}, {x: 5, y: 10}]},
]); ]);
expect(result.length).toBe(5); expect(result.length).toBe(5);
}); });
test("Pac-Man rolls six from position [1,5] (down), should return 17", () => { test("Pac-Man rolls six from position [1,5] (down), should return 17", () => {
pacMan.follow({End: {X: 1, Y: 5}, Direction: Direction.down}); pacMan.follow({end: {x: 1, y: 5}, direction: Direction.down});
const result = possibleMovesAlgorithm(testMap, pacMan, 6, []); const result = possibleMovesAlgorithm(testMap, pacMan, 6, []);
expect(result.length).toBe(21); expect(result.length).toBe(21);
}); });
test("Pac-Man rolls six from position [7,1] (right), path to [9,5] should be five tiles long", () => { test("Pac-Man rolls six from position [7,1] (right), path to [9,5] should be five tiles long", () => {
pacMan.follow({End: {X: 7, Y: 1}, Direction: Direction.right}); pacMan.follow({end: {x: 7, y: 1}, direction: Direction.right});
const result = possibleMovesAlgorithm(testMap, pacMan, 6, []); const result = possibleMovesAlgorithm(testMap, pacMan, 6, []);
expect(result[0].Path?.length).toBe(5); expect(result[0].path?.length).toBe(5);
}); });
test("Pac-Man rolls 5 from position [9,3] (down), should return 7", () => { test("Pac-Man rolls 5 from position [9,3] (down), should return 7", () => {
pacMan.follow({End: {X: 9, Y: 3}, Direction: Direction.down}); pacMan.follow({end: {x: 9, y: 3}, direction: Direction.down});
const result = possibleMovesAlgorithm(testMap, pacMan, 5, []); const result = possibleMovesAlgorithm(testMap, pacMan, 5, []);
expect(result.length).toBe(7); expect(result.length).toBe(7);
}); });
test("Ghost can take Pac-Man, stops exactly on Pac-Man unless Pac-Man is at spawn", () => { test("Ghost can take Pac-Man, stops exactly on Pac-Man unless Pac-Man is at spawn", () => {
ghost.follow({End: {X: 3, Y: 5}, Direction: Direction.up}); ghost.follow({end: {x: 3, y: 5}, direction: Direction.up});
const result = possibleMovesAlgorithm(testMap, ghost, 2, [ghost, pacMan]); const result = possibleMovesAlgorithm(testMap, ghost, 2, [ghost, pacMan]);
expect(result.length).toBe(2); expect(result.length).toBe(2);
arrayEquals(result, [ arrayEquals(result, [
{End: {X: 1, Y: 5}, Direction: Direction.left, Path: [{X: 2, Y: 5}]}, {end: {x: 1, y: 5}, direction: Direction.left, path: [{x: 2, y: 5}]},
{End: {X: 5, Y: 5}, Direction: Direction.right, Path: [{X: 4, Y: 5}]}, {end: {x: 5, y: 5}, direction: Direction.right, path: [{x: 4, y: 5}]},
]) ])
}); });
test("Ghost can take Pac-Man, steps reach Pac-Man exactly", () => { test("Ghost can take Pac-Man, steps reach Pac-Man exactly", () => {
ghost.follow({End: {X: 7, Y: 3}, Direction: Direction.up}); ghost.follow({end: {x: 7, y: 3}, direction: Direction.up});
pacMan.follow({End: {X: 5, Y: 1}, Direction: Direction.right}); pacMan.follow({end: {x: 5, y: 1}, direction: Direction.right});
const result = possibleMovesAlgorithm(testMap, ghost, 4, [ghost, pacMan]); const result = possibleMovesAlgorithm(testMap, ghost, 4, [ghost, pacMan]);
expect(result.length).toBe(2); expect(result.length).toBe(2);
arrayEquals(result, [ arrayEquals(result, [
{End: {X: 5, Y: 1}, Direction: Direction.left, Path: [{X: 7, Y: 2}, {X: 7, Y: 1}, {X: 6, Y: 1}]}, {end: {x: 5, y: 1}, direction: Direction.left, path: [{x: 7, y: 2}, {x: 7, y: 1}, {x: 6, y: 1}]},
{End: {X: 9, Y: 1}, Direction: Direction.right, Path: [{X: 7, Y: 2}, {X: 7, Y: 1}, {X: 8, Y: 1}]}, {end: {x: 9, y: 1}, direction: Direction.right, path: [{x: 7, y: 2}, {x: 7, y: 1}, {x: 8, y: 1}]},
]) ])
}); });
test("Ghost can take Pac-Man, steps overshoot Pac-Man", () => { test("Ghost can take Pac-Man, steps overshoot Pac-Man", () => {
ghost.follow({End: {X: 7, Y: 3}, Direction: Direction.up}); ghost.follow({end: {x: 7, y: 3}, direction: Direction.up});
pacMan.follow({End: {X: 5, Y: 1}, Direction: Direction.right}); pacMan.follow({end: {x: 5, y: 1}, direction: Direction.right});
const result = possibleMovesAlgorithm(testMap, ghost, 6, [ghost, pacMan]); const result = possibleMovesAlgorithm(testMap, ghost, 6, [ghost, pacMan]);
expect(result.length).toBe(2); expect(result.length).toBe(2);
arrayEquals(result, [ arrayEquals(result, [
{End: {X: 5, Y: 1}, Direction: Direction.left, Path: [{X: 7, Y: 2}, {X: 7, Y: 1}, {X: 6, Y: 1}]}, {end: {x: 5, y: 1}, direction: Direction.left, path: [{x: 7, y: 2}, {x: 7, y: 1}, {x: 6, y: 1}]},
{ {
End: {X: 9, Y: 3}, end: {x: 9, y: 3},
Direction: Direction.down, direction: Direction.down,
Path: [{X: 7, Y: 2}, {X: 7, Y: 1}, {X: 8, Y: 1}, {X: 9, Y: 1}, {X: 9, Y: 2}] path: [{x: 7, y: 2}, {x: 7, y: 1}, {x: 8, y: 1}, {x: 9, y: 1}, {x: 9, y: 2}]
}, },
]) ])
}); });

View File

@ -1,4 +1,5 @@
using System.Text.Json; using System.Text.Json;
using System.Text.Json.Serialization;
namespace pacMan.GameStuff; namespace pacMan.GameStuff;
@ -13,8 +14,9 @@ public enum GameAction
public class ActionMessage<T> public class ActionMessage<T>
{ {
public GameAction Action { get; init; } [JsonPropertyName("action")] public GameAction Action { get; init; }
public T? Data { get; set; }
[JsonPropertyName("data")] public T? Data { get; set; }
public static ActionMessage FromJson(string json) => JsonSerializer.Deserialize<ActionMessage>(json)!; public static ActionMessage FromJson(string json) => JsonSerializer.Deserialize<ActionMessage>(json)!;
} }

View File

@ -1,12 +1,20 @@
using System.Text.Json.Serialization;
namespace pacMan.GameStuff; namespace pacMan.GameStuff;
public class Character : IEquatable<Character> public class Character : IEquatable<Character>
{ {
public required string Colour { get; init; } [JsonPropertyName("colour")] public required string Colour { get; init; }
public MovePath? Position { get; set; }
[JsonPropertyName("position")] public MovePath? Position { get; set; }
[JsonInclude]
[JsonPropertyName("isEatable")]
public bool IsEatable { get; set; } = true; public bool IsEatable { get; set; } = true;
public DirectionalPosition? SpawnPosition { get; set; }
public required CharacterType? Type { get; init; } [JsonPropertyName("spawnPosition")] public DirectionalPosition? SpawnPosition { get; set; }
[JsonPropertyName("type")] public required CharacterType? Type { get; init; }
public bool Equals(Character? other) public bool Equals(Character? other)
{ {

View File

@ -1,9 +1,12 @@
using System.Text.Json.Serialization;
namespace pacMan.GameStuff.Items; namespace pacMan.GameStuff.Items;
public class Box : IEquatable<Box> public class Box : IEquatable<Box>
{ {
public List<Pellet>? Pellets { get; init; } = new(); [JsonPropertyName("pellets")] public List<Pellet>? Pellets { get; init; } = new();
public required string Colour { get; init; }
[JsonPropertyName("colour")] public required string Colour { get; init; }
public int CountNormal => Pellets?.Count(pellet => !pellet.IsPowerPellet) ?? 0; public int CountNormal => Pellets?.Count(pellet => !pellet.IsPowerPellet) ?? 0;
@ -15,13 +18,6 @@ public class Box : IEquatable<Box>
Colour == other.Colour; Colour == other.Colour;
} }
public IEnumerator<IPellet> GetEnumerator() => Pellets?.GetEnumerator() ?? new List<Pellet>.Enumerator();
// IEnumerator IEnumerable.GetEnumerator()
// {
// return GetEnumerator();
// }
public void Add(Pellet pellet) public void Add(Pellet pellet)
{ {
Pellets?.Add(pellet); Pellets?.Add(pellet);

View File

@ -1,12 +1,11 @@
using System.Text.Json.Serialization;
namespace pacMan.GameStuff.Items; namespace pacMan.GameStuff.Items;
public interface IPellet public class Pellet : IEquatable<Pellet>
{ {
bool IsPowerPellet { get; init; } [JsonPropertyName("isPowerPellet")] public bool IsPowerPellet { get; init; }
}
public class Pellet : IPellet, IEquatable<Pellet>
{
public bool Equals(Pellet? other) public bool Equals(Pellet? other)
{ {
if (ReferenceEquals(null, other)) return false; if (ReferenceEquals(null, other)) return false;
@ -14,8 +13,6 @@ public class Pellet : IPellet, IEquatable<Pellet>
return IsPowerPellet == other.IsPowerPellet; return IsPowerPellet == other.IsPowerPellet;
} }
public bool IsPowerPellet { get; init; }
public override bool Equals(object? obj) public override bool Equals(object? obj)
{ {
if (ReferenceEquals(null, obj)) return false; if (ReferenceEquals(null, obj)) return false;

View File

@ -1,16 +1,8 @@
using System.Text.Json.Serialization;
using DAL.Database.Models; using DAL.Database.Models;
namespace pacMan.GameStuff.Items; namespace pacMan.GameStuff.Items;
public interface IPlayer
{
string Username { get; init; }
Character PacMan { get; init; }
string Colour { get; init; }
Box? Box { get; init; }
State State { get; set; }
}
public enum State public enum State
{ {
WaitingForPlayers, WaitingForPlayers,
@ -19,8 +11,18 @@ public enum State
Disconnected Disconnected
} }
public class Player : IPlayer, IEquatable<Player> public class Player : IEquatable<Player>
{ {
[JsonPropertyName("username")] public required string Username { get; init; }
[JsonPropertyName("pacMan")] public required Character PacMan { get; init; }
[JsonPropertyName("colour")] public required string Colour { get; init; }
[JsonPropertyName("box")] public Box? Box { get; init; }
[JsonPropertyName("state")] public State State { get; set; } = State.WaitingForPlayers;
public bool Equals(Player? other) public bool Equals(Player? other)
{ {
if (ReferenceEquals(null, other)) return false; if (ReferenceEquals(null, other)) return false;
@ -28,13 +30,6 @@ public class Player : IPlayer, IEquatable<Player>
return Username == other.Username; return Username == other.Username;
} }
// [JsonPropertyName("username")]
public required string Username { get; init; }
public required Character PacMan { get; init; }
public required string Colour { get; init; }
public Box? Box { get; init; }
public State State { get; set; } = State.WaitingForPlayers;
public override bool Equals(object? obj) public override bool Equals(object? obj)
{ {
if (ReferenceEquals(null, obj)) return false; if (ReferenceEquals(null, obj)) return false;

View File

@ -1,10 +1,16 @@
using System.Text.Json.Serialization;
namespace pacMan.GameStuff; namespace pacMan.GameStuff;
public class MovePath : IEquatable<MovePath> public class MovePath : IEquatable<MovePath>
{ {
[JsonInclude]
[JsonPropertyName("path")]
public Position[]? Path { get; set; } public Position[]? Path { get; set; }
public required Position End { get; init; }
public required Direction Direction { get; init; } [JsonPropertyName("end")] public required Position End { get; init; }
[JsonPropertyName("direction")] public required Direction Direction { get; init; }
public bool Equals(MovePath? other) public bool Equals(MovePath? other)
{ {
@ -32,8 +38,9 @@ public class MovePath : IEquatable<MovePath>
public class Position : IEquatable<Position> public class Position : IEquatable<Position>
{ {
public int X { get; init; } [JsonPropertyName("x")] public int X { get; init; }
public int Y { get; init; }
[JsonPropertyName("y")] public int Y { get; init; }
public bool Equals(Position? other) public bool Equals(Position? other)
{ {
@ -62,8 +69,9 @@ public enum Direction
public class DirectionalPosition : IEquatable<DirectionalPosition> public class DirectionalPosition : IEquatable<DirectionalPosition>
{ {
public required Position At { get; init; } [JsonPropertyName("at")] public required Position At { get; init; }
public required Direction Direction { get; init; }
[JsonPropertyName("direction")] public required Direction Direction { get; init; }
public bool Equals(DirectionalPosition? other) public bool Equals(DirectionalPosition? other)
{ {

View File

@ -7,11 +7,11 @@ namespace pacMan.Services;
public interface IActionService public interface IActionService
{ {
IPlayer Player { set; } Player Player { set; }
Game Game { set; } Game Game { set; }
void DoAction(ActionMessage message); void DoAction(ActionMessage message);
List<int> RollDice(); List<int> RollDice();
List<IPlayer> SetPlayerInfo(JsonElement? jsonElement); List<Player> SetPlayerInfo(JsonElement? jsonElement);
object? HandleMoveCharacter(JsonElement? jsonElement); // TODO test object? HandleMoveCharacter(JsonElement? jsonElement); // TODO test
object Ready(); object Ready();
string FindNextPlayer(); string FindNextPlayer();
@ -31,7 +31,7 @@ public class ActionService : IActionService
public Game? Game { get; set; } public Game? Game { get; set; }
public IPlayer? Player { get; set; } public Player? Player { get; set; }
public void DoAction(ActionMessage message) public void DoAction(ActionMessage message)
{ {
@ -55,13 +55,13 @@ public class ActionService : IActionService
return rolls; return rolls;
} }
public List<IPlayer> SetPlayerInfo(JsonElement? jsonElement) public List<Player> SetPlayerInfo(JsonElement? jsonElement)
{ {
var data = jsonElement?.Deserialize<PlayerInfoData>() ?? throw new NullReferenceException("Data is null"); var data = jsonElement?.Deserialize<PlayerInfoData>() ?? throw new NullReferenceException("Data is null");
Player = data.Player; Player = data.Player;
Game? group; Game? group;
IPlayer? player; Player? player;
if ((group = _gameService.FindGameByUsername(Player.Username)) != null && if ((group = _gameService.FindGameByUsername(Player.Username)) != null &&
(player = group.Players.Find(p => p.Username == Player.Username))?.State == State.Disconnected) (player = group.Players.Find(p => p.Username == Player.Username))?.State == State.Disconnected)
{ {
@ -108,7 +108,7 @@ public class ActionService : IActionService
public object? HandleMoveCharacter(JsonElement? jsonElement) public object? HandleMoveCharacter(JsonElement? jsonElement)
{ {
if (Game != null && jsonElement.HasValue) if (Game != null && jsonElement.HasValue)
Game.Ghosts = jsonElement.Value.GetProperty("Ghosts").Deserialize<List<Character>>() ?? Game.Ghosts = jsonElement.Value.GetProperty("ghosts").Deserialize<List<Character>>() ??
throw new JsonException("Ghosts is null"); throw new JsonException("Ghosts is null");
return jsonElement; return jsonElement;
@ -117,12 +117,22 @@ public class ActionService : IActionService
public struct PlayerInfoData public struct PlayerInfoData
{ {
[JsonInclude] public required Player Player { get; init; } [JsonInclude]
[JsonInclude] public required Queue<DirectionalPosition> Spawns { get; init; } [JsonPropertyName("player")]
public required Player Player { get; init; }
[JsonInclude]
[JsonPropertyName("spawns")]
public required Queue<DirectionalPosition> Spawns { get; init; }
} }
public struct ReadyData public struct ReadyData
{ {
[JsonInclude] public required bool AllReady { get; init; } [JsonInclude]
[JsonInclude] public required IEnumerable<IPlayer> Players { get; set; } [JsonPropertyName("allReady")]
public required bool AllReady { get; init; }
[JsonInclude]
[JsonPropertyName("players")]
public required IEnumerable<Player> Players { get; set; }
} }

View File

@ -14,7 +14,7 @@ public class Game // TODO handle disconnects and reconnects
[JsonInclude] public Guid Id { get; } = Guid.NewGuid(); [JsonInclude] public Guid Id { get; } = Guid.NewGuid();
[JsonIgnore] public List<IPlayer> Players { get; } = new(); [JsonIgnore] public List<Player> Players { get; } = new();
[JsonIgnore] public List<Character> Ghosts { get; set; } = new(); [JsonIgnore] public List<Character> Ghosts { get; set; } = new();
@ -27,7 +27,7 @@ public class Game // TODO handle disconnects and reconnects
[JsonInclude] [JsonInclude]
public bool IsGameStarted => Count > 0 && Players.All(player => player.State is State.InGame or State.Disconnected); public bool IsGameStarted => Count > 0 && Players.All(player => player.State is State.InGame or State.Disconnected);
public IPlayer NextPlayer() public Player NextPlayer()
{ {
try try
{ {
@ -45,7 +45,7 @@ public class Game // TODO handle disconnects and reconnects
public event Func<ArraySegment<byte>, Task>? Connections; public event Func<ArraySegment<byte>, Task>? Connections;
public bool AddPlayer(IPlayer player) public bool AddPlayer(Player player)
{ {
if (Players.Count >= Rules.MaxPlayers || IsGameStarted) return false; if (Players.Count >= Rules.MaxPlayers || IsGameStarted) return false;
/* TODO remove above and uncomment below /* TODO remove above and uncomment below
@ -62,7 +62,7 @@ public class Game // TODO handle disconnects and reconnects
return true; return true;
} }
public IPlayer? RemovePlayer(string username) public Player? RemovePlayer(string username)
{ {
var index = Players.FindIndex(p => p.Username == username); var index = Players.FindIndex(p => p.Username == username);
if (index == -1) return null; if (index == -1) return null;
@ -71,7 +71,7 @@ public class Game // TODO handle disconnects and reconnects
return removedPlayer; return removedPlayer;
} }
private void SetSpawn(IPlayer player) private void SetSpawn(Player player)
{ {
if (player.PacMan.SpawnPosition is not null) return; if (player.PacMan.SpawnPosition is not null) return;
var spawn = Spawns.Dequeue(); var spawn = Spawns.Dequeue();
@ -81,7 +81,7 @@ public class Game // TODO handle disconnects and reconnects
public void SendToAll(ArraySegment<byte> segment) => Connections?.Invoke(segment); public void SendToAll(ArraySegment<byte> segment) => Connections?.Invoke(segment);
public IEnumerable<IPlayer> SetReady(IPlayer player) public IEnumerable<Player> SetReady(Player player)
{ {
if (!Players.Contains(player)) 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.");

View File

@ -26,7 +26,7 @@ public class GameService : WebSocketService
Connections?.Invoke(segment); Connections?.Invoke(segment);
} }
public Game AddPlayer(IPlayer player, Queue<DirectionalPosition> spawns) public Game AddPlayer(Player player, Queue<DirectionalPosition> spawns)
{ {
var index = 0; var index = 0;
try try
@ -50,7 +50,7 @@ public class GameService : WebSocketService
/// <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>
/// <returns>Returns the updated Game object after adding the player.</returns> /// <returns>Returns the updated Game object after adding the player.</returns>
/// <exception cref="GameNotFoundException">Thrown if a game with the specified id cannot be found.</exception> /// <exception cref="GameNotFoundException">Thrown if a game with the specified id cannot be found.</exception>
public Game JoinById(Guid id, IPlayer player) public Game JoinById(Guid id, Player player)
{ {
var game = Games.FirstOrDefault(g => g.Id == id) ?? throw new GameNotFoundException(); var game = Games.FirstOrDefault(g => g.Id == id) ?? throw new GameNotFoundException();
game.AddPlayer(player); game.AddPlayer(player);
@ -69,7 +69,7 @@ public class GameService : WebSocketService
/// Thrown if the number of spawns is not equal to the maximum number of players set by /// Thrown if the number of spawns is not equal to the maximum number of players set by
/// the Rules. /// the Rules.
/// </exception> /// </exception>
public Game CreateAndJoin(IPlayer player, Queue<DirectionalPosition> spawns) public Game CreateAndJoin(Player player, Queue<DirectionalPosition> spawns)
{ {
if (spawns.Count != Rules.MaxPlayers) if (spawns.Count != Rules.MaxPlayers)
throw new ArgumentException($"The number of spawns must be equal to {Rules.MaxPlayers}."); throw new ArgumentException($"The number of spawns must be equal to {Rules.MaxPlayers}.");