2023-06-08 19:48:27 +02:00

130 lines
4.0 KiB
TypeScript

import React, {useEffect, useState} from "react";
import {Character, PacMan} from "../game/character";
import findPossiblePositions from "../game/possibleMovesAlgorithm";
import {Direction} from "../game/direction";
import {GameTile} from "./gameTile";
import {TileType} from "../game/tileType";
import Pellet from "../game/pellet";
interface BoardProps extends ComponentProps {
characters: Character[],
selectedDice?: SelectedDice,
onMove?: Action<Position[]>,
map: GameMap
}
const Board: Component<BoardProps> = (
{
className,
characters,
selectedDice,
onMove,
map
}) => {
const [selectedCharacter, setSelectedCharacter] = useState<Character>();
const [possiblePositions, setPossiblePositions] = useState<Path[]>([]); // TODO reset when other client moves a character
const [hoveredPosition, setHoveredPosition] = useState<Path>();
function handleSelectCharacter(character: Character): void {
setSelectedCharacter(character);
}
function handleShowPath(path: Path): void {
setHoveredPosition(path);
}
function handleMoveCharacter(destination: Path): void {
if (selectedCharacter) {
setHoveredPosition(undefined);
if (selectedCharacter.isGhost()) {
tryMovePacManToSpawn(destination);
}
selectedCharacter.follow(destination);
const positions = pickUpPellets(destination);
onMove?.(positions);
setSelectedCharacter(undefined);
}
}
function tryMovePacManToSpawn(destination: Path): void {
const takenChar = characters.find(c => c.isPacMan() && c.isAt(destination.end));
if (takenChar) {
takenChar.moveToSpawn();
// TODO steal from player
}
}
function pickUpPellets(destination: Path): Position[] {
const positions: Position[] = [];
if (selectedCharacter instanceof PacMan) {
const pacMan = selectedCharacter as PacMan;
for (const tile of [...destination.path ?? [], destination.end]) {
const currentTile = map[tile.y][tile.x];
if (currentTile === TileType.pellet) {
pacMan.box.addPellet(new Pellet());
map[tile.y][tile.x] = TileType.empty;
positions.push(tile);
} else if (currentTile === TileType.powerPellet) {
pacMan.box.addPellet(new Pellet(true));
map[tile.y][tile.x] = TileType.empty;
positions.push(tile);
}
}
}
return positions;
}
useEffect(() => {
if (selectedCharacter && selectedDice) {
const possiblePaths = findPossiblePositions(map, selectedCharacter, selectedDice.value, characters);
setPossiblePositions(possiblePaths);
} else {
setPossiblePositions([]);
}
}, [selectedCharacter, selectedDice]);
useEffect(() => {
for (const character of characters) { // TODO make more dynamic
if (character instanceof PacMan) {
character.position = {end: {x: 3, y: 3}, direction: Direction.up};
} else {
character.position = {end: {x: 7, y: 3}, direction: Direction.up};
}
}
}, []);
return (
<div className={`w-fit ${className}`}>
{
map.map((row, rowIndex) =>
<div key={rowIndex} className={"flex"}>
{
row.map((tile, colIndex) =>
<GameTile
key={colIndex + rowIndex * colIndex}
type={tile}
possiblePath={possiblePositions.find(p => p.end.x === colIndex && p.end.y === rowIndex)}
character={characters.find(c => c.isAt({x: colIndex, y: rowIndex}))}
isSelected={selectedCharacter?.isAt({x: colIndex, y: rowIndex})}
showPath={hoveredPosition?.path?.find(pos => pos.x === colIndex && pos.y === rowIndex) !== undefined}
handleMoveCharacter={handleMoveCharacter}
handleSelectCharacter={handleSelectCharacter}
handleStartShowPath={handleShowPath}
handleStopShowPath={() => setHoveredPosition(undefined)}/>
)
}
</div>)
}
</div>
);
};
export default Board;