Moved event to Game, added leave game button
This commit is contained in:
parent
7af502f570
commit
554b8cff2a
@ -1,6 +1,5 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import {Counter} from "./pages/counter";
|
import {Counter} from "./pages/counter";
|
||||||
import Home from "./pages/home";
|
|
||||||
import Game from "./pages/game";
|
import Game from "./pages/game";
|
||||||
import LobbyPage from "./pages/lobby";
|
import LobbyPage from "./pages/lobby";
|
||||||
import Login from "./pages/login";
|
import Login from "./pages/login";
|
||||||
@ -8,7 +7,7 @@ import Login from "./pages/login";
|
|||||||
const AppRoutes = [
|
const AppRoutes = [
|
||||||
{
|
{
|
||||||
index: true,
|
index: true,
|
||||||
element: <Home/>
|
element: <LobbyPage/>
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: "/counter",
|
path: "/counter",
|
||||||
|
@ -9,19 +9,21 @@ import PlayerStats from "../components/playerStats";
|
|||||||
import {useAtom, useAtomValue, useSetAtom} from "jotai";
|
import {useAtom, useAtomValue, useSetAtom} from "jotai";
|
||||||
import {diceAtom, ghostsAtom, playersAtom, rollDiceButtonAtom, selectedDiceAtom} from "../utils/state";
|
import {diceAtom, ghostsAtom, playersAtom, rollDiceButtonAtom, selectedDiceAtom} from "../utils/state";
|
||||||
import GameButton from "./gameButton";
|
import GameButton from "./gameButton";
|
||||||
|
import {Button} from "./button";
|
||||||
|
import {useNavigate} from "react-router-dom";
|
||||||
|
|
||||||
const wsService = new WebSocketService(import.meta.env.VITE_API_WS);
|
const wsService = new WebSocketService(import.meta.env.VITE_API_WS);
|
||||||
|
|
||||||
// TODO bug, when taking player on last dice, the currentPlayer changes and the wrong character get to steal
|
// TODO bug, when taking player on last dice, the currentPlayer changes and the wrong character gets to steal
|
||||||
// TODO bug, first player can sometimes roll dice twice (maybe only on firefox)
|
// TODO bug, first player can sometimes roll dice twice
|
||||||
|
// TODO bug, when refreshing page, the player is still in disconnected state
|
||||||
// TODO bug, when refreshing page, the characters are reset for all players
|
// TODO bug, when refreshing page, the characters are reset for all players
|
||||||
// TODO bug, when refreshing page, some data is missing until other clients make a move
|
// TODO bug, when refreshing page, some data is missing until other clients make a move
|
||||||
|
// TODO bug, teleportation doesn't work
|
||||||
|
|
||||||
// TODO store map in backend and save it in state on each client
|
// TODO store map in backend and save it in state on each client
|
||||||
// TODO add debug menu on dev, for testing and cheating
|
// TODO add debug menu on dev, for testing and cheating
|
||||||
// TODO join/new game lobby
|
|
||||||
// TODO sign up player page
|
// TODO sign up player page
|
||||||
// TODO sign in page
|
|
||||||
// TODO show box with collected pellets
|
// TODO show box with collected pellets
|
||||||
// TODO layout
|
// TODO layout
|
||||||
|
|
||||||
@ -33,6 +35,8 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map
|
|||||||
const setActiveRollDiceButton = useSetAtom(rollDiceButtonAtom);
|
const setActiveRollDiceButton = useSetAtom(rollDiceButtonAtom);
|
||||||
const ghosts = useAtomValue(ghostsAtom);
|
const ghosts = useAtomValue(ghostsAtom);
|
||||||
|
|
||||||
|
const navigate = useNavigate();
|
||||||
|
|
||||||
function rollDice(): void {
|
function rollDice(): void {
|
||||||
if (!player.isTurn()) return;
|
if (!player.isTurn()) return;
|
||||||
|
|
||||||
@ -79,6 +83,11 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map
|
|||||||
wsService.send({action: GameAction.nextPlayer});
|
wsService.send({action: GameAction.nextPlayer});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function leaveGame(): void {
|
||||||
|
wsService.send({action: GameAction.disconnect});
|
||||||
|
navigate("/lobby");
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
wsService.onReceive = doAction;
|
wsService.onReceive = doAction;
|
||||||
wsService.open();
|
wsService.open();
|
||||||
@ -90,6 +99,7 @@ export const GameComponent: FC<{ player: Player, map: GameMap }> = ({player, map
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
<Button onClick={leaveGame}>Leave game</Button>
|
||||||
<div className={"flex justify-center"}>
|
<div className={"flex justify-center"}>
|
||||||
{players?.map(p => <PlayerStats key={p.username} player={p}/>)}
|
{players?.map(p => <PlayerStats key={p.username} player={p}/>)}
|
||||||
</div>
|
</div>
|
||||||
|
@ -50,6 +50,9 @@ export const doAction: MessageEventFunction<string> = (event): void => { // TODO
|
|||||||
case GameAction.nextPlayer:
|
case GameAction.nextPlayer:
|
||||||
nextPlayer(message.data);
|
nextPlayer(message.data);
|
||||||
break;
|
break;
|
||||||
|
case GameAction.disconnect:
|
||||||
|
updatePlayers(message.data);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -61,14 +64,12 @@ type MoveCharacterData = { dice: number[], players: PlayerProps[], ghosts: Chara
|
|||||||
|
|
||||||
function moveCharacter(data?: MoveCharacterData): void {
|
function moveCharacter(data?: MoveCharacterData): void {
|
||||||
store.set(diceAtom, data?.dice);
|
store.set(diceAtom, data?.dice);
|
||||||
updatePlayers(data);
|
updatePlayers(data?.players);
|
||||||
updateGhosts(data);
|
updateGhosts(data);
|
||||||
removeEatenPellets(data);
|
removeEatenPellets(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
function updatePlayers(data?: MoveCharacterData): void {
|
function updatePlayers(updatedPlayers?: PlayerProps[]): void {
|
||||||
const updatedPlayers = data?.players;
|
|
||||||
|
|
||||||
if (updatedPlayers) {
|
if (updatedPlayers) {
|
||||||
const newList: Player[] = updatedPlayers.map(p => new Player(p));
|
const newList: Player[] = updatedPlayers.map(p => new Player(p));
|
||||||
store.set(playersAtom, newList);
|
store.set(playersAtom, newList);
|
||||||
|
@ -66,6 +66,11 @@ public class GameController : GenericController
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected override Task Echo()
|
||||||
|
{
|
||||||
|
_actionService.WebSocket = WebSocket ?? throw new NullReferenceException("WebSocket is null");
|
||||||
|
return base.Echo();
|
||||||
|
}
|
||||||
|
|
||||||
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data)
|
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data)
|
||||||
{
|
{
|
||||||
@ -78,24 +83,7 @@ public class GameController : GenericController
|
|||||||
return action.ToArraySegment();
|
return action.ToArraySegment();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void Send(ArraySegment<byte> segment) => _gameService.SendToAll(segment);
|
protected override void Send(ArraySegment<byte> segment) => _actionService.SendToAll(segment);
|
||||||
|
|
||||||
protected override Task Echo()
|
protected override void Disconnect() => _actionService.Disconnect();
|
||||||
{
|
|
||||||
_gameService.Connections += WsServiceOnFire; // TODO move to ActionService
|
|
||||||
// _actionService.Game.Connections += WsServiceOnFire;
|
|
||||||
return base.Echo();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void Disconnect()
|
|
||||||
{
|
|
||||||
_gameService.Connections -= WsServiceOnFire;
|
|
||||||
_actionService.Disconnect();
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task WsServiceOnFire(ArraySegment<byte> segment)
|
|
||||||
{
|
|
||||||
if (WebSocket == null) return;
|
|
||||||
await GameService.Send(WebSocket, segment);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -31,7 +31,7 @@ public class Character : IEquatable<Character>
|
|||||||
return obj.GetType() == GetType() && Equals((Character)obj);
|
return obj.GetType() == GetType() && Equals((Character)obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override int GetHashCode() => HashCode.Combine(Colour, Position, IsEatable, SpawnPosition, (int)Type);
|
public override int GetHashCode() => HashCode.Combine(Colour, Position, IsEatable, SpawnPosition, (int?)Type);
|
||||||
}
|
}
|
||||||
|
|
||||||
public enum CharacterType
|
public enum CharacterType
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
using System.Net.WebSockets;
|
||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
using System.Text.Json.Serialization;
|
using System.Text.Json.Serialization;
|
||||||
using pacMan.GameStuff;
|
using pacMan.GameStuff;
|
||||||
@ -9,12 +10,15 @@ public interface IActionService
|
|||||||
{
|
{
|
||||||
Player Player { set; }
|
Player Player { set; }
|
||||||
Game Game { set; }
|
Game Game { set; }
|
||||||
|
WebSocket? WebSocket { set; }
|
||||||
void DoAction(ActionMessage message);
|
void DoAction(ActionMessage message);
|
||||||
List<int> RollDice();
|
List<int> RollDice();
|
||||||
List<Player> SetPlayerInfo(JsonElement? jsonElement);
|
List<Player> SetPlayerInfo(JsonElement? jsonElement);
|
||||||
object? HandleMoveCharacter(JsonElement? jsonElement); // TODO test
|
object? HandleMoveCharacter(JsonElement? jsonElement);
|
||||||
object Ready();
|
object Ready();
|
||||||
string FindNextPlayer();
|
string FindNextPlayer();
|
||||||
|
List<Player> LeaveGame();
|
||||||
|
void SendToAll(ArraySegment<byte> segment);
|
||||||
void Disconnect();
|
void Disconnect();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -29,6 +33,8 @@ public class ActionService : IActionService
|
|||||||
_gameService = gameService;
|
_gameService = gameService;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public WebSocket? WebSocket { private get; set; }
|
||||||
|
|
||||||
public Game? Game { get; set; }
|
public Game? Game { get; set; }
|
||||||
|
|
||||||
public Player? Player { get; set; }
|
public Player? Player { get; set; }
|
||||||
@ -42,6 +48,7 @@ public class ActionService : IActionService
|
|||||||
GameAction.PlayerInfo => SetPlayerInfo(message.Data),
|
GameAction.PlayerInfo => SetPlayerInfo(message.Data),
|
||||||
GameAction.Ready => Ready(),
|
GameAction.Ready => Ready(),
|
||||||
GameAction.NextPlayer => FindNextPlayer(),
|
GameAction.NextPlayer => FindNextPlayer(),
|
||||||
|
GameAction.Disconnect => LeaveGame(),
|
||||||
_ => message.Data
|
_ => message.Data
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@ -55,19 +62,28 @@ public class ActionService : IActionService
|
|||||||
return rolls;
|
return rolls;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public object? HandleMoveCharacter(JsonElement? jsonElement)
|
||||||
|
{
|
||||||
|
if (Game != null && jsonElement.HasValue)
|
||||||
|
Game.Ghosts = jsonElement.Value.GetProperty("ghosts").Deserialize<List<Character>>() ??
|
||||||
|
throw new JsonException("Ghosts is null");
|
||||||
|
|
||||||
|
return jsonElement;
|
||||||
|
}
|
||||||
|
|
||||||
public List<Player> SetPlayerInfo(JsonElement? jsonElement) // TODO split up into two actions
|
public List<Player> SetPlayerInfo(JsonElement? jsonElement) // TODO split up into two actions
|
||||||
{
|
{
|
||||||
var data = jsonElement?.Deserialize<PlayerInfoData>() ?? throw new NullReferenceException("Data is null");
|
var data = jsonElement?.Deserialize<PlayerInfoData>() ?? throw new NullReferenceException("Data is null");
|
||||||
Player = data.Player;
|
Player = data.Player;
|
||||||
|
|
||||||
Game? group;
|
Game? game;
|
||||||
Player? player;
|
Player? player;
|
||||||
if ((group = _gameService.FindGameByUsername(Player.Username)) != null &&
|
if ((game = _gameService.FindGameByUsername(Player.Username)) != null &&
|
||||||
(player = group.Players.Find(p => p.Username == Player.Username))?.State == State.Disconnected)
|
(player = game.Players.Find(p => p.Username == Player.Username))?.State == State.Disconnected)
|
||||||
{
|
{
|
||||||
player.State = group.IsGameStarted ? State.InGame : State.WaitingForPlayers;
|
player.State = game.IsGameStarted ? State.InGame : State.WaitingForPlayers; // TODO doesn't work anymore
|
||||||
Player = player;
|
Player = player;
|
||||||
Game = group;
|
Game = game;
|
||||||
// TODO send missing data: Dices, CurrentPlayer, Ghosts
|
// TODO send missing data: Dices, CurrentPlayer, Ghosts
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -75,6 +91,8 @@ public class ActionService : IActionService
|
|||||||
Game = _gameService.AddPlayer(Player, data.Spawns);
|
Game = _gameService.AddPlayer(Player, data.Spawns);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Game.Connections += SendSegment;
|
||||||
|
|
||||||
return Game.Players;
|
return Game.Players;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -100,18 +118,26 @@ public class ActionService : IActionService
|
|||||||
|
|
||||||
public string FindNextPlayer() => Game?.NextPlayer().Username ?? "Error: No group found";
|
public string FindNextPlayer() => Game?.NextPlayer().Username ?? "Error: No group found";
|
||||||
|
|
||||||
public void Disconnect()
|
public List<Player> LeaveGame()
|
||||||
{
|
{
|
||||||
if (Player != null) Player.State = State.Disconnected;
|
if (Game == null || Player == null) throw new NullReferenceException("Game or Player is null");
|
||||||
|
Game.RemovePlayer(Player.Username);
|
||||||
|
return Game.Players;
|
||||||
}
|
}
|
||||||
|
|
||||||
public object? HandleMoveCharacter(JsonElement? jsonElement)
|
public void Disconnect()
|
||||||
{
|
{
|
||||||
if (Game != null && jsonElement.HasValue)
|
if (Player == null) return;
|
||||||
Game.Ghosts = jsonElement.Value.GetProperty("ghosts").Deserialize<List<Character>>() ??
|
Player.State = State.Disconnected;
|
||||||
throw new JsonException("Ghosts is null");
|
if (Game != null) Game.Connections -= SendSegment;
|
||||||
|
}
|
||||||
|
|
||||||
return jsonElement;
|
public void SendToAll(ArraySegment<byte> segment) => Game?.SendToAll(segment);
|
||||||
|
|
||||||
|
private async Task SendSegment(ArraySegment<byte> segment)
|
||||||
|
{
|
||||||
|
if (WebSocket != null) await _gameService.Send(WebSocket, segment);
|
||||||
|
else await Task.FromCanceled(new CancellationToken(true));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,13 +19,6 @@ public class GameService : WebSocketService
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public SynchronizedCollection<Game> Games { get; } = new();
|
public SynchronizedCollection<Game> Games { get; } = new();
|
||||||
|
|
||||||
public event Func<ArraySegment<byte>, Task>? Connections; // TODO remove and use Game
|
|
||||||
|
|
||||||
public void SendToAll(ArraySegment<byte> segment)
|
|
||||||
{
|
|
||||||
Connections?.Invoke(segment);
|
|
||||||
}
|
|
||||||
|
|
||||||
public Game AddPlayer(Player player, Queue<DirectionalPosition> spawns)
|
public Game AddPlayer(Player player, Queue<DirectionalPosition> spawns)
|
||||||
{
|
{
|
||||||
var index = 0;
|
var index = 0;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user