Refactored
This commit is contained in:
parent
44192ea3fe
commit
a16de14d54
@ -41,13 +41,14 @@ const Board: Component<BoardProps> = (
|
|||||||
|
|
||||||
const [tileSize, setTileSize] = useState(2);
|
const [tileSize, setTileSize] = useState(2);
|
||||||
const [selectedCharacter, setSelectedCharacter] = useState<Character>();
|
const [selectedCharacter, setSelectedCharacter] = useState<Character>();
|
||||||
const [possiblePositions, setPossiblePositions] = useState<CharacterPosition[]>([]);
|
// TODO show the paths to the positions when hovering over a possible position (type Path = CharacterPosition[])
|
||||||
|
const [possiblePositions, setPossiblePositions] = useState<Position[]>([]);
|
||||||
|
|
||||||
function handleSelectCharacter(character: Character): void {
|
function handleSelectCharacter(character: Character): void {
|
||||||
setSelectedCharacter(character);
|
setSelectedCharacter(character);
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleMoveCharacter(position: CharacterPosition): void {
|
function handleMoveCharacter(position: Position): void {
|
||||||
if (selectedCharacter) {
|
if (selectedCharacter) {
|
||||||
selectedCharacter.moveTo(position);
|
selectedCharacter.moveTo(position);
|
||||||
onMove?.(selectedCharacter);
|
onMove?.(selectedCharacter);
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
import React, {useEffect, useRef, useState} from "react";
|
import React, {useEffect, useRef, useState} from "react";
|
||||||
import Game from "../game/game";
|
|
||||||
import {AllDice} from "./dice";
|
import {AllDice} from "./dice";
|
||||||
import {Action} from "../websockets/actions";
|
import {Action} from "../websockets/actions";
|
||||||
import GameBoard from "./gameBoard";
|
import GameBoard from "./gameBoard";
|
||||||
import {Character, Ghost, PacMan} from "../game/character";
|
import {Character, Ghost, PacMan} from "../game/character";
|
||||||
|
import WebSocketService from "../websockets/WebSocketService";
|
||||||
|
|
||||||
let game: Game;
|
const wsService = new WebSocketService("wss://localhost:3000/api/game");
|
||||||
|
|
||||||
export const GameComponent: Component = () => {
|
export const GameComponent: Component = () => {
|
||||||
// Better for testing than outside of the component
|
// Better for testing than outside of the component
|
||||||
@ -16,39 +16,38 @@ export const GameComponent: Component = () => {
|
|||||||
|
|
||||||
function handleDiceClick(selected: SelectedDice): void {
|
function handleDiceClick(selected: SelectedDice): void {
|
||||||
setSelectedDice(selected);
|
setSelectedDice(selected);
|
||||||
game.selectedDice = selected;
|
}
|
||||||
|
|
||||||
|
function rollDice(): void {
|
||||||
|
wsService.send({Action: Action.rollDice});
|
||||||
}
|
}
|
||||||
|
|
||||||
function startGameLoop(): void {
|
function startGameLoop(): void {
|
||||||
setSelectedDice(undefined);
|
setSelectedDice(undefined);
|
||||||
if (!game.isConnected()) {
|
if (!wsService.isOpen()) {
|
||||||
setTimeout(startGameLoop, 50);
|
setTimeout(startGameLoop, 50);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
void game.gameLoop(setDice);
|
rollDice();
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateState(): void {
|
function doAction(message: MessageEvent<string>): void {
|
||||||
game.wsService.onReceive = (message) => {
|
const parsed: ActionMessage = JSON.parse(message.data);
|
||||||
const parsed: ActionMessage = JSON.parse(message.data);
|
|
||||||
|
|
||||||
switch (parsed.Action) {
|
switch (parsed.Action) {
|
||||||
case Action.rollDice:
|
case Action.rollDice:
|
||||||
setDice(parsed.Data as number[]); // Updates the state of other players
|
setDice(parsed.Data as number[]);
|
||||||
break;
|
break;
|
||||||
case Action.moveCharacter:
|
case Action.moveCharacter:
|
||||||
setDice(parsed.Data?.dice as number[]);
|
setDice(parsed.Data?.dice as number[]);
|
||||||
const character = parsed.Data?.character as Character;
|
const character = parsed.Data?.character as Character;
|
||||||
characters.current.find(c => c.color === character.color)?.moveTo(character.position);
|
characters.current.find(c => c.color === character.color)?.moveTo(character.position);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function onCharacterMove(character: Character): void {
|
function onCharacterMove(character: Character): void {
|
||||||
if (dice && selectedDice) {
|
if (dice && selectedDice) {
|
||||||
// Remove the dice that was used from the list of dice
|
|
||||||
|
|
||||||
dice.splice(selectedDice.index, 1);
|
dice.splice(selectedDice.index, 1);
|
||||||
setDice([...dice]);
|
setDice([...dice]);
|
||||||
}
|
}
|
||||||
@ -60,16 +59,15 @@ export const GameComponent: Component = () => {
|
|||||||
character: character
|
character: character
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
game.wsService.send(data);
|
wsService.send(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
game = new Game();
|
wsService.onReceive = doAction;
|
||||||
updateState();
|
wsService.open();
|
||||||
|
|
||||||
game.connectToServer();
|
|
||||||
startGameLoop();
|
startGameLoop();
|
||||||
return () => game.disconnect();
|
return () => wsService.close();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -2,29 +2,29 @@ type CharacterColor = "red" | "blue" | "yellow" | "green" | "purple";
|
|||||||
|
|
||||||
export abstract class Character {
|
export abstract class Character {
|
||||||
public color: CharacterColor;
|
public color: CharacterColor;
|
||||||
public position: CharacterPosition;
|
public position: Position;
|
||||||
|
|
||||||
public constructor(color: CharacterColor, startPosition: CharacterPosition = {x: 0, y: 0}) {
|
public constructor(color: CharacterColor, startPosition: Position = {x: 0, y: 0}) {
|
||||||
this.color = color;
|
this.color = color;
|
||||||
this.position = startPosition;
|
this.position = startPosition;
|
||||||
}
|
}
|
||||||
|
|
||||||
public abstract moveTo(position: CharacterPosition): void;
|
public abstract moveTo(position: Position): void;
|
||||||
|
|
||||||
public isAt(position: CharacterPosition): boolean {
|
public isAt(position: Position): boolean {
|
||||||
return this.position.x === position.x && this.position.y === position.y;
|
return this.position.x === position.x && this.position.y === position.y;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class PacMan extends Character {
|
export class PacMan extends Character {
|
||||||
moveTo(position: CharacterPosition): void {
|
moveTo(position: Position): void {
|
||||||
this.position = position;
|
this.position = position;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Ghost extends Character {
|
export class Ghost extends Character {
|
||||||
moveTo(position: CharacterPosition): void {
|
moveTo(position: Position): void {
|
||||||
this.position = position;
|
this.position = position;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,79 +0,0 @@
|
|||||||
import WebSocketService from "../websockets/WebSocketService";
|
|
||||||
import {Action} from "../websockets/actions";
|
|
||||||
|
|
||||||
export default class Game {
|
|
||||||
|
|
||||||
private readonly _wsService: WebSocketService;
|
|
||||||
public selectedDice?: SelectedDice;
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
this._wsService = new WebSocketService("wss://localhost:3000/api/game");
|
|
||||||
// Connect to the server
|
|
||||||
|
|
||||||
// Create players
|
|
||||||
|
|
||||||
// Pick player pieces
|
|
||||||
|
|
||||||
// Roll to start
|
|
||||||
}
|
|
||||||
|
|
||||||
public async gameLoop(setDice: Setter<number[] | undefined>): Promise<void> {
|
|
||||||
// Throw the dice
|
|
||||||
const result = await this.rollDice();
|
|
||||||
const dice = result.Data;
|
|
||||||
setDice(dice); // Updates the state of the current player
|
|
||||||
|
|
||||||
// Use the remaining dice to move pac-man if the player moved a ghost or vice versa
|
|
||||||
|
|
||||||
// Check if the game is over
|
|
||||||
|
|
||||||
// If not, next player
|
|
||||||
}
|
|
||||||
|
|
||||||
public connectToServer(): void {
|
|
||||||
this._wsService.open();
|
|
||||||
this._wsService.registerEvents();
|
|
||||||
}
|
|
||||||
|
|
||||||
public isConnected(): boolean {
|
|
||||||
return this._wsService.isOpen();
|
|
||||||
}
|
|
||||||
|
|
||||||
public disconnect(): void {
|
|
||||||
this._wsService.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
private createPlayers(): void {
|
|
||||||
throw new Error("Not implemented");
|
|
||||||
}
|
|
||||||
|
|
||||||
private pickPlayerPieces(): void {
|
|
||||||
throw new Error("Not implemented");
|
|
||||||
}
|
|
||||||
|
|
||||||
private rollToStart(): void {
|
|
||||||
throw new Error("Not implemented");
|
|
||||||
}
|
|
||||||
|
|
||||||
private async rollDice(): Promise<ActionMessage<number[]>> {
|
|
||||||
let result: ActionMessage<number[]>;
|
|
||||||
result = await this._wsService.sendAndReceive<ActionMessage<number[]>>({Action: Action.rollDice});
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
private isGameOver(): boolean {
|
|
||||||
throw new Error("Not implemented");
|
|
||||||
}
|
|
||||||
|
|
||||||
private nextPlayer(): void {
|
|
||||||
throw new Error("Not implemented");
|
|
||||||
}
|
|
||||||
|
|
||||||
private chooseCharacter(): void {
|
|
||||||
throw new Error("Method not implemented.");
|
|
||||||
}
|
|
||||||
|
|
||||||
get wsService(): WebSocketService {
|
|
||||||
return this._wsService;
|
|
||||||
}
|
|
||||||
}
|
|
@ -6,14 +6,14 @@ import {TileType} from "./tileType";
|
|||||||
* @param currentPos The current position of the character
|
* @param currentPos The current position of the character
|
||||||
* @param steps The number of steps the character can move
|
* @param steps The number of steps the character can move
|
||||||
*/
|
*/
|
||||||
export default function findPossiblePositions(board: number[][], currentPos: CharacterPosition, steps: number): CharacterPosition[] {
|
export default function findPossiblePositions(board: number[][], currentPos: Position, steps: number): Position[] {
|
||||||
const possiblePositions: CharacterPosition[] = [];
|
const possiblePositions: Position[] = [];
|
||||||
findPossibleRecursive(board, currentPos, steps, possiblePositions, []);
|
findPossibleRecursive(board, currentPos, steps, possiblePositions, []);
|
||||||
return possiblePositions;
|
return possiblePositions;
|
||||||
}
|
}
|
||||||
|
|
||||||
function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, steps: number,
|
function findPossibleRecursive(board: number[][], currentPos: Position, steps: number,
|
||||||
possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): CharacterPosition | null {
|
possibleList: Position[], visitedTiles: Position[]): Position | null {
|
||||||
if (isOutsideBoard(currentPos, board.length)) {
|
if (isOutsideBoard(currentPos, board.length)) {
|
||||||
addTeleportationTiles(board, currentPos, steps, possibleList, visitedTiles);
|
addTeleportationTiles(board, currentPos, steps, possibleList, visitedTiles);
|
||||||
} else if (visitedTiles.find(tile => tile.x === currentPos.x && tile.y === currentPos.y)) {
|
} else if (visitedTiles.find(tile => tile.x === currentPos.x && tile.y === currentPos.y)) {
|
||||||
@ -36,9 +36,9 @@ function findPossibleRecursive(board: number[][], currentPos: CharacterPosition,
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
function addTeleportationTiles(board: number[][], currentPos: CharacterPosition, steps: number,
|
function addTeleportationTiles(board: number[][], currentPos: Position, steps: number,
|
||||||
possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): void {
|
possibleList: Position[], visitedTiles: Position[]): void {
|
||||||
const newPositons: (CharacterPosition | null)[] = [];
|
const newPositons: (Position | null)[] = [];
|
||||||
const possiblePositions = findTeleportationTiles(board);
|
const possiblePositions = findTeleportationTiles(board);
|
||||||
for (const pos of possiblePositions) {
|
for (const pos of possiblePositions) {
|
||||||
if (pos.x !== Math.max(currentPos.x, 0) || pos.y !== Math.max(currentPos.y, 0)) {
|
if (pos.x !== Math.max(currentPos.x, 0) || pos.y !== Math.max(currentPos.y, 0)) {
|
||||||
@ -48,7 +48,7 @@ function addTeleportationTiles(board: number[][], currentPos: CharacterPosition,
|
|||||||
pushToList(board, possibleList, newPositons);
|
pushToList(board, possibleList, newPositons);
|
||||||
}
|
}
|
||||||
|
|
||||||
function pushToList(board: number[][], list: CharacterPosition[], newEntries: (CharacterPosition | null)[]): void {
|
function pushToList(board: number[][], list: Position[], newEntries: (Position | null)[]): void {
|
||||||
for (const entry of newEntries) {
|
for (const entry of newEntries) {
|
||||||
if (entry !== null && !list.find(p => p.x === entry.x && p.y === entry.y) && !isOutsideBoard(entry, board.length) && !isSpawn(board, entry)) {
|
if (entry !== null && !list.find(p => p.x === entry.x && p.y === entry.y) && !isOutsideBoard(entry, board.length) && !isSpawn(board, entry)) {
|
||||||
list.push(entry);
|
list.push(entry);
|
||||||
@ -56,8 +56,8 @@ function pushToList(board: number[][], list: CharacterPosition[], newEntries: (C
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function findTeleportationTiles(board: number[][]): CharacterPosition[] {
|
function findTeleportationTiles(board: number[][]): Position[] {
|
||||||
const possiblePositions: CharacterPosition[] = [];
|
const possiblePositions: Position[] = [];
|
||||||
const edge = [0, board.length - 1];
|
const edge = [0, board.length - 1];
|
||||||
|
|
||||||
for (const e of edge) {
|
for (const e of edge) {
|
||||||
@ -75,15 +75,15 @@ function findTeleportationTiles(board: number[][]): CharacterPosition[] {
|
|||||||
return possiblePositions;
|
return possiblePositions;
|
||||||
}
|
}
|
||||||
|
|
||||||
function isOutsideBoard(currentPos: CharacterPosition, boardSize: number): boolean {
|
function isOutsideBoard(currentPos: Position, boardSize: number): boolean {
|
||||||
return currentPos.x < 0 || currentPos.x >= boardSize || currentPos.y < 0 || currentPos.y >= boardSize;
|
return currentPos.x < 0 || currentPos.x >= boardSize || currentPos.y < 0 || currentPos.y >= boardSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
function isWall(board: number[][], currentPos: CharacterPosition): boolean {
|
function isWall(board: number[][], currentPos: Position): boolean {
|
||||||
return board[currentPos.y][currentPos.x] === TileType.wall; // TODO shouldn't work, but it does
|
return board[currentPos.y][currentPos.x] === TileType.wall; // TODO shouldn't work, but it does
|
||||||
}
|
}
|
||||||
|
|
||||||
function isSpawn(board: number[][], currentPos: CharacterPosition): boolean {
|
function isSpawn(board: number[][], currentPos: Position): boolean {
|
||||||
return board[currentPos.x][currentPos.y] === TileType.pacmanSpawn ||
|
return board[currentPos.x][currentPos.y] === TileType.pacmanSpawn ||
|
||||||
board[currentPos.x][currentPos.y] === TileType.ghostSpawn;
|
board[currentPos.x][currentPos.y] === TileType.ghostSpawn;
|
||||||
}
|
}
|
||||||
|
@ -22,10 +22,7 @@ export const Counter: Component = () => {
|
|||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
ws.onReceive = receiveMessage;
|
ws.onReceive = receiveMessage;
|
||||||
ws.open();
|
ws.open();
|
||||||
ws.registerEvents();
|
return () => ws.close();
|
||||||
return () => {
|
|
||||||
ws.close();
|
|
||||||
};
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
type MessageEventFunction = (data: MessageEvent<any>) => void;
|
type MessageEventFunction<T = any> = (data: MessageEvent<T>) => void;
|
||||||
|
|
||||||
type Setter<T> = React.Dispatch<React.SetStateAction<T>>;
|
type Setter<T> = React.Dispatch<React.SetStateAction<T>>;
|
||||||
|
|
||||||
@ -14,4 +14,4 @@ type SelectedDice = {
|
|||||||
index: number
|
index: number
|
||||||
};
|
};
|
||||||
|
|
||||||
type CharacterPosition = { x: number, y: number };
|
type Position = { x: number, y: number };
|
||||||
|
@ -23,10 +23,6 @@ export default class WebSocketService {
|
|||||||
|
|
||||||
public open(): void {
|
public open(): void {
|
||||||
this.ws = new WebSocket(this._url);
|
this.ws = new WebSocket(this._url);
|
||||||
}
|
|
||||||
|
|
||||||
public registerEvents(): void {
|
|
||||||
if (!this.ws) return;
|
|
||||||
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;
|
||||||
if (this._onClose) this.ws.onclose = this._onClose;
|
if (this._onClose) this.ws.onclose = this._onClose;
|
||||||
|
@ -44,7 +44,7 @@ public class GameController : GenericController
|
|||||||
message.Data = rolls;
|
message.Data = rolls;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
Logger.Log(LogLevel.Information, "Sending message to all clients");
|
Logger.Log(LogLevel.Information, "Forwarding message to all clients");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -36,12 +36,12 @@ public abstract class GenericController : ControllerBase
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var buffer = new byte[BufferSize];
|
|
||||||
WebSocketReceiveResult? result;
|
WebSocketReceiveResult? result;
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
|
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);
|
||||||
|
@ -38,6 +38,7 @@
|
|||||||
<TypeScriptCompile Remove="ClientApp\src\classes\tileMap.ts" />
|
<TypeScriptCompile Remove="ClientApp\src\classes\tileMap.ts" />
|
||||||
<TypeScriptCompile Remove="ClientApp\src\game\tileMap.ts" />
|
<TypeScriptCompile Remove="ClientApp\src\game\tileMap.ts" />
|
||||||
<TypeScriptCompile Remove="ClientApp\src\components\gameCanvas.tsx" />
|
<TypeScriptCompile Remove="ClientApp\src\components\gameCanvas.tsx" />
|
||||||
|
<TypeScriptCompile Remove="ClientApp\src\game\game.ts" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<Target Name="DebugEnsureNodeEnv" BeforeTargets="Build" Condition=" '$(Configuration)' == 'Debug' And !Exists('$(SpaRoot)node_modules') ">
|
<Target Name="DebugEnsureNodeEnv" BeforeTargets="Build" Condition=" '$(Configuration)' == 'Debug' And !Exists('$(SpaRoot)node_modules') ">
|
||||||
|
Loading…
x
Reference in New Issue
Block a user