PacMan has a box containing pellets, pellets will be removed from map, when picked up, info and counter at top
This commit is contained in:
parent
69ec14c04c
commit
5cffdf1762
@ -3,11 +3,13 @@ import {Character, PacMan} from "../game/character";
|
|||||||
import findPossiblePositions from "../game/possibleMovesAlgorithm";
|
import findPossiblePositions from "../game/possibleMovesAlgorithm";
|
||||||
import {Direction} from "../game/direction";
|
import {Direction} from "../game/direction";
|
||||||
import {GameTile} from "./gameTile";
|
import {GameTile} from "./gameTile";
|
||||||
|
import {TileType} from "../game/tileType";
|
||||||
|
import {NormalPellet, PowerPellet} from "../game/pellet";
|
||||||
|
|
||||||
interface BoardProps extends ComponentProps {
|
interface BoardProps extends ComponentProps {
|
||||||
characters: Character[],
|
characters: Character[],
|
||||||
selectedDice?: SelectedDice,
|
selectedDice?: SelectedDice,
|
||||||
onMove?: (character: Character) => void,
|
onMove?: Action<Character>,
|
||||||
map: GameMap
|
map: GameMap
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -32,15 +34,34 @@ const Board: Component<BoardProps> = (
|
|||||||
setHoveredPosition(path);
|
setHoveredPosition(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleMoveCharacter(path: Path): void {
|
function handleMoveCharacter(destination: Path): void {
|
||||||
if (selectedCharacter) {
|
if (selectedCharacter) {
|
||||||
setHoveredPosition(undefined);
|
setHoveredPosition(undefined);
|
||||||
selectedCharacter.follow(path);
|
selectedCharacter.follow(destination);
|
||||||
|
|
||||||
|
pickUpPellets(destination);
|
||||||
onMove?.(selectedCharacter);
|
onMove?.(selectedCharacter);
|
||||||
setSelectedCharacter(undefined);
|
setSelectedCharacter(undefined);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function pickUpPellets(destination: Path): void {
|
||||||
|
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 NormalPellet());
|
||||||
|
map[tile.y][tile.x] = TileType.empty;
|
||||||
|
} else if (currentTile === TileType.powerPellet) {
|
||||||
|
pacMan.box.addPellet(new PowerPellet());
|
||||||
|
map[tile.y][tile.x] = TileType.empty;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (selectedCharacter && selectedDice) {
|
if (selectedCharacter && selectedDice) {
|
||||||
const possiblePaths = findPossiblePositions(map, selectedCharacter, selectedDice.value);
|
const possiblePaths = findPossiblePositions(map, selectedCharacter, selectedDice.value);
|
||||||
|
@ -1,16 +1,21 @@
|
|||||||
import React, {useEffect, useRef, useState} from "react";
|
import React, {useEffect, useState} from "react";
|
||||||
import {AllDice} from "./dice";
|
import {AllDice} from "./dice";
|
||||||
import {Action} from "../websockets/actions";
|
import {GameAction} 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";
|
import WebSocketService from "../websockets/WebSocketService";
|
||||||
import {testMap} from "../game/map";
|
import {testMap} from "../game/map";
|
||||||
|
import {Direction} from "../game/direction";
|
||||||
|
import Box from "../game/box";
|
||||||
|
|
||||||
const wsService = new WebSocketService("wss://localhost:3000/api/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
|
// TODO find spawn points
|
||||||
const characters = useRef([new PacMan("yellow"), new Ghost("purple")]);
|
const [characters, setCharacters] = useState([
|
||||||
|
new PacMan("yellow", {at: {x: 3, y: 3}, direction: Direction.up}),
|
||||||
|
new Ghost("purple", {at: {x: 8, y: 3}, direction: Direction.up})
|
||||||
|
]);
|
||||||
|
|
||||||
const [dice, setDice] = useState<number[]>();
|
const [dice, setDice] = useState<number[]>();
|
||||||
const [selectedDice, setSelectedDice] = useState<SelectedDice>();
|
const [selectedDice, setSelectedDice] = useState<SelectedDice>();
|
||||||
@ -20,7 +25,7 @@ export const GameComponent: Component = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function rollDice(): void {
|
function rollDice(): void {
|
||||||
wsService.send({Action: Action.rollDice});
|
wsService.send({Action: GameAction.rollDice});
|
||||||
}
|
}
|
||||||
|
|
||||||
function startGameLoop(): void {
|
function startGameLoop(): void {
|
||||||
@ -36,13 +41,23 @@ export const GameComponent: Component = () => {
|
|||||||
const parsed: ActionMessage = JSON.parse(message.data);
|
const parsed: ActionMessage = JSON.parse(message.data);
|
||||||
|
|
||||||
switch (parsed.Action) {
|
switch (parsed.Action) {
|
||||||
case Action.rollDice:
|
case GameAction.rollDice:
|
||||||
setDice(parsed.Data as number[]);
|
setDice(parsed.Data as number[]);
|
||||||
break;
|
break;
|
||||||
case Action.moveCharacter:
|
case GameAction.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 satisfies Ghost | PacMan;
|
||||||
characters.current.find(c => c.color === character.color)?.follow(character.position);
|
const currentCharacter = characters.find(c => c.color === character.color);
|
||||||
|
|
||||||
|
if (currentCharacter) {
|
||||||
|
currentCharacter.position = character.position;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO update pellets on other clients (character and on map)
|
||||||
|
// if (character satisfies PacMan) {
|
||||||
|
// (characters[currentCharacter] as PacMan).box = new Box(character.box.colour, character.box.pellets);
|
||||||
|
// console.log(characters[currentCharacter]);
|
||||||
|
// }
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -53,7 +68,7 @@ export const GameComponent: Component = () => {
|
|||||||
}
|
}
|
||||||
setSelectedDice(undefined);
|
setSelectedDice(undefined);
|
||||||
const data: ActionMessage = {
|
const data: ActionMessage = {
|
||||||
Action: Action.moveCharacter,
|
Action: GameAction.moveCharacter,
|
||||||
Data: {
|
Data: {
|
||||||
dice: dice?.length ?? 0 > 0 ? dice : null,
|
dice: dice?.length ?? 0 > 0 ? dice : null,
|
||||||
character: character
|
character: character
|
||||||
@ -73,11 +88,19 @@ export const GameComponent: Component = () => {
|
|||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<h1 className={"w-fit mx-auto"}>Pac-Man The Board Game</h1>
|
<h1 className={"w-fit mx-auto"}>Pac-Man The Board Game</h1>
|
||||||
<div className={"flex justify-center"}>
|
<div className={"flex-center"}>
|
||||||
<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 my-2"} characters={characters.current} selectedDice={selectedDice}
|
{
|
||||||
|
(characters.filter(c => c instanceof PacMan) as PacMan[]).map(c =>
|
||||||
|
<div key={c.color} className={"mx-auto w-fit m-2"}>
|
||||||
|
<p>Pac-Man: {c.color}</p>
|
||||||
|
<p>Pellets: {c.box.count}</p>
|
||||||
|
<p>PowerPellets: {c.box.countPowerPellets}</p>
|
||||||
|
</div>)
|
||||||
|
}
|
||||||
|
<GameBoard className={"mx-auto my-2"} characters={characters} selectedDice={selectedDice}
|
||||||
onMove={onCharacterMove} map={testMap}/>
|
onMove={onCharacterMove} map={testMap}/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -2,15 +2,16 @@ import React, {useEffect, useState} from "react";
|
|||||||
import {TileType} from "../game/tileType";
|
import {TileType} from "../game/tileType";
|
||||||
import {Character, Dummy} from "../game/character";
|
import {Character, Dummy} from "../game/character";
|
||||||
import {Direction} from "../game/direction";
|
import {Direction} from "../game/direction";
|
||||||
|
import {getCSSColour} from "../utils/colours";
|
||||||
|
|
||||||
interface TileWithCharacterProps extends ComponentProps {
|
interface TileWithCharacterProps extends ComponentProps {
|
||||||
possiblePath?: Path,
|
possiblePath?: Path,
|
||||||
character?: Character,
|
character?: Character,
|
||||||
type?: TileType,
|
type?: TileType,
|
||||||
handleMoveCharacter?: (path: Path) => void,
|
handleMoveCharacter?: Action<Path>,
|
||||||
handleSelectCharacter?: (character: Character) => void,
|
handleSelectCharacter?: Action<Character>,
|
||||||
handleStartShowPath?: (path: Path) => void,
|
handleStartShowPath?: Action<Path>,
|
||||||
handleStopShowPath?: () => void,
|
handleStopShowPath?: VoidFunction,
|
||||||
isSelected?: boolean,
|
isSelected?: boolean,
|
||||||
showPath?: boolean
|
showPath?: boolean
|
||||||
}
|
}
|
||||||
@ -26,32 +27,34 @@ export const GameTile: Component<TileWithCharacterProps> = (
|
|||||||
handleStopShowPath,
|
handleStopShowPath,
|
||||||
isSelected = false,
|
isSelected = false,
|
||||||
showPath = false
|
showPath = false
|
||||||
}) => {
|
}) => (
|
||||||
return (
|
<Tile className={`${possiblePath?.end ? "border-4 border-white" : ""}`}
|
||||||
<Tile className={`${possiblePath?.end ? "border-4 border-white" : ""}`}
|
type={type}
|
||||||
type={type}
|
onClick={possiblePath ? () => handleMoveCharacter?.(possiblePath) : undefined}
|
||||||
onClick={possiblePath ? () => handleMoveCharacter?.(possiblePath) : undefined}
|
onMouseEnter={possiblePath ? () => handleStartShowPath?.(possiblePath) : undefined}
|
||||||
onMouseEnter={possiblePath ? () => handleStartShowPath?.(possiblePath) : undefined}
|
onMouseLeave={handleStopShowPath}>
|
||||||
onMouseLeave={handleStopShowPath}>
|
<>
|
||||||
<>
|
{character &&
|
||||||
{character &&
|
<div className={"flex-center w-full h-full"}>
|
||||||
<div className={"flex-center w-full h-full"}>
|
<CharacterComponent
|
||||||
<CharacterComponent
|
character={character}
|
||||||
character={character}
|
onClick={handleSelectCharacter}
|
||||||
onClick={handleSelectCharacter}
|
className={isSelected ? "animate-bounce" : ""}/>
|
||||||
className={isSelected ? "animate-bounce" : ""}/>
|
</div>
|
||||||
</div>
|
}
|
||||||
}
|
{showPath && <Circle/>}
|
||||||
{showPath && <PathSymbol/>}
|
<AddDummy path={possiblePath}/>
|
||||||
<AddDummy path={possiblePath}/>
|
</>
|
||||||
</>
|
</Tile>
|
||||||
</Tile>
|
);
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const PathSymbol: Component = () => ( // TODO sometimes shows up when it shouldn't
|
interface CircleProps extends ComponentProps {
|
||||||
|
colour?: Colour,
|
||||||
|
}
|
||||||
|
|
||||||
|
const Circle: Component<CircleProps> = ({colour = "white"}) => (
|
||||||
<div className={"flex-center w-full h-full"}>
|
<div className={"flex-center w-full h-full"}>
|
||||||
<div className={"w-1/2 h-1/2 rounded-full bg-white"}/>
|
<div className={`w-1/2 h-1/2 rounded-full ${getCSSColour(colour)}`}/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -79,18 +82,14 @@ const Tile: Component<TileProps> = (
|
|||||||
|
|
||||||
function setColor(): string {
|
function setColor(): string {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case TileType.empty:
|
|
||||||
return "bg-black";
|
|
||||||
case TileType.wall:
|
case TileType.wall:
|
||||||
return "bg-blue-500";
|
return "bg-blue-500";
|
||||||
case TileType.pellet:
|
|
||||||
return "bg-yellow-500";
|
|
||||||
case TileType.powerPellet:
|
|
||||||
return "bg-orange-500";
|
|
||||||
case TileType.ghostSpawn:
|
case TileType.ghostSpawn:
|
||||||
return "bg-red-500";
|
return "bg-red-500";
|
||||||
case TileType.pacmanSpawn:
|
case TileType.pacmanSpawn:
|
||||||
return "bg-green-500";
|
return "bg-green-500";
|
||||||
|
default:
|
||||||
|
return "bg-black";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -113,6 +112,8 @@ const Tile: Component<TileProps> = (
|
|||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
onMouseEnter={onMouseEnter}
|
onMouseEnter={onMouseEnter}
|
||||||
onMouseLeave={onMouseLeave}>
|
onMouseLeave={onMouseLeave}>
|
||||||
|
{type === TileType.pellet && <Circle colour={"yellow"}/>}
|
||||||
|
{type === TileType.powerPellet && <Circle colour={"red"}/>}
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@ -126,7 +127,7 @@ const AddDummy: Component<AddDummyProps> = ({path}) => (
|
|||||||
<>
|
<>
|
||||||
{path &&
|
{path &&
|
||||||
<div className={"flex-center w-full h-full"}>
|
<div className={"flex-center w-full h-full"}>
|
||||||
<CharacterComponent character={new Dummy(path)}/>
|
<CharacterComponent character={new Dummy({at: path.end, direction: path.direction})}/>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
</>
|
</>
|
||||||
|
28
pac-man-board-game/ClientApp/src/game/box.ts
Normal file
28
pac-man-board-game/ClientApp/src/game/box.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import {NormalPellet, Pellet, PowerPellet} from "./pellet";
|
||||||
|
|
||||||
|
export default class Box {
|
||||||
|
private pellets: Pellet[] = [];
|
||||||
|
public readonly colour: Colour;
|
||||||
|
|
||||||
|
public constructor(colour: Colour, pellets: Pellet[] = []) {
|
||||||
|
this.colour = colour;
|
||||||
|
this.pellets = pellets;
|
||||||
|
}
|
||||||
|
|
||||||
|
public addPellet(pellet: Pellet): void {
|
||||||
|
this.pellets.push(pellet);
|
||||||
|
}
|
||||||
|
|
||||||
|
get powerPellet(): Pellet | undefined {
|
||||||
|
return this.pellets.find(pellet => pellet instanceof PowerPellet);
|
||||||
|
}
|
||||||
|
|
||||||
|
get count(): number {
|
||||||
|
return this.pellets.filter(pellet => pellet instanceof NormalPellet).length;
|
||||||
|
}
|
||||||
|
|
||||||
|
get countPowerPellets(): number {
|
||||||
|
return this.pellets.filter(pellet => pellet instanceof PowerPellet).length;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,20 +1,15 @@
|
|||||||
import {Direction} from "./direction";
|
import Box from "./box";
|
||||||
|
|
||||||
type CharacterColor = "red" | "blue" | "yellow" | "green" | "purple" | "grey";
|
|
||||||
|
|
||||||
const defaultDirection: Path = {
|
|
||||||
end: {x: 0, y: 0},
|
|
||||||
direction: Direction.up
|
|
||||||
};
|
|
||||||
|
|
||||||
export abstract class Character {
|
export abstract class Character {
|
||||||
public color: CharacterColor;
|
public color: Colour;
|
||||||
public position: Path;
|
public position: Path;
|
||||||
public isEatable: boolean = false;
|
public isEatable = false;
|
||||||
|
public readonly spawnPosition: DirectionalPosition;
|
||||||
|
|
||||||
protected constructor(color: CharacterColor, startPosition = defaultDirection) {
|
protected constructor(color: Colour, spawnPosition: DirectionalPosition) {
|
||||||
this.color = color;
|
this.color = color;
|
||||||
this.position = startPosition;
|
this.position = {end: spawnPosition.at, direction: spawnPosition.direction};
|
||||||
|
this.spawnPosition = spawnPosition;
|
||||||
}
|
}
|
||||||
|
|
||||||
public follow(path: Path): void {
|
public follow(path: Path): void {
|
||||||
@ -30,24 +25,27 @@ export abstract class Character {
|
|||||||
|
|
||||||
export class PacMan extends Character {
|
export class PacMan extends Character {
|
||||||
|
|
||||||
constructor(color: CharacterColor, startPosition = defaultDirection) {
|
public box: Box;
|
||||||
super(color, startPosition);
|
|
||||||
|
public constructor(color: Colour, spawnPosition: DirectionalPosition) {
|
||||||
|
super(color, spawnPosition);
|
||||||
this.isEatable = true;
|
this.isEatable = true;
|
||||||
|
this.box = new Box(color);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Ghost extends Character {
|
export class Ghost extends Character {
|
||||||
|
|
||||||
constructor(color: CharacterColor, startPosition = defaultDirection) {
|
public constructor(color: Colour, spawnPosition: DirectionalPosition) {
|
||||||
super(color, startPosition);
|
super(color, spawnPosition);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Dummy extends Character {
|
export class Dummy extends Character {
|
||||||
|
|
||||||
constructor(path: Path) {
|
public constructor(position: DirectionalPosition) {
|
||||||
super("grey", path);
|
super("grey", position);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
20
pac-man-board-game/ClientApp/src/game/pellet.ts
Normal file
20
pac-man-board-game/ClientApp/src/game/pellet.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
export abstract class Pellet {
|
||||||
|
public readonly colour: Colour;
|
||||||
|
|
||||||
|
protected constructor(colour: Colour) {
|
||||||
|
this.colour = colour;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export class NormalPellet extends Pellet {
|
||||||
|
public constructor() {
|
||||||
|
super("white");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class PowerPellet extends Pellet {
|
||||||
|
public constructor() {
|
||||||
|
super("yellow");
|
||||||
|
}
|
||||||
|
}
|
@ -5,10 +5,12 @@ type Setter<T> = React.Dispatch<React.SetStateAction<T>>;
|
|||||||
type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView;
|
type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView;
|
||||||
|
|
||||||
type ActionMessage<T = any> = {
|
type ActionMessage<T = any> = {
|
||||||
Action: import("../websockets/actions").Action,
|
Action: import("../websockets/actions").GameAction,
|
||||||
Data?: T
|
Data?: T
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Action<T> = (obj: T) => void;
|
||||||
|
|
||||||
type SelectedDice = {
|
type SelectedDice = {
|
||||||
value: number,
|
value: number,
|
||||||
index: number
|
index: number
|
||||||
@ -28,3 +30,5 @@ type Path = {
|
|||||||
end: Position,
|
end: Position,
|
||||||
direction: import("../game/direction").Direction
|
direction: import("../game/direction").Direction
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Colour = "white" | "red" | "blue" | "yellow" | "green" | "purple" | "grey";
|
||||||
|
27
pac-man-board-game/ClientApp/src/utils/colours.ts
Normal file
27
pac-man-board-game/ClientApp/src/utils/colours.ts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
export function getCSSColour(colour: Colour): string {
|
||||||
|
let tailwindColour: string;
|
||||||
|
switch (colour) {
|
||||||
|
case "red":
|
||||||
|
tailwindColour = "bg-red-500";
|
||||||
|
break;
|
||||||
|
case "blue":
|
||||||
|
tailwindColour = "bg-blue-500";
|
||||||
|
break;
|
||||||
|
case "yellow":
|
||||||
|
tailwindColour = "bg-yellow-500";
|
||||||
|
break;
|
||||||
|
case "green":
|
||||||
|
tailwindColour = "bg-green-500";
|
||||||
|
break;
|
||||||
|
case "purple":
|
||||||
|
tailwindColour = "bg-purple-500";
|
||||||
|
break;
|
||||||
|
case "grey":
|
||||||
|
tailwindColour = "bg-gray-500";
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
tailwindColour = "bg-white";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return tailwindColour;
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
export enum Action {
|
export enum GameAction {
|
||||||
rollDice,
|
rollDice,
|
||||||
moveCharacter,
|
moveCharacter,
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user