Added movement to characters, and remove used dice

This commit is contained in:
Martin Berg Alstad 2023-05-21 19:18:21 +02:00
parent c5d8ecc362
commit 44192ea3fe
9 changed files with 68 additions and 40 deletions

View File

@ -28,13 +28,15 @@ const map: number[][] = [
interface BoardProps extends ComponentProps { interface BoardProps extends ComponentProps {
characters: Character[], characters: Character[],
selectedDice?: SelectedDice, selectedDice?: SelectedDice,
onMove?: (character: Character) => void
} }
const Board: Component<BoardProps> = ( const Board: Component<BoardProps> = (
{ {
className, className,
characters, characters,
selectedDice selectedDice,
onMove
}) => { }) => {
const [tileSize, setTileSize] = useState(2); const [tileSize, setTileSize] = useState(2);
@ -45,6 +47,14 @@ const Board: Component<BoardProps> = (
setSelectedCharacter(character); setSelectedCharacter(character);
} }
function handleMoveCharacter(position: CharacterPosition): void {
if (selectedCharacter) {
selectedCharacter.moveTo(position);
onMove?.(selectedCharacter);
setSelectedCharacter(undefined);
}
}
useEffect(() => { useEffect(() => {
if (selectedCharacter && selectedDice) { if (selectedCharacter && selectedDice) {
const possiblePositions = findPossiblePositions(map, selectedCharacter.position, selectedDice.value); const possiblePositions = findPossiblePositions(map, selectedCharacter.position, selectedDice.value);
@ -89,7 +99,9 @@ const Board: Component<BoardProps> = (
type={tile} type={tile}
size={tileSize} size={tileSize}
character={characters.find(c => c.isAt({x: colIndex, y: rowIndex}))} 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 { interface TileProps extends ComponentProps {
size: number, size: number,
type?: TileType, type?: TileType,
onClick?: () => void,
character?: Character, character?: Character,
onClick?: (character: Character) => void, onCharacterClick?: (character: Character) => void,
characterClass?: string, characterClass?: string,
} }
@ -113,10 +126,11 @@ const Tile: Component<TileProps> = (
{ {
size, size,
type = TileType.empty, type = TileType.empty,
character,
onClick, onClick,
className, character,
onCharacterClick,
characterClass, characterClass,
className,
}) => { }) => {
function setColor(): string { function setColor(): string {
@ -138,10 +152,11 @@ const Tile: Component<TileProps> = (
return ( return (
<div className={`${setColor()} hover:border relative max-w-[75px] max-h-[75px] ${className}`} <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 && {character &&
<div className={"inline-flex justify-center items-center w-full h-full"}> <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> </div>
} }

View File

@ -1,15 +1,15 @@
import React, {useEffect, useState} from "react"; import React, {useEffect, useRef, useState} from "react";
import Game from "../game/game"; 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 {Ghost, PacMan} from "../game/character"; import {Character, Ghost, PacMan} from "../game/character";
let game: Game; let game: Game;
const characters = [new PacMan("yellow"), new Ghost("purple")];
export const GameComponent: Component = () => { 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 [dice, setDice] = useState<number[]>();
const [selectedDice, setSelectedDice] = useState<SelectedDice>(); const [selectedDice, setSelectedDice] = useState<SelectedDice>();
@ -36,10 +36,33 @@ export const GameComponent: Component = () => {
case Action.rollDice: case Action.rollDice:
setDice(parsed.Data as number[]); // Updates the state of other players setDice(parsed.Data as number[]); // Updates the state of other players
break; 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(() => { useEffect(() => {
game = new Game(); game = new Game();
updateState(); updateState();
@ -56,7 +79,8 @@ export const GameComponent: Component = () => {
<button onClick={startGameLoop}>Roll dice</button> <button onClick={startGameLoop}>Roll dice</button>
</div> </div>
<AllDice values={dice} onclick={handleDiceClick} selectedDiceIndex={selectedDice?.index}/> <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> </div>
); );
}; };

View File

@ -18,12 +18,14 @@ export abstract class Character {
export class PacMan extends Character { export class PacMan extends Character {
moveTo(position: CharacterPosition): void { moveTo(position: CharacterPosition): void {
this.position = position;
} }
} }
export class Ghost extends Character { export class Ghost extends Character {
moveTo(position: CharacterPosition): void { moveTo(position: CharacterPosition): void {
this.position = position;
} }
} }

View File

@ -23,11 +23,6 @@ export default class Game {
const dice = result.Data; const dice = result.Data;
setDice(dice); // Updates the state of the current player 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 // Use the remaining dice to move pac-man if the player moved a ghost or vice versa
// Check if the game is over // Check if the game is over
@ -66,14 +61,6 @@ export default class Game {
return result; 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 { private isGameOver(): boolean {
throw new Error("Not implemented"); throw new Error("Not implemented");
} }

View File

@ -14,18 +14,16 @@ export default function findPossiblePositions(board: number[][], currentPos: Cha
function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, steps: number, function findPossibleRecursive(board: number[][], currentPos: CharacterPosition, steps: number,
possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): CharacterPosition | null { possibleList: CharacterPosition[], visitedTiles: CharacterPosition[]): CharacterPosition | null {
if (isOutsideBoard(currentPos)) { 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)) {
return null; return null;
} else { } else if (isWall(board, currentPos)) {
if (isWall(board, currentPos)) { return null;
return null;
}
} }
visitedTiles.push(currentPos); visitedTiles.push(currentPos);
if (steps === 0) return currentPos; if (steps === 0) return currentPos;
const nextStep = steps - 1; const nextStep = steps - 1;
const result = { const result = {
up: findPossibleRecursive(board, {x: currentPos.x, y: currentPos.y + 1}, nextStep, possibleList, visitedTiles), 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 { function pushToList(board: number[][], list: CharacterPosition[], newEntries: (CharacterPosition | 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) && !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);
} }
} }
@ -77,8 +75,8 @@ function findTeleportationTiles(board: number[][]): CharacterPosition[] {
return possiblePositions; return possiblePositions;
} }
function isOutsideBoard(currentPos: CharacterPosition): boolean { function isOutsideBoard(currentPos: CharacterPosition, boardSize: number): boolean {
return currentPos.x < 0 || currentPos.y < 0; 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: CharacterPosition): boolean {

View File

@ -4,7 +4,7 @@ type Setter<T> = React.Dispatch<React.SetStateAction<T>>;
type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView; type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView;
type ActionMessage<T = object> = { type ActionMessage<T = any> = {
Action: import("../websockets/actions").Action, Action: import("../websockets/actions").Action,
Data?: T Data?: T
} }

View File

@ -1,4 +1,4 @@
export enum Action { export enum Action {
rollDice, rollDice,
pickDice, moveCharacter,
} }

View File

@ -24,7 +24,7 @@ public class GameController : GenericController
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data) 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); Logger.Log(LogLevel.Information, "Received: {}", stringResult);
var action = ActionMessage.FromJson(stringResult); var action = ActionMessage.FromJson(stringResult);
@ -39,12 +39,13 @@ public class GameController : GenericController
{ {
case GameAction.RollDice: case GameAction.RollDice:
var rolls = _diceCup.Roll(); var rolls = _diceCup.Roll();
Logger.Log(LogLevel.Information, "Rolled {}", string.Join(", ", rolls)); Logger.Log(LogLevel.Information, "Rolled [{}]", string.Join(", ", rolls));
message.Data = rolls; message.Data = rolls;
break; break;
default: default:
throw new ArgumentOutOfRangeException(nameof(message), "Action not allowed"); Logger.Log(LogLevel.Information, "Sending message to all clients");
break;
} }
} }
} }

View File

@ -4,7 +4,8 @@ namespace pacMan.Game;
public enum GameAction public enum GameAction
{ {
RollDice RollDice,
MoveCharacter,
} }
public class ActionMessage<T> public class ActionMessage<T>