Sets the spawn in the backend when player is created, and other fixes :p

This commit is contained in:
Martin Berg Alstad 2023-07-14 19:14:19 +02:00
parent 63502405e1
commit 47196161ac
22 changed files with 276 additions and 124 deletions

View File

@ -11,27 +11,49 @@ namespace BackendTests.Services;
public class ActionServiceTests
{
private readonly IPlayer _blackPlayer = Players.Create("black");
private readonly IPlayer _redPlayer = Players.Create("red");
private readonly IPlayer _whitePlayer = Players.Create("white");
private readonly Player _blackPlayer = (Player)Players.Create("black");
private readonly Player _redPlayer = (Player)Players.Create("red");
private readonly Player _whitePlayer = (Player)Players.Create("white");
private ActionMessage _blackMessage = null!;
private ActionMessage _redMessage = null!;
private IActionService _service = null!;
private Queue<DirectionalPosition> _spawns = null!;
private ActionMessage _whiteMessage = null!;
private IWebSocketService _wssSub = null!;
[SetUp]
public void Setup()
{
_spawns = CreateQueue();
_whiteMessage = new ActionMessage
{ Action = GameAction.PlayerInfo, Data = JsonSerializer.Serialize(_whitePlayer) };
{
Action = GameAction.PlayerInfo,
Data = JsonSerializer.Serialize(new { Player = _whitePlayer, Spawns = CreateQueue() })
};
_blackMessage = new ActionMessage
{ Action = GameAction.PlayerInfo, Data = JsonSerializer.Serialize(_blackPlayer) };
_redMessage = new ActionMessage { Action = GameAction.PlayerInfo, Data = JsonSerializer.Serialize(_redPlayer) };
{
Action = GameAction.PlayerInfo,
Data = JsonSerializer.Serialize(new { Player = _blackPlayer, Spawns = CreateQueue() })
};
_redMessage = new ActionMessage
{
Action = GameAction.PlayerInfo,
Data = JsonSerializer.Serialize(new { Player = _redPlayer, Spawns = CreateQueue() })
};
_wssSub = Substitute.For<WebSocketService>(Substitute.For<ILogger<WebSocketService>>());
_service = new ActionService(Substitute.For<ILogger<ActionService>>(), _wssSub);
}
private static Queue<DirectionalPosition> CreateQueue() =>
new(new[]
{
new DirectionalPosition { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up },
new() { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down }
});
#region RollDice()
[Test]
@ -74,6 +96,9 @@ public class ActionServiceTests
{
var players = _service.SetPlayerInfo(_whiteMessage);
var pos = _spawns.Dequeue();
_whitePlayer.PacMan.Position = pos;
_whitePlayer.PacMan.SpawnPosition = pos;
Assert.That(new List<IPlayer> { _whitePlayer }, Is.EqualTo(players));
}
@ -140,7 +165,7 @@ public class ActionServiceTests
[Test]
public void Ready_TwoReady()
{
var group = new GameGroup { Players = { _blackPlayer, _whitePlayer } };
var group = new GameGroup(new Queue<DirectionalPosition>()) { Players = { _blackPlayer, _whitePlayer } };
_service.Group = group;
_service.Player = _blackPlayer;

View File

@ -9,17 +9,34 @@ namespace BackendTests.Services;
public class GameGroupTests
{
private readonly DirectionalPosition _spawn3By3Up = new()
{ At = new Position { X = 3, Y = 3 }, Direction = Direction.Up };
private readonly DirectionalPosition _spawn7By7Down = new()
{ At = new Position { X = 7, Y = 7 }, Direction = Direction.Down };
private readonly DirectionalPosition _spawn7By7Left = new()
{ At = new Position { X = 7, Y = 7 }, Direction = Direction.Left };
private readonly DirectionalPosition _spawn7By7Right = new()
{ At = new Position { X = 7, Y = 7 }, Direction = Direction.Right };
private IPlayer _bluePlayer = null!;
private GameGroup _gameGroup = null!;
private IPlayer _greenPlayer = null!;
private IPlayer _purplePlayer = null!;
private IPlayer _redPlayer = null!;
private Queue<DirectionalPosition> _spawns = null!;
private IPlayer _yellowPlayer = null!;
[SetUp]
public void Setup()
{
_gameGroup = new GameGroup();
_spawns = new Queue<DirectionalPosition>(
new[] { _spawn3By3Up, _spawn7By7Left, _spawn7By7Down, _spawn7By7Right });
_gameGroup = new GameGroup(_spawns);
_redPlayer = Players.Create("red");
_bluePlayer = Players.Create("blue");
_yellowPlayer = Players.Create("yellow");
@ -70,6 +87,14 @@ public class GameGroupTests
Assert.That(_redPlayer.State, Is.EqualTo(State.WaitingForPlayers));
}
[Test]
public void AddPlayer_AddSpawnPosition()
{
_gameGroup.AddPlayer(_redPlayer);
Assert.That(_redPlayer.PacMan.SpawnPosition, Is.Not.Null);
Assert.That(_redPlayer.PacMan.SpawnPosition, Is.EqualTo(_spawn3By3Up));
}
#endregion
#region Sendtoall(ArraySegment<byte> segment)

View File

@ -11,12 +11,25 @@ namespace BackendTests.Services;
public class WebSocketServiceTests
{
private readonly DirectionalPosition _spawn3By3Up = new()
{ At = new Position { X = 3, Y = 3 }, Direction = Direction.Up };
private IWebSocketService _service = null!;
private Queue<DirectionalPosition> _spawns = null!;
[SetUp]
public void SetUp()
{
_service = new WebSocketService(Substitute.For<ILogger<WebSocketService>>());
_spawns = new Queue<DirectionalPosition>(new[]
{
_spawn3By3Up,
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down },
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down },
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down }
});
}
#region Send(Websocket, ArraySegment<byte>)
@ -135,7 +148,7 @@ public class WebSocketServiceTests
public void AddPlayer_ToEmptyGroup()
{
var player = Players.Create("white");
var group = _service.AddPlayer(player);
var group = _service.AddPlayer(player, _spawns);
Assert.Multiple(() =>
{
@ -151,12 +164,12 @@ public class WebSocketServiceTests
for (var i = 0; i < 4; i++)
{
var player = Players.Create(i.ToString());
_service.AddPlayer(player);
_service.AddPlayer(player, _spawns);
}
var player5 = Players.Create("white");
var group = _service.AddPlayer(player5);
var group = _service.AddPlayer(player5, new Queue<DirectionalPosition>(new[] { _spawn3By3Up }));
Assert.Multiple(() =>
{

View File

@ -4,10 +4,9 @@ import findPossiblePositions from "../game/possibleMovesAlgorithm";
import {GameTile} from "./gameTile";
import {TileType} from "../game/tileType";
import {useAtomValue} from "jotai";
import {allCharactersAtom} from "../utils/state";
import {allCharactersAtom, selectedDiceAtom} from "../utils/state";
interface BoardProps extends ComponentProps {
selectedDice?: SelectedDice,
onMove?: Action<Position[]>,
map: GameMap
}
@ -15,12 +14,12 @@ interface BoardProps extends ComponentProps {
const Board: Component<BoardProps> = (
{
className,
selectedDice,
onMove,
map
}) => {
const characters = useAtomValue(allCharactersAtom);
const selectedDice = useAtomValue(selectedDiceAtom);
const [selectedCharacter, setSelectedCharacter] = useState<Character>();
const [possiblePositions, setPossiblePositions] = useState<Path[]>([]); // TODO reset when other client moves a character
const [hoveredPosition, setHoveredPosition] = useState<Path>();
@ -63,15 +62,15 @@ const Board: Component<BoardProps> = (
const pacMan = selectedCharacter as PacMan;
for (const tile of [...destination.Path ?? [], destination.End]) {
const currentTile = map[tile.y][tile.x];
const currentTile = map[tile.Y][tile.X];
if (currentTile === TileType.pellet) {
// pacMan.box.addPellet(new Pellet()); // TODO update to current player
map[tile.y][tile.x] = TileType.empty;
map[tile.Y][tile.X] = TileType.empty;
positions.push(tile);
} else if (currentTile === TileType.powerPellet) {
// pacMan.box.addPellet(new Pellet(true));
map[tile.y][tile.x] = TileType.empty;
map[tile.Y][tile.X] = TileType.empty;
positions.push(tile);
}
}
@ -98,9 +97,9 @@ const Board: Component<BoardProps> = (
<GameTile
key={colIndex + rowIndex * colIndex}
type={tile}
possiblePath={possiblePositions.find(p => p.End.x === colIndex && p.End.y === rowIndex)}
character={characters.find(c => c.isAt({x: colIndex, y: rowIndex}))}
isSelected={selectedCharacter?.isAt({x: colIndex, y: rowIndex})}
possiblePath={possiblePositions.find(p => p.End.X === colIndex && p.End.Y === rowIndex)}
character={characters.find(c => c.isAt({X: colIndex, Y: rowIndex}))}
isSelected={selectedCharacter?.isAt({X: colIndex, Y: rowIndex})}
showPath={hoveredPosition?.Path?.find(pos => pos.x === colIndex && pos.y === rowIndex) !== undefined}
handleMoveCharacter={handleMoveCharacter}
handleSelectCharacter={handleSelectCharacter}

View File

@ -3,11 +3,12 @@ import {AllDice} from "./dice";
import {doAction, GameAction} from "../utils/actions";
import GameBoard from "./gameBoard";
import WebSocketService from "../websockets/WebSocketService";
import {testMap} from "../game/map";
import {getCharacterSpawns, testMap} from "../game/map";
import Player, {State} from "../game/player";
import PlayerStats from "../components/playerStats";
import {getDefaultStore, useAtom, useAtomValue} from "jotai";
import {currentPlayerAtom, diceAtom, ghostsAtom, playersAtom, selectedDiceAtom} from "../utils/state";
import {CharacterType} from "../game/character";
const wsService = new WebSocketService(import.meta.env.VITE_API);
@ -43,7 +44,19 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => {
}
async function sendPlayer(): Promise<void> {
wsService.send({Action: GameAction.playerInfo, Data: player});
// TODO set spawn position and position
wsService.send({
Action: GameAction.playerInfo,
Data: {
Player: player, Spawns: getCharacterSpawns(testMap)
.filter(s => s.type === CharacterType.pacMan)
.map(s => s.position)
}
});
}
function sendReady(): void {
wsService.send({Action: GameAction.ready});
}
useEffect(() => {
@ -55,10 +68,6 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => {
return () => wsService.close();
}, []);
function sendReady(): void {
wsService.send({Action: GameAction.ready});
}
return (
<>
<div className={"flex-center"}>
@ -70,9 +79,7 @@ export const GameComponent: Component<{ player: Player }> = ({player}) => {
</div>
<AllDice values={dice} selectedDiceIndex={selectedDice?.index}/>
{players?.map(p => <PlayerStats key={p.Name} player={p} isCurrentPlayer={currentPlayer?.Name === p.Name}/>)}
<GameBoard className={"mx-auto my-2"}
selectedDice={selectedDice}
onMove={onCharacterMove} map={testMap}/>
<GameBoard className={"mx-auto my-2"} onMove={onCharacterMove} map={testMap}/>
</>
);
};

View File

@ -3,6 +3,7 @@ import {TileType} from "../game/tileType";
import {Character, Dummy} from "../game/character";
import {Direction} from "../game/direction";
import {getCSSColour} from "../utils/colours";
import {Colour} from "../game/colour";
interface TileWithCharacterProps extends ComponentProps {
possiblePath?: Path,
@ -113,8 +114,8 @@ const Tile: Component<TileProps> = (
onMouseEnter={onMouseEnter}
onMouseLeave={onMouseLeave}>
{children}
{type === TileType.pellet && <Circle colour={"yellow"}/>}
{type === TileType.powerPellet && <Circle colour={"red"}/>}
{type === TileType.pellet && <Circle colour={Colour.Yellow}/>}
{type === TileType.powerPellet && <Circle colour={Colour.Red}/>}
</div>
);
};
@ -146,7 +147,7 @@ const CharacterComponent: Component<CharacterComponentProps> = (
}) => {
function getSide() {
switch (character?.Position.Direction) {
switch (character?.Position?.Direction) {
case Direction.up:
return "right-1/4 top-0";
case Direction.down:

View File

@ -57,7 +57,7 @@ export class Character {
}
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;
}
}
@ -83,7 +83,7 @@ export class Dummy extends Character {
Colour: Colour.Grey,
Position: position,
IsEatable: false,
SpawnPosition: {At: {x: 0, y: 0}, Direction: Direction.up},
SpawnPosition: {At: {X: 0, Y: 0}, Direction: Direction.up},
Type: CharacterType.dummy,
});
}

View File

@ -30,10 +30,10 @@ export function getCharacterSpawns(map: GameMap): { type: CharacterType, positio
for (let col = 0; col < map.length; col++) {
// TODO find direction
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) {
result.push({
type: CharacterType.pacMan, position: {At: {x: col, y: row}, Direction: Direction.up}
type: CharacterType.pacMan, position: {At: {X: col, Y: row}, Direction: Direction.up}
});
}
}

View File

@ -108,23 +108,23 @@ function tryMove(board: GameMap, path: Path, direction: Direction, steps: number
switch (direction) {
case Direction.left:
return {
x: path.End.x - 1,
y: path.End.y
X: path.End.X - 1,
Y: path.End.Y
};
case Direction.up:
return {
x: path.End.x,
y: path.End.y - 1
X: path.End.X,
Y: path.End.Y - 1
};
case Direction.right:
return {
x: path.End.x + 1,
y: path.End.y
X: path.End.X + 1,
Y: path.End.Y
};
case Direction.down:
return {
x: path.End.x,
y: path.End.y + 1
X: path.End.X,
Y: path.End.Y + 1
};
}
}
@ -150,8 +150,8 @@ function addTeleportationTiles(board: GameMap, currentPath: Path, steps: number,
const possiblePositions = findTeleportationTiles(board);
const paths: Path[] = [];
for (const pos of possiblePositions) {
if (pos.End.x !== interval(0, board.length - 1, currentPath.End.x) ||
pos.End.y !== interval(0, board.length - 1, currentPath.End.y)) {
if (pos.End.X !== interval(0, board.length - 1, currentPath.End.X) ||
pos.End.Y !== interval(0, board.length - 1, currentPath.End.Y)) {
pos.Path = currentPath.Path;
paths.push(...findPossibleRecursive(board, pos, steps, character, characters));
@ -193,7 +193,7 @@ function findTeleportationTiles(board: GameMap): Path[] {
*/
function pushPath(board: GameMap, possiblePositions: Path[], x: number, y: number): void {
if (board[x][y] !== TileType.wall) {
possiblePositions.push({End: {x, y}, Direction: findDirection(x, y, board.length)});
possiblePositions.push({End: {X: x, Y: y}, Direction: findDirection(x, y, board.length)});
}
}
@ -224,7 +224,7 @@ function findDirection(x: number, y: number, boardSize: number): Direction {
*/
function isOutsideBoard(currentPos: Path, boardSize: number): boolean {
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;
}
/**
@ -234,7 +234,7 @@ function isOutsideBoard(currentPos: Path, boardSize: number): boolean {
*/
function isWall(board: GameMap, currentPos: Path): boolean {
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
}
/**
@ -244,7 +244,7 @@ function isWall(board: GameMap, currentPos: Path): boolean {
*/
function isSpawn(board: GameMap, currentPos: Path) {
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;
}
/**
@ -255,6 +255,6 @@ function isSpawn(board: GameMap, currentPos: Path) {
function isOwnSpawn(currentPos: Path, character: Character): boolean {
const pos = currentPos.End;
const charPos = character.SpawnPosition!.At;
return charPos.x === pos.x && charPos.y === pos.y;
return charPos.X === pos.X && charPos.Y === pos.Y;
}

View File

@ -20,7 +20,7 @@ type SelectedDice = {
index: number
};
type Position = { x: number, y: number };
type Position = { X: number, Y: number };
type GameMap = number[][];

View File

@ -89,7 +89,7 @@ function removeEatenPellets(data?: MoveCharacterData): void {
const pellets = data?.EatenPellets;
for (const pellet of pellets ?? []) {
testMap[pellet.y][pellet.x] = TileType.empty;
testMap[pellet.Y][pellet.X] = TileType.empty;
}
}

View File

@ -8,7 +8,7 @@ let pacMan: Character;
beforeEach(() => {
pacMan = new PacMan({
Colour: "yellow", SpawnPosition: {At: {x: 3, y: 3}, Direction: Direction.up}
Colour: "yellow", SpawnPosition: {At: {X: 3, Y: 3}, Direction: Direction.up}
});
});
@ -29,21 +29,21 @@ test("Pac-Man rolls two from start, should return one position", () => {
test("Pac-Man rolls three from start, should return two positions", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 3, []);
expect(result.length).toBe(2);
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}]}]);
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}]}]);
});
test("Pac-Man rolls four from start, should return two positions", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 4, []);
expect(result.length).toBe(2);
arrayEquals(result, [{
End: {x: 1, y: 1},
End: {X: 1, Y: 1},
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,
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}]
}]);
});
@ -51,21 +51,21 @@ test("Pac-Man rolls five from start, should return four positions", () => {
const result = possibleMovesAlgorithm(testMap, pacMan, 5, []);
expect(result.length).toBe(4);
arrayEquals(result, [{
End: {x: 5, y: 0},
End: {X: 5, Y: 0},
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,
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,
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,
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}]
}
]);
});
@ -75,72 +75,72 @@ test("Pac-Man rolls six from start, should return six positions", () => {
expect(result.length).toBe(6);
arrayEquals(result, [
{
End: {x: 1, y: 3},
End: {X: 1, Y: 3},
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,
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,
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,
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,
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,
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", () => {
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, []);
expect(result.length).toBe(11);
});
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, []);
expect(result.length).toBe(12);
});
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, []);
arrayEquals(result, [
{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: 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: 5, y: 9}, Direction: Direction.up, Path: [{x: 0, y: 5}, {x: 5, y: 10}]},
{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: 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: 5, Y: 9}, Direction: Direction.up, Path: [{X: 0, Y: 5}, {X: 5, Y: 10}]},
]);
expect(result.length).toBe(5);
});
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, []);
expect(result.length).toBe(17);
});
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, []);
expect(result[0].Path?.length).toBe(5);
});
test("Pac-Man rolls 5 from position [9,3] (down), should return 5", () => {
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, []);
expect(result.length).toBe(5);
});

View File

@ -9,7 +9,7 @@ namespace pacMan.Controllers;
[ApiController]
[Route("api/[controller]")]
public class GameController : GenericController
public class GameController : GenericController // TODO reconnect using player id
{
private readonly IActionService _actionService;

View File

@ -12,15 +12,10 @@ public enum GameAction
public class ActionMessage<T>
{
public GameAction Action { get; set; }
public GameAction Action { get; init; }
public T? Data { get; set; }
public static ActionMessage FromJson(string json)
{
return JsonSerializer.Deserialize<ActionMessage>(json)!;
}
public static ActionMessage FromJson(string json) => JsonSerializer.Deserialize<ActionMessage>(json)!;
}
public class ActionMessage : ActionMessage<dynamic>
{
}
public class ActionMessage : ActionMessage<dynamic> { }

View File

@ -2,11 +2,11 @@ namespace pacMan.Game;
public class Character : IEquatable<Character>
{
public required string Colour { get; set; }
public required string Colour { get; init; }
public MovePath? Position { get; set; }
public bool IsEatable { get; set; } = true;
public DirectionalPosition? SpawnPosition { get; set; }
public required CharacterType Type { get; set; }
public required CharacterType? Type { get; init; }
public bool Equals(Character? other)
{

View File

@ -2,7 +2,7 @@ namespace pacMan.Game.Items;
public class Box : IEquatable<Box>
{
public required List<Pellet>? Pellets { get; init; } = new();
public List<Pellet>? Pellets { get; init; } = new();
public required string Colour { get; init; }
public int CountNormal => Pellets?.Count(pellet => !pellet.IsPowerPellet) ?? 0;

View File

@ -5,7 +5,7 @@ public interface IPlayer
string Name { get; init; }
Character PacMan { get; init; }
string Colour { get; init; }
Box Box { get; init; }
Box? Box { get; init; }
State State { get; set; }
}
@ -29,7 +29,7 @@ public class Player : IPlayer, IEquatable<Player>
public required string Name { get; init; }
public required Character PacMan { get; init; }
public required string Colour { get; init; }
public required Box Box { get; init; }
public Box? Box { get; init; }
public State State { get; set; } = State.WaitingForPlayers;
public override bool Equals(object? obj)

View File

@ -1,16 +1,55 @@
namespace pacMan.Game;
public class MovePath
public class MovePath : IEquatable<MovePath>
{
public Position[]? Path { get; set; }
public required Position End { get; set; }
public required Direction Direction { get; set; }
public required Position End { get; init; }
public required Direction Direction { get; init; }
public bool Equals(MovePath? other)
{
if (ReferenceEquals(null, other)) return false;
if (ReferenceEquals(this, other)) return true;
return Equals(Path, other.Path) && End.Equals(other.End) && Direction == other.Direction;
}
public static implicit operator MovePath(DirectionalPosition path) =>
new()
{
End = path.At,
Direction = path.Direction
};
public override bool Equals(object? obj)
{
if (ReferenceEquals(null, obj)) return false;
if (ReferenceEquals(this, obj)) return true;
return obj.GetType() == GetType() && Equals((MovePath)obj);
}
public override int GetHashCode() => HashCode.Combine(Path, End, (int)Direction);
}
public class Position
public class Position : IEquatable<Position>
{
public int X { get; set; } = 0;
public int Y { get; set; } = 0;
public int X { get; init; }
public int Y { get; init; }
public bool Equals(Position? other)
{
if (ReferenceEquals(null, other)) return false;
if (ReferenceEquals(this, other)) return true;
return X == other.X && Y == other.Y;
}
public override bool Equals(object? obj)
{
if (ReferenceEquals(null, obj)) return false;
if (ReferenceEquals(this, obj)) return true;
return obj.GetType() == GetType() && Equals((Position)obj);
}
public override int GetHashCode() => HashCode.Combine(X, Y);
}
public enum Direction
@ -21,8 +60,31 @@ public enum Direction
Down
}
public class DirectionalPosition
public class DirectionalPosition : IEquatable<DirectionalPosition>
{
public required Position At { get; set; }
public required Direction Direction { get; set; }
}
public required Position At { get; init; }
public required Direction Direction { get; init; }
public bool Equals(DirectionalPosition? other)
{
if (ReferenceEquals(null, other)) return false;
if (ReferenceEquals(this, other)) return true;
return At.Equals(other.At) && Direction == other.Direction;
}
public static explicit operator DirectionalPosition(MovePath path) =>
new()
{
At = path.End,
Direction = path.Direction
};
public override bool Equals(object? obj)
{
if (ReferenceEquals(null, obj)) return false;
if (ReferenceEquals(this, obj)) return true;
return obj.GetType() == GetType() && Equals((DirectionalPosition)obj);
}
public override int GetHashCode() => HashCode.Combine(At, (int)Direction);
}

View File

@ -1,4 +1,5 @@
using System.Net.WebSockets;
using pacMan.Game;
using pacMan.Game.Items;
using pacMan.Services;
@ -13,5 +14,5 @@ public interface IWebSocketService
void SendToAll(ArraySegment<byte> segment);
Task<WebSocketReceiveResult> Receive(WebSocket webSocket, byte[] buffer);
Task Close(WebSocket webSocket, WebSocketCloseStatus closeStatus, string? closeStatusDescription);
GameGroup AddPlayer(IPlayer player);
GameGroup AddPlayer(IPlayer player, Queue<DirectionalPosition> spawns);
}

View File

@ -29,7 +29,7 @@ public class ActionService : IActionService
_wsService = wsService;
}
public GameGroup Group { get; set; } = new();
public GameGroup? Group { get; set; }
public IPlayer? Player { get; set; }
@ -56,14 +56,16 @@ public class ActionService : IActionService
{
try
{
// Receieved JsonElement from frontend
Player = JsonSerializer.Deserialize<Player>(message.Data);
Group = _wsService.AddPlayer(Player);
// Receieving JsonElement from frontend
PlayerInfoData data = JsonSerializer.Deserialize<PlayerInfoData>(message.Data);
Player = data.Player;
Group = _wsService.AddPlayer(Player, data.Spawns);
}
catch (RuntimeBinderException e)
{
Console.WriteLine(e);
if (message.Data == null) throw new NullReferenceException();
if (message.Data is null) throw new NullReferenceException();
throw;
}
@ -74,7 +76,7 @@ public class ActionService : IActionService
public object Ready()
{
object data;
if (Player != null)
if (Player != null && Group != null)
{
var players = Group.SetReady(Player).ToArray();
if (players.All(p => p.State == State.Ready))
@ -96,3 +98,9 @@ public class ActionService : IActionService
return data;
}
}
public class PlayerInfoData
{
public required Player Player { get; set; }
public required Queue<DirectionalPosition> Spawns { get; set; }
}

View File

@ -8,14 +8,20 @@ namespace pacMan.Services;
public class GameGroup : IEnumerable<IPlayer>
{
private readonly Random _random = new();
public GameGroup(Queue<DirectionalPosition> spawns) => Spawns = spawns;
public List<IPlayer> Players { get; } = new();
private Queue<DirectionalPosition> Spawns { get; }
public IPlayer RandomPlayer => Players[_random.Next(Count)];
public int Count => Players.Count;
public IEnumerator<IPlayer> GetEnumerator() => Players.GetEnumerator();
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
public event Func<ArraySegment<byte>, Task>? Connections;
public bool AddPlayer(IPlayer player) // TODO if name exists, use that player instead
@ -25,9 +31,18 @@ public class GameGroup : IEnumerable<IPlayer>
player.State = State.WaitingForPlayers;
if (Players.Exists(p => p.Name == player.Name)) return true;
Players.Add(player);
if (player.PacMan.SpawnPosition is null) SetSpawn(player);
return true;
}
private void SetSpawn(IPlayer player)
{
if (player.PacMan.SpawnPosition is not null) return;
var spawn = Spawns.Dequeue();
player.PacMan.SpawnPosition = spawn;
player.PacMan.Position = spawn;
}
public void SendToAll(ArraySegment<byte> segment) => Connections?.Invoke(segment);
public IEnumerable<IPlayer> SetReady(IPlayer player)

View File

@ -1,4 +1,5 @@
using System.Net.WebSockets;
using pacMan.Game;
using pacMan.Game.Items;
using pacMan.Interfaces;
using pacMan.Utils;
@ -56,7 +57,7 @@ public class WebSocketService : IWebSocketService
_logger.Log(LogLevel.Information, "WebSocket connection closed");
}
public GameGroup AddPlayer(IPlayer player)
public GameGroup AddPlayer(IPlayer player, Queue<DirectionalPosition> spawns)
{
var index = 0;
try
@ -65,7 +66,7 @@ public class WebSocketService : IWebSocketService
}
catch (ArgumentOutOfRangeException)
{
var game = new GameGroup();
var game = new GameGroup(spawns);
game.AddPlayer(player);
Games.Add(game);
}