Score and pellets should update on other clients
This commit is contained in:
parent
2543581211
commit
63c531635e
@ -4,12 +4,12 @@ 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 {TileType} from "../game/tileType";
|
||||||
import {NormalPellet, PowerPellet} from "../game/pellet";
|
import Pellet from "../game/pellet";
|
||||||
|
|
||||||
interface BoardProps extends ComponentProps {
|
interface BoardProps extends ComponentProps {
|
||||||
characters: Character[],
|
characters: Character[],
|
||||||
selectedDice?: SelectedDice,
|
selectedDice?: SelectedDice,
|
||||||
onMove?: Action<Character>,
|
onMove?: BiAction<Character, Position[]>,
|
||||||
map: GameMap
|
map: GameMap
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -38,28 +38,33 @@ const Board: Component<BoardProps> = (
|
|||||||
if (selectedCharacter) {
|
if (selectedCharacter) {
|
||||||
setHoveredPosition(undefined);
|
setHoveredPosition(undefined);
|
||||||
selectedCharacter.follow(destination);
|
selectedCharacter.follow(destination);
|
||||||
|
|
||||||
pickUpPellets(destination);
|
const positions = pickUpPellets(destination);
|
||||||
onMove?.(selectedCharacter);
|
onMove?.(selectedCharacter, positions);
|
||||||
setSelectedCharacter(undefined);
|
setSelectedCharacter(undefined);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function pickUpPellets(destination: Path): void {
|
function pickUpPellets(destination: Path): Position[] {
|
||||||
|
const positions: Position[] = [];
|
||||||
if (selectedCharacter instanceof PacMan) {
|
if (selectedCharacter instanceof PacMan) {
|
||||||
const pacMan = selectedCharacter as PacMan;
|
const pacMan = selectedCharacter as PacMan;
|
||||||
|
|
||||||
for (const tile of [...destination.path ?? [], destination.end]) {
|
for (const tile of [...destination.path ?? [], destination.end]) {
|
||||||
const currentTile = map[tile.y][tile.x];
|
const currentTile = map[tile.y][tile.x];
|
||||||
|
|
||||||
if (currentTile === TileType.pellet) {
|
if (currentTile === TileType.pellet) {
|
||||||
pacMan.box.addPellet(new NormalPellet());
|
pacMan.box.addPellet(new Pellet());
|
||||||
map[tile.y][tile.x] = TileType.empty;
|
map[tile.y][tile.x] = TileType.empty;
|
||||||
|
positions.push(tile);
|
||||||
} else if (currentTile === TileType.powerPellet) {
|
} else if (currentTile === TileType.powerPellet) {
|
||||||
pacMan.box.addPellet(new PowerPellet());
|
pacMan.box.addPellet(new Pellet(true));
|
||||||
map[tile.y][tile.x] = TileType.empty;
|
map[tile.y][tile.x] = TileType.empty;
|
||||||
|
positions.push(tile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return positions;
|
||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -6,15 +6,19 @@ 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 {Direction} from "../game/direction";
|
||||||
import Box from "../game/box";
|
import {TileType} from "../game/tileType";
|
||||||
|
|
||||||
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 = () => {
|
||||||
// TODO find spawn points
|
// TODO find spawn points
|
||||||
const [characters, setCharacters] = useState([
|
const [characters, setCharacters] = useState([
|
||||||
new PacMan("yellow", {at: {x: 3, y: 3}, direction: Direction.up}),
|
new PacMan({
|
||||||
new Ghost("purple", {at: {x: 8, y: 3}, direction: Direction.up})
|
colour: "yellow", spawnPosition: {at: {x: 3, y: 3}, direction: Direction.up}
|
||||||
|
}),
|
||||||
|
new Ghost({
|
||||||
|
colour: "purple", spawnPosition: {at: {x: 8, y: 3}, direction: Direction.up}
|
||||||
|
})
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const [dice, setDice] = useState<number[]>();
|
const [dice, setDice] = useState<number[]>();
|
||||||
@ -29,11 +33,11 @@ export const GameComponent: Component = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function startGameLoop(): void {
|
function startGameLoop(): void {
|
||||||
setSelectedDice(undefined);
|
|
||||||
if (!wsService.isOpen()) {
|
if (!wsService.isOpen()) {
|
||||||
setTimeout(startGameLoop, 50);
|
setTimeout(startGameLoop, 50);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
setSelectedDice(undefined);
|
||||||
rollDice();
|
rollDice();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -46,23 +50,34 @@ export const GameComponent: Component = () => {
|
|||||||
break;
|
break;
|
||||||
case GameAction.moveCharacter:
|
case GameAction.moveCharacter:
|
||||||
setDice(parsed.Data?.dice as number[]);
|
setDice(parsed.Data?.dice as number[]);
|
||||||
const character = parsed.Data?.character satisfies Ghost | PacMan;
|
updateCharacter(parsed);
|
||||||
const currentCharacter = characters.find(c => c.color === character.color);
|
removeEatenPellets(parsed);
|
||||||
|
|
||||||
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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function onCharacterMove(character: Character): void {
|
function removeEatenPellets(parsed: ActionMessage): void {
|
||||||
|
const pellets = parsed.Data?.eatenPellets as Position[];
|
||||||
|
|
||||||
|
for (const pellet of pellets){
|
||||||
|
testMap[pellet.y][pellet.x] = TileType.empty;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateCharacter(parsed: ActionMessage): void {
|
||||||
|
const updatedCharacter = parsed.Data?.character satisfies Ghost | PacMan;
|
||||||
|
const characterIndex = characters.findIndex(c => c.colour === updatedCharacter.colour);
|
||||||
|
|
||||||
|
if (characters[characterIndex]) {
|
||||||
|
if (updatedCharacter satisfies PacMan) {
|
||||||
|
(characters[characterIndex] as PacMan) = new PacMan(updatedCharacter);
|
||||||
|
} else if (updatedCharacter satisfies Ghost) {
|
||||||
|
(characters[characterIndex] as Ghost) = new Ghost(updatedCharacter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCharacterMove(character: Character, eatenPellets: Position[]): void {
|
||||||
if (dice && selectedDice) {
|
if (dice && selectedDice) {
|
||||||
dice.splice(selectedDice.index, 1);
|
dice.splice(selectedDice.index, 1);
|
||||||
}
|
}
|
||||||
@ -71,7 +86,8 @@ export const GameComponent: Component = () => {
|
|||||||
Action: GameAction.moveCharacter,
|
Action: GameAction.moveCharacter,
|
||||||
Data: {
|
Data: {
|
||||||
dice: dice?.length ?? 0 > 0 ? dice : null,
|
dice: dice?.length ?? 0 > 0 ? dice : null,
|
||||||
character: character
|
character: character,
|
||||||
|
eatenPellets: eatenPellets
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
wsService.send(data);
|
wsService.send(data);
|
||||||
@ -94,8 +110,8 @@ export const GameComponent: Component = () => {
|
|||||||
<AllDice values={dice} onclick={handleDiceClick} selectedDiceIndex={selectedDice?.index}/>
|
<AllDice values={dice} onclick={handleDiceClick} selectedDiceIndex={selectedDice?.index}/>
|
||||||
{
|
{
|
||||||
(characters.filter(c => c instanceof PacMan) as PacMan[]).map(c =>
|
(characters.filter(c => c instanceof PacMan) as PacMan[]).map(c =>
|
||||||
<div key={c.color} className={"mx-auto w-fit m-2"}>
|
<div key={c.colour} className={"mx-auto w-fit m-2"}>
|
||||||
<p>Pac-Man: {c.color}</p>
|
<p>Pac-Man: {c.colour}</p>
|
||||||
<p>Pellets: {c.box.count}</p>
|
<p>Pellets: {c.box.count}</p>
|
||||||
<p>PowerPellets: {c.box.countPowerPellets}</p>
|
<p>PowerPellets: {c.box.countPowerPellets}</p>
|
||||||
</div>)
|
</div>)
|
||||||
|
@ -35,7 +35,7 @@ export const GameTile: Component<TileWithCharacterProps> = (
|
|||||||
onMouseLeave={handleStopShowPath}>
|
onMouseLeave={handleStopShowPath}>
|
||||||
<>
|
<>
|
||||||
{character &&
|
{character &&
|
||||||
<div className={"flex-center w-full h-full"}>
|
<div className={"flex-center wh-full"}>
|
||||||
<CharacterComponent
|
<CharacterComponent
|
||||||
character={character}
|
character={character}
|
||||||
onClick={handleSelectCharacter}
|
onClick={handleSelectCharacter}
|
||||||
@ -60,11 +60,11 @@ const Circle: Component<CircleProps> = ({colour = "white"}) => (
|
|||||||
|
|
||||||
interface TileProps extends ChildProps {
|
interface TileProps extends ChildProps {
|
||||||
type?: TileType,
|
type?: TileType,
|
||||||
onClick?: () => void,
|
onClick?: VoidFunction,
|
||||||
onMouseEnter?: () => void,
|
onMouseEnter?: VoidFunction,
|
||||||
onMouseLeave?: () => void,
|
onMouseLeave?: VoidFunction,
|
||||||
character?: Character,
|
character?: Character,
|
||||||
onCharacterClick?: (character: Character) => void,
|
onCharacterClick?: Action<Character>,
|
||||||
characterClass?: string,
|
characterClass?: string,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -112,9 +112,9 @@ const Tile: Component<TileProps> = (
|
|||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
onMouseEnter={onMouseEnter}
|
onMouseEnter={onMouseEnter}
|
||||||
onMouseLeave={onMouseLeave}>
|
onMouseLeave={onMouseLeave}>
|
||||||
|
{children}
|
||||||
{type === TileType.pellet && <Circle colour={"yellow"}/>}
|
{type === TileType.pellet && <Circle colour={"yellow"}/>}
|
||||||
{type === TileType.powerPellet && <Circle colour={"red"}/>}
|
{type === TileType.powerPellet && <Circle colour={"red"}/>}
|
||||||
{children}
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
@ -126,8 +126,8 @@ interface AddDummyProps extends ComponentProps {
|
|||||||
const AddDummy: Component<AddDummyProps> = ({path}) => (
|
const AddDummy: Component<AddDummyProps> = ({path}) => (
|
||||||
<>
|
<>
|
||||||
{path &&
|
{path &&
|
||||||
<div className={"flex-center w-full h-full"}>
|
<div className={"flex-center wh-full"}>
|
||||||
<CharacterComponent character={new Dummy({at: path.end, direction: path.direction})}/>
|
<CharacterComponent character={new Dummy(path)}/>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
</>
|
</>
|
||||||
@ -135,7 +135,7 @@ const AddDummy: Component<AddDummyProps> = ({path}) => (
|
|||||||
|
|
||||||
interface CharacterComponentProps extends ComponentProps {
|
interface CharacterComponentProps extends ComponentProps {
|
||||||
character?: Character,
|
character?: Character,
|
||||||
onClick?: (character: Character) => void,
|
onClick?: Action<Character>,
|
||||||
}
|
}
|
||||||
|
|
||||||
const CharacterComponent: Component<CharacterComponentProps> = (
|
const CharacterComponent: Component<CharacterComponentProps> = (
|
||||||
@ -162,7 +162,7 @@ const CharacterComponent: Component<CharacterComponentProps> = (
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={`rounded-full w-4/5 h-4/5 cursor-pointer hover:border border-black relative ${className}`}
|
<div className={`rounded-full w-4/5 h-4/5 cursor-pointer hover:border border-black relative ${className}`}
|
||||||
style={{backgroundColor: `${character.color}`}}
|
style={{backgroundColor: `${character.colour}`}}
|
||||||
onClick={() => onClick?.(character)}>
|
onClick={() => onClick?.(character)}>
|
||||||
<div>
|
<div>
|
||||||
<div className={`absolute ${getSide()} w-1/2 h-1/2 rounded-full bg-black`}/>
|
<div className={`absolute ${getSide()} w-1/2 h-1/2 rounded-full bg-black`}/>
|
||||||
|
@ -1,10 +1,15 @@
|
|||||||
import {NormalPellet, Pellet, PowerPellet} from "./pellet";
|
import Pellet from "./pellet";
|
||||||
|
|
||||||
|
export interface BoxProps {
|
||||||
|
pellets?: Pellet[],
|
||||||
|
readonly colour: Colour,
|
||||||
|
}
|
||||||
|
|
||||||
export default class Box {
|
export default class Box {
|
||||||
private pellets: Pellet[] = [];
|
public pellets: Pellet[];
|
||||||
public readonly colour: Colour;
|
public readonly colour: Colour;
|
||||||
|
|
||||||
public constructor(colour: Colour, pellets: Pellet[] = []) {
|
public constructor({colour, pellets = []}: BoxProps) {
|
||||||
this.colour = colour;
|
this.colour = colour;
|
||||||
this.pellets = pellets;
|
this.pellets = pellets;
|
||||||
}
|
}
|
||||||
@ -14,15 +19,15 @@ export default class Box {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get powerPellet(): Pellet | undefined {
|
get powerPellet(): Pellet | undefined {
|
||||||
return this.pellets.find(pellet => pellet instanceof PowerPellet);
|
return this.pellets.find(pellet => pellet.isPowerPellet);
|
||||||
}
|
}
|
||||||
|
|
||||||
get count(): number {
|
get count(): number {
|
||||||
return this.pellets.filter(pellet => pellet instanceof NormalPellet).length;
|
return this.pellets.filter(pellet => !pellet.isPowerPellet).length;
|
||||||
}
|
}
|
||||||
|
|
||||||
get countPowerPellets(): number {
|
get countPowerPellets(): number {
|
||||||
return this.pellets.filter(pellet => pellet instanceof PowerPellet).length;
|
return this.pellets.filter(pellet => pellet.isPowerPellet).length;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,23 @@
|
|||||||
import Box from "./box";
|
import Box, {BoxProps} from "./box";
|
||||||
|
import {Direction} from "./direction";
|
||||||
|
|
||||||
|
interface CharacterProps {
|
||||||
|
colour: Colour,
|
||||||
|
position?: Path,
|
||||||
|
isEatable?: boolean,
|
||||||
|
spawnPosition: DirectionalPosition
|
||||||
|
}
|
||||||
|
|
||||||
export abstract class Character {
|
export abstract class Character {
|
||||||
public color: Colour;
|
public readonly colour: Colour;
|
||||||
public position: Path;
|
public position: Path;
|
||||||
public isEatable = false;
|
public isEatable: boolean;
|
||||||
public readonly spawnPosition: DirectionalPosition;
|
public readonly spawnPosition: DirectionalPosition;
|
||||||
|
|
||||||
protected constructor(color: Colour, spawnPosition: DirectionalPosition) {
|
protected constructor({colour, position, isEatable = false, spawnPosition}: CharacterProps) {
|
||||||
this.color = color;
|
this.colour = colour;
|
||||||
this.position = {end: spawnPosition.at, direction: spawnPosition.direction};
|
this.position = position ?? {end: spawnPosition.at, direction: spawnPosition.direction};
|
||||||
|
this.isEatable = isEatable;
|
||||||
this.spawnPosition = spawnPosition;
|
this.spawnPosition = spawnPosition;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -18,34 +27,46 @@ export abstract class Character {
|
|||||||
this.position.path = undefined;
|
this.position.path = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public moveToSpawn(): void {
|
||||||
|
this.follow({end: this.spawnPosition.at, direction: this.spawnPosition.direction});
|
||||||
|
}
|
||||||
|
|
||||||
public isAt(position: Position): boolean {
|
public isAt(position: Position): boolean {
|
||||||
return this.position.end.x === position.x && this.position.end.y === position.y;
|
return this.position.end.x === position.x && this.position.end.y === position.y;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface PacManProps extends CharacterProps {
|
||||||
|
box?: BoxProps,
|
||||||
|
}
|
||||||
|
|
||||||
export class PacMan extends Character {
|
export class PacMan extends Character {
|
||||||
|
|
||||||
public box: Box;
|
public box: Box;
|
||||||
|
|
||||||
public constructor(color: Colour, spawnPosition: DirectionalPosition) {
|
public constructor({colour, position, isEatable = true, spawnPosition, box = {colour}}: PacManProps) {
|
||||||
super(color, spawnPosition);
|
super({colour, position, isEatable, spawnPosition});
|
||||||
this.isEatable = true;
|
this.isEatable = isEatable;
|
||||||
this.box = new Box(color);
|
this.box = new Box(box);
|
||||||
|
}
|
||||||
|
|
||||||
|
public stealFrom(other: PacMan): void {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Ghost extends Character {
|
export class Ghost extends Character {
|
||||||
|
|
||||||
public constructor(color: Colour, spawnPosition: DirectionalPosition) {
|
public constructor({colour, position, isEatable, spawnPosition}: CharacterProps) {
|
||||||
super(color, spawnPosition);
|
super({colour, position, isEatable, spawnPosition});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Dummy extends Character {
|
export class Dummy extends Character {
|
||||||
|
|
||||||
public constructor(position: DirectionalPosition) {
|
public constructor(position: Path) { // TODO see-through
|
||||||
super("grey", position);
|
super({colour: "grey", position, isEatable: false, spawnPosition: {at: {x: 0, y: 0}, direction: Direction.up}});
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,20 +1,7 @@
|
|||||||
export abstract class Pellet {
|
export default class Pellet {
|
||||||
public readonly colour: Colour;
|
public readonly isPowerPellet: boolean;
|
||||||
|
|
||||||
protected constructor(colour: Colour) {
|
public constructor(isPowerPellet = false) {
|
||||||
this.colour = colour;
|
this.isPowerPellet = isPowerPellet;
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
export class NormalPellet extends Pellet {
|
|
||||||
public constructor() {
|
|
||||||
super("white");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export class PowerPellet extends Pellet {
|
|
||||||
public constructor() {
|
|
||||||
super("yellow");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -11,6 +11,10 @@
|
|||||||
@apply flex justify-center items-center;
|
@apply flex justify-center items-center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.wh-full {
|
||||||
|
@apply w-full h-full;
|
||||||
|
}
|
||||||
|
|
||||||
h1 {
|
h1 {
|
||||||
@apply text-4xl;
|
@apply text-4xl;
|
||||||
}
|
}
|
||||||
|
@ -11,6 +11,8 @@ type ActionMessage<T = any> = {
|
|||||||
|
|
||||||
type Action<T> = (obj: T) => void;
|
type Action<T> = (obj: T) => void;
|
||||||
|
|
||||||
|
type BiAction<T1, T2> = (obj1: T1, obj2: T2) => void;
|
||||||
|
|
||||||
type SelectedDice = {
|
type SelectedDice = {
|
||||||
value: number,
|
value: number,
|
||||||
index: number
|
index: number
|
||||||
|
Loading…
x
Reference in New Issue
Block a user