Added movement to characters, and remove used dice
This commit is contained in:
parent
c5d8ecc362
commit
44192ea3fe
@ -28,13 +28,15 @@ const map: number[][] = [
|
||||
interface BoardProps extends ComponentProps {
|
||||
characters: Character[],
|
||||
selectedDice?: SelectedDice,
|
||||
onMove?: (character: Character) => void
|
||||
}
|
||||
|
||||
const Board: Component<BoardProps> = (
|
||||
{
|
||||
className,
|
||||
characters,
|
||||
selectedDice
|
||||
selectedDice,
|
||||
onMove
|
||||
}) => {
|
||||
|
||||
const [tileSize, setTileSize] = useState(2);
|
||||
@ -45,6 +47,14 @@ const Board: Component<BoardProps> = (
|
||||
setSelectedCharacter(character);
|
||||
}
|
||||
|
||||
function handleMoveCharacter(position: CharacterPosition): void {
|
||||
if (selectedCharacter) {
|
||||
selectedCharacter.moveTo(position);
|
||||
onMove?.(selectedCharacter);
|
||||
setSelectedCharacter(undefined);
|
||||
}
|
||||
}
|
||||
|
||||
useEffect(() => {
|
||||
if (selectedCharacter && selectedDice) {
|
||||
const possiblePositions = findPossiblePositions(map, selectedCharacter.position, selectedDice.value);
|
||||
@ -89,7 +99,9 @@ const Board: Component<BoardProps> = (
|
||||
type={tile}
|
||||
size={tileSize}
|
||||
character={characters.find(c => c.isAt({x: colIndex, y: rowIndex}))}
|
||||
onClick={handleSelectCharacter}
|
||||
onCharacterClick={handleSelectCharacter}
|
||||
onClick={possiblePositions.find(p => p.x === colIndex && p.y === rowIndex) ?
|
||||
() => handleMoveCharacter({x: colIndex, y: rowIndex}) : undefined}
|
||||
/>
|
||||
)
|
||||
}
|
||||
@ -104,8 +116,9 @@ export default Board;
|
||||
interface TileProps extends ComponentProps {
|
||||
size: number,
|
||||
type?: TileType,
|
||||
onClick?: () => void,
|
||||
character?: Character,
|
||||
onClick?: (character: Character) => void,
|
||||
onCharacterClick?: (character: Character) => void,
|
||||
characterClass?: string,
|
||||
}
|
||||
|
||||
@ -113,10 +126,11 @@ const Tile: Component<TileProps> = (
|
||||
{
|
||||
size,
|
||||
type = TileType.empty,
|
||||
character,
|
||||
onClick,
|
||||
className,
|
||||
character,
|
||||
onCharacterClick,
|
||||
characterClass,
|
||||
className,
|
||||
}) => {
|
||||
|
||||
function setColor(): string {
|
||||
@ -138,10 +152,11 @@ const Tile: Component<TileProps> = (
|
||||
|
||||
return (
|
||||
<div className={`${setColor()} hover:border relative max-w-[75px] max-h-[75px] ${className}`}
|
||||
style={{width: `${size}px`, height: `${size}px`}}>
|
||||
style={{width: `${size}px`, height: `${size}px`}}
|
||||
onClick={onClick}>
|
||||
{character &&
|
||||
<div className={"inline-flex justify-center items-center w-full h-full"}>
|
||||
<CharacterComponent character={character} onClick={onClick} className={characterClass}/>
|
||||
<CharacterComponent character={character} onClick={onCharacterClick} className={characterClass}/>
|
||||
</div>
|
||||
}
|
||||
|
||||
|
@ -1,15 +1,15 @@
|
||||
import React, {useEffect, useState} from "react";
|
||||
import React, {useEffect, useRef, useState} from "react";
|
||||
import Game from "../game/game";
|
||||
import {AllDice} from "./dice";
|
||||
import {Action} from "../websockets/actions";
|
||||
import GameBoard from "./gameBoard";
|
||||
import {Ghost, PacMan} from "../game/character";
|
||||
import {Character, Ghost, PacMan} from "../game/character";
|
||||
|
||||
let game: Game;
|
||||
|
||||
const characters = [new PacMan("yellow"), new Ghost("purple")];
|
||||
|
||||
export const GameComponent: Component = () => {
|
||||
// Better for testing than outside of the component
|
||||
const characters = useRef([new PacMan("yellow"), new Ghost("purple")]);
|
||||
|
||||
const [dice, setDice] = useState<number[]>();
|
||||
const [selectedDice, setSelectedDice] = useState<SelectedDice>();
|
||||
@ -36,10 +36,33 @@ export const GameComponent: Component = () => {
|
||||
case Action.rollDice:
|
||||
setDice(parsed.Data as number[]); // Updates the state of other players
|
||||
break;
|
||||
case Action.moveCharacter:
|
||||
setDice(parsed.Data?.dice as number[]);
|
||||
const character = parsed.Data?.character as Character;
|
||||
characters.current.find(c => c.color === character.color)?.moveTo(character.position);
|
||||
break;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
function onCharacterMove(character: Character): void {
|
||||
if (dice && selectedDice) {
|
||||
// Remove the dice that was used from the list of dice
|
||||
|
||||
dice.splice(selectedDice.index, 1);
|
||||
setDice([...dice]);
|
||||
}
|
||||
setSelectedDice(undefined);
|
||||
const data: ActionMessage = {
|
||||
Action: Action.moveCharacter,
|
||||
Data: {
|
||||
dice: dice?.length ?? 0 > 0 ? dice : null,
|
||||
character: character
|
||||
}
|
||||
};
|
||||
game.wsService.send(data);
|
||||
}
|
||||
|
||||
useEffect(() => {
|
||||
game = new Game();
|
||||
updateState();
|
||||
@ -56,7 +79,8 @@ export const GameComponent: Component = () => {
|
||||
<button onClick={startGameLoop}>Roll dice</button>
|
||||
</div>
|
||||
<AllDice values={dice} onclick={handleDiceClick} selectedDiceIndex={selectedDice?.index}/>
|
||||
<GameBoard className={"mx-auto"} characters={characters} selectedDice={selectedDice}/>
|
||||
<GameBoard className={"mx-auto my-2"} characters={characters.current} selectedDice={selectedDice}
|
||||
onMove={onCharacterMove}/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
@ -18,12 +18,14 @@ export abstract class Character {
|
||||
|
||||
export class PacMan extends Character {
|
||||
moveTo(position: CharacterPosition): void {
|
||||
this.position = position;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
export class Ghost extends Character {
|
||||
moveTo(position: CharacterPosition): void {
|
||||
this.position = position;
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -23,11 +23,6 @@ export default class Game {
|
||||
const dice = result.Data;
|
||||
setDice(dice); // Updates the state of the current player
|
||||
|
||||
// Choose a dice
|
||||
|
||||
// Choose a character to move
|
||||
// this.chooseCharacter();
|
||||
|
||||
// Use the remaining dice to move pac-man if the player moved a ghost or vice versa
|
||||
|
||||
// Check if the game is over
|
||||
@ -66,14 +61,6 @@ export default class Game {
|
||||
return result;
|
||||
}
|
||||
|
||||
private movePacMan(steps: number): void {
|
||||
throw new Error("Not implemented");
|
||||
}
|
||||
|
||||
private moveGhost(steps: number): void {
|
||||
throw new Error("Not implemented");
|
||||
}
|
||||
|
||||
private isGameOver(): boolean {
|
||||
throw new Error("Not implemented");
|
||||
}
|
||||
|
@ -14,18 +14,16 @@ export default function findPossiblePositions(board: number[][], currentPos: Cha
|
||||
|
||||
function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, steps: number,
|
||||
possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): CharacterPosition | null {
|
||||
if (isOutsideBoard(currentPos)) {
|
||||
if (isOutsideBoard(currentPos, board.length)) {
|
||||
addTeleportationTiles(board, currentPos, steps, possibleList, visitedTiles);
|
||||
} else if (visitedTiles.find(tile => tile.x === currentPos.x && tile.y === currentPos.y)) {
|
||||
return null;
|
||||
} else {
|
||||
if (isWall(board, currentPos)) {
|
||||
return null;
|
||||
}
|
||||
} else if (isWall(board, currentPos)) {
|
||||
return null;
|
||||
}
|
||||
visitedTiles.push(currentPos);
|
||||
if (steps === 0) return currentPos;
|
||||
|
||||
|
||||
const nextStep = steps - 1;
|
||||
const result = {
|
||||
up: findPossibleRecursive(board, {x: currentPos.x, y: currentPos.y + 1}, nextStep, possibleList, visitedTiles),
|
||||
@ -52,7 +50,7 @@ function addTeleportationTiles(board: number[][], currentPos: CharacterPosition,
|
||||
|
||||
function pushToList(board: number[][], list: CharacterPosition[], newEntries: (CharacterPosition | null)[]): void {
|
||||
for (const entry of newEntries) {
|
||||
if (entry !== null && !list.find(p => p.x === entry.x && p.y === entry.y) && !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);
|
||||
}
|
||||
}
|
||||
@ -77,8 +75,8 @@ function findTeleportationTiles(board: number[][]): CharacterPosition[] {
|
||||
return possiblePositions;
|
||||
}
|
||||
|
||||
function isOutsideBoard(currentPos: CharacterPosition): boolean {
|
||||
return currentPos.x < 0 || currentPos.y < 0;
|
||||
function isOutsideBoard(currentPos: CharacterPosition, boardSize: number): boolean {
|
||||
return currentPos.x < 0 || currentPos.x >= boardSize || currentPos.y < 0 || currentPos.y >= boardSize;
|
||||
}
|
||||
|
||||
function isWall(board: number[][], currentPos: CharacterPosition): boolean {
|
||||
|
@ -4,7 +4,7 @@ type Setter<T> = React.Dispatch<React.SetStateAction<T>>;
|
||||
|
||||
type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView;
|
||||
|
||||
type ActionMessage<T = object> = {
|
||||
type ActionMessage<T = any> = {
|
||||
Action: import("../websockets/actions").Action,
|
||||
Data?: T
|
||||
}
|
||||
|
@ -1,4 +1,4 @@
|
||||
export enum Action {
|
||||
rollDice,
|
||||
pickDice,
|
||||
moveCharacter,
|
||||
}
|
@ -24,7 +24,7 @@ public class GameController : GenericController
|
||||
|
||||
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data)
|
||||
{
|
||||
var stringResult = data.GetString(data.Length);
|
||||
var stringResult = data.GetString(result.Count);
|
||||
|
||||
Logger.Log(LogLevel.Information, "Received: {}", stringResult);
|
||||
var action = ActionMessage.FromJson(stringResult);
|
||||
@ -39,12 +39,13 @@ public class GameController : GenericController
|
||||
{
|
||||
case GameAction.RollDice:
|
||||
var rolls = _diceCup.Roll();
|
||||
Logger.Log(LogLevel.Information, "Rolled {}", string.Join(", ", rolls));
|
||||
Logger.Log(LogLevel.Information, "Rolled [{}]", string.Join(", ", rolls));
|
||||
|
||||
message.Data = rolls;
|
||||
break;
|
||||
default:
|
||||
throw new ArgumentOutOfRangeException(nameof(message), "Action not allowed");
|
||||
Logger.Log(LogLevel.Information, "Sending message to all clients");
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
@ -4,7 +4,8 @@ namespace pacMan.Game;
|
||||
|
||||
public enum GameAction
|
||||
{
|
||||
RollDice
|
||||
RollDice,
|
||||
MoveCharacter,
|
||||
}
|
||||
|
||||
public class ActionMessage<T>
|
||||
|
Loading…
x
Reference in New Issue
Block a user