Added groups in backend, refactored property names
This commit is contained in:
parent
fbe9594192
commit
c469b92739
@ -1,10 +1,8 @@
|
|||||||
import React, {useEffect, useState} from "react";
|
import React, {useEffect, useState} from "react";
|
||||||
import {Character, PacMan} from "../game/character";
|
import {Character, PacMan} from "../game/character";
|
||||||
import findPossiblePositions from "../game/possibleMovesAlgorithm";
|
import findPossiblePositions from "../game/possibleMovesAlgorithm";
|
||||||
import {Direction} from "../game/direction";
|
|
||||||
import {GameTile} from "./gameTile";
|
import {GameTile} from "./gameTile";
|
||||||
import {TileType} from "../game/tileType";
|
import {TileType} from "../game/tileType";
|
||||||
import Pellet from "../game/pellet";
|
|
||||||
|
|
||||||
interface BoardProps extends ComponentProps {
|
interface BoardProps extends ComponentProps {
|
||||||
characters: Character[],
|
characters: Character[],
|
||||||
@ -49,9 +47,9 @@ const Board: Component<BoardProps> = (
|
|||||||
setSelectedCharacter(undefined);
|
setSelectedCharacter(undefined);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
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();
|
||||||
// TODO steal from player
|
// TODO steal from player
|
||||||
@ -63,15 +61,15 @@ const Board: Component<BoardProps> = (
|
|||||||
if (selectedCharacter instanceof PacMan) {
|
if (selectedCharacter instanceof PacMan) {
|
||||||
const pacMan = selectedCharacter as PacMan;
|
const pacMan = selectedCharacter as PacMan;
|
||||||
|
|
||||||
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];
|
||||||
|
|
||||||
if (currentTile === TileType.pellet) {
|
if (currentTile === TileType.pellet) {
|
||||||
pacMan.box.addPellet(new 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);
|
positions.push(tile);
|
||||||
} else if (currentTile === TileType.powerPellet) {
|
} else if (currentTile === TileType.powerPellet) {
|
||||||
pacMan.box.addPellet(new Pellet(true));
|
// pacMan.box.addPellet(new Pellet(true));
|
||||||
map[tile.y][tile.x] = TileType.empty;
|
map[tile.y][tile.x] = TileType.empty;
|
||||||
positions.push(tile);
|
positions.push(tile);
|
||||||
}
|
}
|
||||||
@ -99,10 +97,10 @@ const Board: Component<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}
|
||||||
|
@ -11,20 +11,20 @@ import Player from "../game/player";
|
|||||||
|
|
||||||
const wsService = new WebSocketService("wss://localhost:3000/api/game");
|
const wsService = new WebSocketService("wss://localhost:3000/api/game");
|
||||||
|
|
||||||
export const GameComponent: Component<{ player: Player }> = ({player = new Player({colour: "yellow"})}) => {
|
export const GameComponent: Component<{ player: Player }> = (
|
||||||
|
{
|
||||||
|
player = new Player({
|
||||||
|
name: "Martin",
|
||||||
|
colour: "yellow",
|
||||||
|
})
|
||||||
|
}) => {
|
||||||
// TODO find spawn points
|
// TODO find spawn points
|
||||||
const [characters, setCharacters] = useState([
|
const [characters, setCharacters] = useState([
|
||||||
new PacMan({
|
new Ghost({
|
||||||
colour: "yellow", spawnPosition: {at: {x: 3, y: 3}, direction: Direction.up}
|
colour: "purple", spawnPosition: {At: {x: 7, y: 3}, Direction: Direction.up}
|
||||||
}),
|
|
||||||
new PacMan({
|
|
||||||
colour: "blue", spawnPosition: {at: {x: 7, y: 7}, direction: Direction.down}
|
|
||||||
}),
|
}),
|
||||||
new Ghost({
|
new Ghost({
|
||||||
colour: "purple", spawnPosition: {at: {x: 7, y: 3}, direction: Direction.up}
|
colour: "purple", spawnPosition: {At: {x: 3, y: 7}, Direction: Direction.down}
|
||||||
}),
|
|
||||||
new Ghost({
|
|
||||||
colour: "purple", spawnPosition: {at: {x: 3, y: 7}, direction: Direction.down}
|
|
||||||
})
|
})
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@ -63,6 +63,10 @@ export const GameComponent: Component<{ player: Player }> = ({player = new Playe
|
|||||||
updateCharacters(parsed);
|
updateCharacters(parsed);
|
||||||
removeEatenPellets(parsed);
|
removeEatenPellets(parsed);
|
||||||
break;
|
break;
|
||||||
|
case GameAction.playerInfo:
|
||||||
|
const players = parsed.Data as Player[];
|
||||||
|
// TODO set all characters
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -102,11 +106,16 @@ export const GameComponent: Component<{ player: Player }> = ({player = new Playe
|
|||||||
wsService.send(data);
|
wsService.send(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function sendPlayer(): Promise<void> {
|
||||||
|
await wsService.waitForOpen();
|
||||||
|
wsService.send({Action: GameAction.playerInfo, Data: player});
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
wsService.onReceive = doAction;
|
wsService.onReceive = doAction;
|
||||||
wsService.open();
|
wsService.open();
|
||||||
|
|
||||||
// TODO send player info to backend
|
void sendPlayer();
|
||||||
// TODO send action to backend when all players are ready
|
// TODO send action to backend when all players are ready
|
||||||
// The backend should then send the first player as current player
|
// The backend should then send the first player as current player
|
||||||
return () => wsService.close();
|
return () => wsService.close();
|
||||||
@ -121,10 +130,10 @@ export const GameComponent: Component<{ player: Player }> = ({player = new Playe
|
|||||||
<AllDice values={dice} onclick={handleDiceClick} selectedDiceIndex={selectedDice?.index}/>
|
<AllDice values={dice} onclick={handleDiceClick} selectedDiceIndex={selectedDice?.index}/>
|
||||||
{
|
{
|
||||||
(characters.filter(c => c instanceof PacMan) as PacMan[]).map(c =>
|
(characters.filter(c => c instanceof PacMan) as PacMan[]).map(c =>
|
||||||
<div key={c.colour} className={"mx-auto w-fit m-2"}>
|
<div key={c.Colour} className={"mx-auto w-fit m-2"}>
|
||||||
<p>Player: {player.colour}</p>
|
<p className={currentPlayer === player ? "underline" : ""}>Player: {player.Colour}</p>
|
||||||
<p>Pellets: {player.box.count}</p>
|
<p>Pellets: {player.Box.count}</p>
|
||||||
<p>PowerPellets: {player.box.countPowerPellets}</p>
|
<p>PowerPellets: {player.Box.countPowerPellets}</p>
|
||||||
</div>)
|
</div>)
|
||||||
}
|
}
|
||||||
<GameBoard className={"mx-auto my-2"} characters={characters} selectedDice={selectedDice}
|
<GameBoard className={"mx-auto my-2"} characters={characters} selectedDice={selectedDice}
|
||||||
|
@ -28,7 +28,7 @@ export const GameTile: Component<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}
|
||||||
@ -146,7 +146,7 @@ const CharacterComponent: Component<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:
|
||||||
@ -162,7 +162,7 @@ const CharacterComponent: Component<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`}/>
|
||||||
|
@ -1,28 +1,28 @@
|
|||||||
import Pellet from "./pellet";
|
import Pellet from "./pellet";
|
||||||
|
|
||||||
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;
|
||||||
}
|
|
||||||
|
|
||||||
public addPellet(pellet: Pellet): void {
|
|
||||||
this.pellets.push(pellet);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
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 {
|
||||||
|
this.Pellets.push(pellet);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,53 +1,62 @@
|
|||||||
import {Direction} from "./direction";
|
import {Direction} from "./direction";
|
||||||
|
|
||||||
export enum CharacterType {
|
export enum CharacterType {
|
||||||
pacMan = "pacMan",
|
pacMan,
|
||||||
ghost = "ghost",
|
ghost,
|
||||||
dummy = "dummy",
|
dummy,
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Character {
|
export class Character {
|
||||||
public readonly colour: Colour;
|
public readonly Colour: Colour;
|
||||||
public position: Path;
|
public Position: Path | null;
|
||||||
public isEatable: boolean;
|
public IsEatable: boolean;
|
||||||
public readonly spawnPosition: DirectionalPosition;
|
public readonly SpawnPosition: DirectionalPosition | null;
|
||||||
public readonly type: CharacterType;
|
public readonly Type: CharacterType;
|
||||||
|
|
||||||
public constructor(
|
public constructor(
|
||||||
{
|
{
|
||||||
colour,
|
colour,
|
||||||
position,
|
position = null,
|
||||||
type = CharacterType.dummy,
|
type = CharacterType.dummy,
|
||||||
isEatable = type === CharacterType.pacMan,
|
isEatable = type === CharacterType.pacMan,
|
||||||
spawnPosition
|
spawnPosition = null
|
||||||
}: CharacterProps) {
|
}: CharacterProps) {
|
||||||
this.colour = colour;
|
this.Colour = colour;
|
||||||
this.position = position ?? {end: spawnPosition.at, direction: spawnPosition.direction};
|
this.IsEatable = isEatable;
|
||||||
this.isEatable = isEatable;
|
this.SpawnPosition = spawnPosition;
|
||||||
this.spawnPosition = spawnPosition;
|
|
||||||
this.type = type;
|
this.Position = position ?? spawnPosition ? {
|
||||||
|
End: spawnPosition!.At,
|
||||||
|
Direction: spawnPosition!.Direction
|
||||||
|
} : null;
|
||||||
|
this.Type = type;
|
||||||
}
|
}
|
||||||
|
|
||||||
public follow(path: Path): void {
|
public follow(path: Path): void {
|
||||||
this.position.end = path.end;
|
if (!this.Position) {
|
||||||
this.position.direction = path.direction;
|
this.Position = path;
|
||||||
this.position.path = undefined;
|
} else {
|
||||||
|
this.Position.End = path.End;
|
||||||
|
this.Position.Direction = path.Direction;
|
||||||
|
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 {
|
||||||
this.follow({end: this.spawnPosition.at, direction: this.spawnPosition.direction});
|
if (!this.SpawnPosition) return;
|
||||||
|
this.follow({End: this.SpawnPosition.At, Direction: this.SpawnPosition.Direction});
|
||||||
}
|
}
|
||||||
|
|
||||||
public isAt(position: Position): boolean {
|
public isAt(position: Position): boolean {
|
||||||
return 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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -73,7 +82,7 @@ export class Dummy extends Character {
|
|||||||
colour: "grey",
|
colour: "grey",
|
||||||
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,27 +1,27 @@
|
|||||||
import {Character, CharacterType} from "./character";
|
import {Character, CharacterType} from "./character";
|
||||||
import Box from "./box";
|
import Box from "./box";
|
||||||
import {Direction} from "./direction";
|
|
||||||
|
|
||||||
export default class Player {
|
export default class Player {
|
||||||
public readonly pacMan: Character;
|
public readonly Name: string;
|
||||||
public readonly colour: Colour;
|
public readonly PacMan: Character;
|
||||||
public readonly box: Box;
|
public readonly Colour: Colour;
|
||||||
|
public readonly Box: Box;
|
||||||
|
|
||||||
constructor(props: PlayerProps) {
|
constructor(props: PlayerProps) {
|
||||||
this.colour = props.colour;
|
this.Name = props.name;
|
||||||
this.box = new Box(props.box ?? {colour: props.colour});
|
this.Colour = props.colour;
|
||||||
this.pacMan = new Character(props.pacMan ?? {
|
this.Box = new Box(props.box ?? {colour: props.colour});
|
||||||
|
this.PacMan = new Character(props.pacMan ?? {
|
||||||
colour: props.colour,
|
colour: props.colour,
|
||||||
spawnPosition: {at: {x: 0, y: 0}, direction: Direction.up},
|
|
||||||
type: CharacterType.pacMan
|
type: CharacterType.pacMan
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public stealFrom(other: Player): void {
|
public stealFrom(other: Player): void {
|
||||||
for (let i = 0; i < 2; i++) {
|
for (let i = 0; i < 2; 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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@ 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[] {
|
||||||
return findPossibleRecursive(board, character.position, steps, character, characters);
|
return findPossibleRecursive(board, character.Position, steps, character, characters);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -64,7 +64,7 @@ function findPossibleRecursive(board: GameMap, currentPath: Path, steps: number,
|
|||||||
* @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[]): boolean {
|
function ghostHitsPacMan(character: Character, currentPath: Path, characters: Character[]): boolean {
|
||||||
return character.isGhost() && characters.find(c => c.isPacMan() && c.isAt(currentPath.end)) !== undefined;
|
return character.isGhost() && characters.find(c => c.isPacMan() && c.isAt(currentPath.End)) !== undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -75,7 +75,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -83,10 +83,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];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -107,31 +107,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 [];
|
||||||
@ -149,10 +149,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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -192,7 +192,7 @@ function findTeleportationTiles(board: GameMap): 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][y] !== TileType.wall) {
|
||||||
possiblePositions.push({end: {x, y}, direction: findDirection(x, y, board.length)});
|
possiblePositions.push({End: {x, y}, Direction: findDirection(x, y, board.length)});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -222,7 +222,7 @@ 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;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -232,7 +232,7 @@ 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
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -242,7 +242,7 @@ 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;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -252,8 +252,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,9 +13,9 @@ interface ChildProps extends ComponentProps {
|
|||||||
|
|
||||||
interface CharacterProps {
|
interface CharacterProps {
|
||||||
colour: Colour,
|
colour: Colour,
|
||||||
position?: Path,
|
position?: Path | null,
|
||||||
isEatable?: boolean,
|
isEatable?: boolean,
|
||||||
spawnPosition: DirectionalPosition,
|
spawnPosition?: DirectionalPosition | null,
|
||||||
type?: import("../game/character").CharacterType,
|
type?: import("../game/character").CharacterType,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -25,6 +25,7 @@ interface BoxProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
interface PlayerProps {
|
interface PlayerProps {
|
||||||
|
readonly name: string,
|
||||||
readonly pacMan?: CharacterProps,
|
readonly pacMan?: CharacterProps,
|
||||||
readonly colour: Colour,
|
readonly colour: Colour,
|
||||||
readonly box?: BoxProps,
|
readonly box?: BoxProps,
|
||||||
|
@ -25,14 +25,14 @@ 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[],
|
Path?: Position[] | null,
|
||||||
end: Position,
|
End: Position,
|
||||||
direction: import("../game/direction").Direction
|
Direction: import("../game/direction").Direction
|
||||||
}
|
}
|
||||||
|
|
||||||
type Colour = "white" | "red" | "blue" | "yellow" | "green" | "purple" | "grey";
|
type Colour = "white" | "red" | "blue" | "yellow" | "green" | "purple" | "grey";
|
||||||
|
@ -1,27 +1,3 @@
|
|||||||
export function getCSSColour(colour: Colour): string {
|
export function getCSSColour(colour: Colour): string {
|
||||||
let tailwindColour: string;
|
return `bg-${colour}${colour === "white" ? "-500" : ""}`;
|
||||||
switch (colour) {
|
|
||||||
case "red":
|
|
||||||
tailwindColour = "bg-red-500";
|
|
||||||
break;
|
|
||||||
case "blue":
|
|
||||||
tailwindColour = "bg-blue-500";
|
|
||||||
break;
|
|
||||||
case "yellow":
|
|
||||||
tailwindColour = "bg-yellow-500";
|
|
||||||
break;
|
|
||||||
case "green":
|
|
||||||
tailwindColour = "bg-green-500";
|
|
||||||
break;
|
|
||||||
case "purple":
|
|
||||||
tailwindColour = "bg-purple-500";
|
|
||||||
break;
|
|
||||||
case "grey":
|
|
||||||
tailwindColour = "bg-gray-500";
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
tailwindColour = "bg-white";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return tailwindColour;
|
|
||||||
}
|
}
|
||||||
|
@ -8,10 +8,6 @@ interface IWebSocket {
|
|||||||
export default class WebSocketService {
|
export default class WebSocketService {
|
||||||
private ws?: WebSocket;
|
private ws?: WebSocket;
|
||||||
private readonly _url: string;
|
private readonly _url: string;
|
||||||
private _onOpen?: VoidFunction;
|
|
||||||
private _onReceive?: MessageEventFunction;
|
|
||||||
private _onClose?: VoidFunction;
|
|
||||||
private _onError?: VoidFunction;
|
|
||||||
|
|
||||||
constructor(url: string, {onOpen, onReceive, onClose, onError}: IWebSocket = {}) {
|
constructor(url: string, {onOpen, onReceive, onClose, onError}: IWebSocket = {}) {
|
||||||
this._url = url;
|
this._url = url;
|
||||||
@ -21,8 +17,40 @@ export default class WebSocketService {
|
|||||||
this._onError = onError;
|
this._onError = onError;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private _onOpen?: VoidFunction;
|
||||||
|
|
||||||
|
set onOpen(onOpen: VoidFunction) {
|
||||||
|
this._onOpen = onOpen;
|
||||||
|
if (!this.ws) return;
|
||||||
|
this.ws.onopen = onOpen;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onReceive?: MessageEventFunction;
|
||||||
|
|
||||||
|
set onReceive(onReceive: MessageEventFunction) {
|
||||||
|
this._onReceive = onReceive;
|
||||||
|
if (!this.ws) return;
|
||||||
|
this.ws.onmessage = onReceive;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onClose?: VoidFunction;
|
||||||
|
|
||||||
|
set onClose(onClose: VoidFunction) {
|
||||||
|
this._onClose = onClose;
|
||||||
|
if (!this.ws) return;
|
||||||
|
this.ws.onclose = onClose;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onError?: VoidFunction;
|
||||||
|
|
||||||
|
set onError(onError: VoidFunction) {
|
||||||
|
this._onError = onError;
|
||||||
|
if (!this.ws) return;
|
||||||
|
this.ws.onerror = onError;
|
||||||
|
}
|
||||||
|
|
||||||
public open(): void {
|
public open(): void {
|
||||||
if (typeof WebSocket === "undefined") return;
|
if (typeof WebSocket === "undefined" || this.isConnecting()) return;
|
||||||
this.ws = new WebSocket(this._url);
|
this.ws = new WebSocket(this._url);
|
||||||
if (this._onOpen) this.ws.onopen = this._onOpen;
|
if (this._onOpen) this.ws.onopen = this._onOpen;
|
||||||
if (this._onReceive) this.ws.onmessage = this._onReceive;
|
if (this._onReceive) this.ws.onmessage = this._onReceive;
|
||||||
@ -30,13 +58,27 @@ export default class WebSocketService {
|
|||||||
if (this._onError) this.ws.onerror = this._onError;
|
if (this._onError) this.ws.onerror = this._onError;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public waitForOpen(): Promise<void> {
|
||||||
|
return new Promise<void>((resolve) => {
|
||||||
|
const f = () => {
|
||||||
|
if (this.isOpen()) {
|
||||||
|
if (this._onOpen) this.onOpen = this._onOpen;
|
||||||
|
return resolve();
|
||||||
|
}
|
||||||
|
setTimeout(f, 50);
|
||||||
|
};
|
||||||
|
|
||||||
|
f();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
public send(data: ActionMessage | string): void {
|
public send(data: ActionMessage | string): void {
|
||||||
if (typeof data !== "string") {
|
if (typeof data !== "string") {
|
||||||
data = JSON.stringify(data);
|
data = JSON.stringify(data);
|
||||||
}
|
}
|
||||||
this.ws?.send(data);
|
this.ws?.send(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async sendAndReceive<R>(data: ActionMessage): Promise<R> {
|
public async sendAndReceive<R>(data: ActionMessage): Promise<R> {
|
||||||
if (!this.isOpen()) return Promise.reject("WebSocket is not open");
|
if (!this.isOpen()) return Promise.reject("WebSocket is not open");
|
||||||
|
|
||||||
@ -69,27 +111,11 @@ export default class WebSocketService {
|
|||||||
return this.ws?.readyState === WebSocket?.OPEN;
|
return this.ws?.readyState === WebSocket?.OPEN;
|
||||||
}
|
}
|
||||||
|
|
||||||
set onOpen(onOpen: VoidFunction) {
|
public isConnecting(): boolean {
|
||||||
this._onOpen = onOpen;
|
return this.ws?.readyState === WebSocket?.CONNECTING;
|
||||||
if (!this.ws) return;
|
|
||||||
this.ws.onopen = onOpen;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
set onReceive(onReceive: MessageEventFunction) {
|
public isClosed(): boolean {
|
||||||
this._onReceive = onReceive;
|
return this.ws?.readyState === WebSocket?.CLOSED;
|
||||||
if (!this.ws) return;
|
|
||||||
this.ws.onmessage = onReceive;
|
|
||||||
}
|
|
||||||
|
|
||||||
set onClose(onClose: VoidFunction) {
|
|
||||||
this._onClose = onClose;
|
|
||||||
if (!this.ws) return;
|
|
||||||
this.ws.onclose = onClose;
|
|
||||||
}
|
|
||||||
|
|
||||||
set onError(onError: VoidFunction) {
|
|
||||||
this._onError = onError;
|
|
||||||
if (!this.ws) return;
|
|
||||||
this.ws.onerror = onError;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
export enum GameAction {
|
export enum GameAction {
|
||||||
rollDice,
|
rollDice,
|
||||||
moveCharacter,
|
moveCharacter,
|
||||||
|
playerInfo,
|
||||||
|
ready,
|
||||||
}
|
}
|
@ -8,42 +8,42 @@ let pacMan: Character;
|
|||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
pacMan = new PacMan({
|
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}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
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: []}]);
|
expect(result).toEqual([{end: {x: 3, y: 2}, direction: Direction.up, 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}]}]);
|
expect(result).toEqual([{end: {x: 3, y: 1}, direction: Direction.up, path: [{x: 3, y: 2}]}]);
|
||||||
});
|
});
|
||||||
|
|
||||||
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}]
|
||||||
}]);
|
}]);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -51,21 +51,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}]
|
||||||
}
|
}
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
@ -75,72 +75,72 @@ 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(17);
|
expect(result.length).toBe(17);
|
||||||
});
|
});
|
||||||
|
|
||||||
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 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, []);
|
const result = possibleMovesAlgorithm(testMap, pacMan, 5, []);
|
||||||
expect(result.length).toBe(5);
|
expect(result.length).toBe(5);
|
||||||
});
|
});
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using System.Net.WebSockets;
|
using System.Net.WebSockets;
|
||||||
|
using System.Text.Json;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using pacMan.Game;
|
using pacMan.Game;
|
||||||
using pacMan.Game.Interfaces;
|
using pacMan.Game.Interfaces;
|
||||||
@ -13,19 +14,17 @@ namespace pacMan.Controllers;
|
|||||||
public class GameController : GenericController
|
public class GameController : GenericController
|
||||||
{
|
{
|
||||||
private readonly IDiceCup _diceCup;
|
private readonly IDiceCup _diceCup;
|
||||||
private readonly IPlayer _player; // TODO recieve player from client and choose a starter
|
|
||||||
|
|
||||||
public GameController(ILogger<GameController> logger, IWebSocketService wsService) : base(logger, wsService)
|
public GameController(ILogger<GameController> logger, IWebSocketService wsService) : base(logger, wsService)
|
||||||
{
|
{
|
||||||
_diceCup = new DiceCup();
|
_diceCup = new DiceCup();
|
||||||
_player = new Player
|
|
||||||
{
|
|
||||||
Box = new Box()
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet]
|
[HttpGet]
|
||||||
public override async Task Accept() => await base.Accept();
|
public override async Task Accept()
|
||||||
|
{
|
||||||
|
await base.Accept();
|
||||||
|
}
|
||||||
|
|
||||||
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data)
|
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data)
|
||||||
{
|
{
|
||||||
@ -48,10 +47,14 @@ public class GameController : GenericController
|
|||||||
|
|
||||||
message.Data = rolls;
|
message.Data = rolls;
|
||||||
break;
|
break;
|
||||||
case GameAction.AppendBox:
|
case GameAction.PlayerInfo:
|
||||||
// TODO
|
Player player = JsonSerializer.Deserialize<Player>(message.Data);
|
||||||
// Add pellets to box
|
var group = WsService.AddPlayer(player); // TODO missing some data?
|
||||||
// Forward box to all clients
|
|
||||||
|
message.Data = group.Players;
|
||||||
|
break;
|
||||||
|
case GameAction.Ready:
|
||||||
|
// TODO select starter player
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
Logger.Log(LogLevel.Information, "Forwarding message to all clients");
|
Logger.Log(LogLevel.Information, "Forwarding message to all clients");
|
||||||
|
@ -6,15 +6,15 @@ namespace pacMan.Controllers;
|
|||||||
|
|
||||||
public abstract class GenericController : ControllerBase
|
public abstract class GenericController : ControllerBase
|
||||||
{
|
{
|
||||||
protected readonly ILogger<GenericController> Logger;
|
|
||||||
private readonly IWebSocketService _wsService;
|
|
||||||
private WebSocket? _webSocket;
|
|
||||||
private const int BufferSize = 1024 * 4;
|
private const int BufferSize = 1024 * 4;
|
||||||
|
protected readonly ILogger<GenericController> Logger;
|
||||||
|
protected readonly IWebSocketService WsService;
|
||||||
|
private WebSocket? _webSocket;
|
||||||
|
|
||||||
protected GenericController(ILogger<GenericController> logger, IWebSocketService wsService)
|
protected GenericController(ILogger<GenericController> logger, IWebSocketService wsService)
|
||||||
{
|
{
|
||||||
Logger = logger;
|
Logger = logger;
|
||||||
_wsService = wsService;
|
WsService = wsService;
|
||||||
Logger.Log(LogLevel.Debug, "WebSocket Controller created");
|
Logger.Log(LogLevel.Debug, "WebSocket Controller created");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -25,7 +25,7 @@ public abstract class GenericController : ControllerBase
|
|||||||
using var webSocket = await HttpContext.WebSockets.AcceptWebSocketAsync();
|
using var webSocket = await HttpContext.WebSockets.AcceptWebSocketAsync();
|
||||||
Logger.Log(LogLevel.Information, "WebSocket connection established to {}", HttpContext.Connection.Id);
|
Logger.Log(LogLevel.Information, "WebSocket connection established to {}", HttpContext.Connection.Id);
|
||||||
_webSocket = webSocket;
|
_webSocket = webSocket;
|
||||||
_wsService.Connections += WsServiceOnFire;
|
WsService.Connections += WsServiceOnFire;
|
||||||
await Echo();
|
await Echo();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -37,7 +37,7 @@ public abstract class GenericController : ControllerBase
|
|||||||
private async Task WsServiceOnFire(ArraySegment<byte> segment)
|
private async Task WsServiceOnFire(ArraySegment<byte> segment)
|
||||||
{
|
{
|
||||||
if (_webSocket == null) return;
|
if (_webSocket == null) return;
|
||||||
await _wsService.Send(_webSocket, segment);
|
await WsService.Send(_webSocket, segment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -50,23 +50,23 @@ public abstract class GenericController : ControllerBase
|
|||||||
do
|
do
|
||||||
{
|
{
|
||||||
var buffer = new byte[BufferSize];
|
var buffer = new byte[BufferSize];
|
||||||
result = await _wsService.Receive(_webSocket, buffer);
|
result = await WsService.Receive(_webSocket, buffer);
|
||||||
|
|
||||||
if (result.CloseStatus.HasValue) break;
|
if (result.CloseStatus.HasValue) break;
|
||||||
|
|
||||||
var segment = Run(result, buffer);
|
var segment = Run(result, buffer);
|
||||||
|
|
||||||
_wsService.SendToAll(segment);
|
WsService.SendToAll(segment);
|
||||||
} while (true);
|
} while (true);
|
||||||
|
|
||||||
await _wsService.Close(_webSocket, result.CloseStatus.Value, result.CloseStatusDescription ?? "No reason");
|
await WsService.Close(_webSocket, result.CloseStatus.Value, result.CloseStatusDescription ?? "No reason");
|
||||||
}
|
}
|
||||||
catch (WebSocketException e)
|
catch (WebSocketException e)
|
||||||
{
|
{
|
||||||
Logger.Log(LogLevel.Error, "{}", e.Message);
|
Logger.Log(LogLevel.Error, "{}", e.Message);
|
||||||
}
|
}
|
||||||
|
|
||||||
_wsService.Connections -= WsServiceOnFire;
|
WsService.Connections -= WsServiceOnFire;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data);
|
protected abstract ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data);
|
||||||
|
@ -6,7 +6,8 @@ public enum GameAction
|
|||||||
{
|
{
|
||||||
RollDice,
|
RollDice,
|
||||||
MoveCharacter,
|
MoveCharacter,
|
||||||
AppendBox
|
PlayerInfo,
|
||||||
|
Ready
|
||||||
}
|
}
|
||||||
|
|
||||||
public class ActionMessage<T>
|
public class ActionMessage<T>
|
||||||
@ -14,7 +15,10 @@ public class ActionMessage<T>
|
|||||||
public GameAction Action { get; set; }
|
public GameAction Action { get; set; }
|
||||||
public T? Data { get; set; }
|
public T? Data { get; set; }
|
||||||
|
|
||||||
public static ActionMessage FromJson(string json) => JsonSerializer.Deserialize<ActionMessage>(json)!;
|
public static ActionMessage FromJson(string json)
|
||||||
|
{
|
||||||
|
return JsonSerializer.Deserialize<ActionMessage>(json)!;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class ActionMessage : ActionMessage<dynamic>
|
public class ActionMessage : ActionMessage<dynamic>
|
||||||
|
17
pac-man-board-game/Game/Character.cs
Normal file
17
pac-man-board-game/Game/Character.cs
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
namespace pacMan.Game;
|
||||||
|
|
||||||
|
public class Character
|
||||||
|
{
|
||||||
|
public required string Colour { get; set; }
|
||||||
|
public MovePath? Position { get; set; }
|
||||||
|
public required bool IsEatable { get; set; }
|
||||||
|
public DirectionalPosition? SpawnPosition { get; set; }
|
||||||
|
public required CharacterType Type { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum CharacterType
|
||||||
|
{
|
||||||
|
PacMan,
|
||||||
|
Ghost,
|
||||||
|
Dummy
|
||||||
|
}
|
@ -1,8 +1,9 @@
|
|||||||
|
using pacMan.Game.Items;
|
||||||
|
|
||||||
namespace pacMan.Game.Interfaces;
|
namespace pacMan.Game.Interfaces;
|
||||||
|
|
||||||
public interface IBox : IEnumerable<IPellet>
|
public interface IBox : IEnumerable<IPellet>
|
||||||
{
|
{
|
||||||
void Add(IPellet pellet);
|
|
||||||
|
|
||||||
int CountNormal { get; }
|
int CountNormal { get; }
|
||||||
|
void Add(Pellet pellet);
|
||||||
}
|
}
|
@ -1,12 +1,6 @@
|
|||||||
namespace pacMan.Game.Interfaces;
|
namespace pacMan.Game.Interfaces;
|
||||||
|
|
||||||
public enum PelletType
|
|
||||||
{
|
|
||||||
Normal,
|
|
||||||
PowerPellet
|
|
||||||
}
|
|
||||||
|
|
||||||
public interface IPellet
|
public interface IPellet
|
||||||
{
|
{
|
||||||
PelletType Get { get; set; }
|
bool IsPowerPellet { get; init; }
|
||||||
}
|
}
|
@ -1,6 +1,11 @@
|
|||||||
|
using pacMan.Game.Items;
|
||||||
|
|
||||||
namespace pacMan.Game.Interfaces;
|
namespace pacMan.Game.Interfaces;
|
||||||
|
|
||||||
public interface IPlayer
|
public interface IPlayer
|
||||||
{
|
{
|
||||||
IBox Box { get; init; }
|
string Name { get; init; }
|
||||||
|
Character PacMan { get; init; }
|
||||||
|
string Colour { get; init; }
|
||||||
|
Box Box { get; init; }
|
||||||
}
|
}
|
@ -1,17 +1,26 @@
|
|||||||
using System.Collections;
|
|
||||||
using pacMan.Game.Interfaces;
|
using pacMan.Game.Interfaces;
|
||||||
|
|
||||||
namespace pacMan.Game.Items;
|
namespace pacMan.Game.Items;
|
||||||
|
|
||||||
public class Box : IBox
|
public class Box
|
||||||
{
|
{
|
||||||
private readonly IList<IPellet> _pellets = new List<IPellet>();
|
public required List<Pellet>? Pellets { get; init; } = new();
|
||||||
|
public required string Colour { get; init; }
|
||||||
public int CountNormal => _pellets.Count(pellet => pellet.Get == PelletType.Normal);
|
|
||||||
|
|
||||||
public void Add(IPellet pellet) => _pellets.Add(pellet);
|
public int CountNormal => Pellets?.Count(pellet => !pellet.IsPowerPellet) ?? 0;
|
||||||
|
|
||||||
public IEnumerator<IPellet> GetEnumerator() => _pellets.GetEnumerator();
|
public IEnumerator<IPellet> GetEnumerator()
|
||||||
|
{
|
||||||
|
return Pellets?.GetEnumerator() ?? new List<Pellet>.Enumerator();
|
||||||
|
}
|
||||||
|
|
||||||
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
// IEnumerator IEnumerable.GetEnumerator()
|
||||||
|
// {
|
||||||
|
// return GetEnumerator();
|
||||||
|
// }
|
||||||
|
|
||||||
|
public void Add(Pellet pellet)
|
||||||
|
{
|
||||||
|
Pellets?.Add(pellet);
|
||||||
|
}
|
||||||
}
|
}
|
@ -4,5 +4,5 @@ namespace pacMan.Game.Items;
|
|||||||
|
|
||||||
public class Pellet : IPellet
|
public class Pellet : IPellet
|
||||||
{
|
{
|
||||||
public PelletType Get { get; set; }
|
public bool IsPowerPellet { get; init; }
|
||||||
}
|
}
|
@ -4,5 +4,8 @@ namespace pacMan.Game.Items;
|
|||||||
|
|
||||||
public class Player : IPlayer
|
public class Player : IPlayer
|
||||||
{
|
{
|
||||||
public required IBox Box { get; init; }
|
public required string Name { get; init; }
|
||||||
|
public required Character PacMan { get; init; }
|
||||||
|
public required string Colour { get; init; }
|
||||||
|
public required Box Box { get; init; }
|
||||||
}
|
}
|
28
pac-man-board-game/Game/Positions.cs
Normal file
28
pac-man-board-game/Game/Positions.cs
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
namespace pacMan.Game;
|
||||||
|
|
||||||
|
public class MovePath
|
||||||
|
{
|
||||||
|
public Position[]? Path { get; set; }
|
||||||
|
public required Position End { get; set; }
|
||||||
|
public required Direction Direction { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Position
|
||||||
|
{
|
||||||
|
public int X { get; set; } = 0;
|
||||||
|
public int Y { get; set; } = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum Direction
|
||||||
|
{
|
||||||
|
Left,
|
||||||
|
Up,
|
||||||
|
Right,
|
||||||
|
Down
|
||||||
|
}
|
||||||
|
|
||||||
|
public class DirectionalPosition
|
||||||
|
{
|
||||||
|
public required Position At { get; set; }
|
||||||
|
public required Direction Direction { get; set; }
|
||||||
|
}
|
@ -1,4 +1,6 @@
|
|||||||
using System.Net.WebSockets;
|
using System.Net.WebSockets;
|
||||||
|
using pacMan.Game.Interfaces;
|
||||||
|
using pacMan.Services;
|
||||||
|
|
||||||
namespace pacMan.Interfaces;
|
namespace pacMan.Interfaces;
|
||||||
|
|
||||||
@ -10,4 +12,5 @@ public interface IWebSocketService
|
|||||||
Task<WebSocketReceiveResult> Receive(WebSocket webSocket, byte[] buffer);
|
Task<WebSocketReceiveResult> Receive(WebSocket webSocket, byte[] buffer);
|
||||||
Task Close(WebSocket webSocket, WebSocketCloseStatus closeStatus, string closeStatusDescription);
|
Task Close(WebSocket webSocket, WebSocketCloseStatus closeStatus, string closeStatusDescription);
|
||||||
int CountConnected();
|
int CountConnected();
|
||||||
|
GameGroup AddPlayer(IPlayer player);
|
||||||
}
|
}
|
24
pac-man-board-game/Services/GameGroup.cs
Normal file
24
pac-man-board-game/Services/GameGroup.cs
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
using pacMan.Game;
|
||||||
|
using pacMan.Game.Interfaces;
|
||||||
|
|
||||||
|
namespace pacMan.Services;
|
||||||
|
|
||||||
|
public class GameGroup
|
||||||
|
{
|
||||||
|
public List<IPlayer> Players { get; } = new();
|
||||||
|
public event Func<ArraySegment<byte>, Task>? Connections;
|
||||||
|
|
||||||
|
public bool AddPlayer(IPlayer player)
|
||||||
|
{
|
||||||
|
if (Players.Count >= Rules.MaxPlayers) return false;
|
||||||
|
if (Players.Exists(p => p.Name == player.Name)) return false;
|
||||||
|
|
||||||
|
Players.Add(player);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SendToAll(ArraySegment<byte> segment)
|
||||||
|
{
|
||||||
|
Connections?.Invoke(segment);
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,5 @@
|
|||||||
using System.Net.WebSockets;
|
using System.Net.WebSockets;
|
||||||
|
using pacMan.Game.Interfaces;
|
||||||
using pacMan.Interfaces;
|
using pacMan.Interfaces;
|
||||||
using pacMan.Utils;
|
using pacMan.Utils;
|
||||||
|
|
||||||
@ -7,7 +8,6 @@ namespace pacMan.Services;
|
|||||||
public class WebSocketService : IWebSocketService
|
public class WebSocketService : IWebSocketService
|
||||||
{
|
{
|
||||||
private readonly ILogger<WebSocketService> _logger;
|
private readonly ILogger<WebSocketService> _logger;
|
||||||
public event Func<ArraySegment<byte>, Task>? Connections; // TODO separate connections into groups (1 event per game)
|
|
||||||
|
|
||||||
public WebSocketService(ILogger<WebSocketService> logger)
|
public WebSocketService(ILogger<WebSocketService> logger)
|
||||||
{
|
{
|
||||||
@ -15,6 +15,10 @@ public class WebSocketService : IWebSocketService
|
|||||||
logger.Log(LogLevel.Debug, "WebSocket Service created");
|
logger.Log(LogLevel.Debug, "WebSocket Service created");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public SynchronizedCollection<GameGroup> Games { get; } = new();
|
||||||
|
|
||||||
|
public event Func<ArraySegment<byte>, Task>? Connections;
|
||||||
|
|
||||||
public async Task Send(WebSocket webSocket, ArraySegment<byte> segment)
|
public async Task Send(WebSocket webSocket, ArraySegment<byte> segment)
|
||||||
{
|
{
|
||||||
await webSocket.SendAsync(
|
await webSocket.SendAsync(
|
||||||
@ -26,7 +30,10 @@ public class WebSocketService : IWebSocketService
|
|||||||
_logger.Log(LogLevel.Trace, "Message sent to WebSocket");
|
_logger.Log(LogLevel.Trace, "Message sent to WebSocket");
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SendToAll(ArraySegment<byte> segment) => Connections?.Invoke(segment);
|
public void SendToAll(ArraySegment<byte> segment)
|
||||||
|
{
|
||||||
|
Connections?.Invoke(segment);
|
||||||
|
}
|
||||||
|
|
||||||
public async Task<WebSocketReceiveResult> Receive(WebSocket webSocket, byte[] buffer)
|
public async Task<WebSocketReceiveResult> Receive(WebSocket webSocket, byte[] buffer)
|
||||||
{
|
{
|
||||||
@ -47,5 +54,25 @@ public class WebSocketService : IWebSocketService
|
|||||||
_logger.Log(LogLevel.Information, "WebSocket connection closed");
|
_logger.Log(LogLevel.Information, "WebSocket connection closed");
|
||||||
}
|
}
|
||||||
|
|
||||||
public int CountConnected() => Connections?.GetInvocationList().Length ?? 0;
|
public int CountConnected()
|
||||||
|
{
|
||||||
|
return Connections?.GetInvocationList().Length ?? 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GameGroup AddPlayer(IPlayer player)
|
||||||
|
{
|
||||||
|
var index = 0;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
while (!Games[index].AddPlayer(player)) index++;
|
||||||
|
}
|
||||||
|
catch (ArgumentOutOfRangeException)
|
||||||
|
{
|
||||||
|
var game = new GameGroup();
|
||||||
|
game.AddPlayer(player);
|
||||||
|
Games.Add(game);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Games[index];
|
||||||
|
}
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user