Json is being handled with camelCase
This commit is contained in:
parent
0c9ba333ea
commit
1db515d796
@ -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));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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,
|
||||||
|
@ -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();
|
||||||
|
@ -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
|
||||||
|
@ -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}/>
|
||||||
|
@ -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`}/>
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
@ -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}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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));
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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");
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
}
|
}
|
||||||
|
@ -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[],
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
|
@ -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.
|
||||||
|
@ -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}]
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
});
|
});
|
||||||
|
@ -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)!;
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
{
|
{
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
|
@ -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)
|
||||||
{
|
{
|
||||||
|
@ -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; }
|
||||||
}
|
}
|
||||||
|
@ -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.");
|
||||||
|
@ -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}.");
|
||||||
|
Loading…
x
Reference in New Issue
Block a user