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 {
|
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>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -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");
|
||||||
}
|
}
|
||||||
|
@ -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 {
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
export enum Action {
|
export enum Action {
|
||||||
rollDice,
|
rollDice,
|
||||||
pickDice,
|
moveCharacter,
|
||||||
}
|
}
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -4,7 +4,8 @@ namespace pacMan.Game;
|
|||||||
|
|
||||||
public enum GameAction
|
public enum GameAction
|
||||||
{
|
{
|
||||||
RollDice
|
RollDice,
|
||||||
|
MoveCharacter,
|
||||||
}
|
}
|
||||||
|
|
||||||
public class ActionMessage<T>
|
public class ActionMessage<T>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user