Created a simple lobby containing all games
This commit is contained in:
parent
ac8560e61c
commit
ad0d8c7d0a
@ -1,3 +1,5 @@
|
|||||||
PORT=44435
|
PORT=44435
|
||||||
HTTPS=true
|
HTTPS=true
|
||||||
VITE_API=wss://localhost:3000/api/game
|
VITE_API_URI=localhost:3000/api/game
|
||||||
|
VITE_API_HTTP=https://$VITE_API_URI
|
||||||
|
VITE_API_WS=wss://$VITE_API_URI/connect
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import {Route, Routes} from "react-router-dom";
|
import {Route, Routes} from "react-router-dom";
|
||||||
import {Layout} from "./components/Layout";
|
import Layout from "./components/layout";
|
||||||
import AppRoutes from "./AppRoutes";
|
import AppRoutes from "./AppRoutes";
|
||||||
import "./index.css";
|
import "./index.css";
|
||||||
|
|
||||||
|
@ -2,6 +2,7 @@ import React from "react";
|
|||||||
import {Counter} from "./pages/counter";
|
import {Counter} from "./pages/counter";
|
||||||
import Home from "./pages/home";
|
import Home from "./pages/home";
|
||||||
import Game from "./pages/game";
|
import Game from "./pages/game";
|
||||||
|
import LobbyPage from "./pages/lobby";
|
||||||
|
|
||||||
const AppRoutes = [
|
const AppRoutes = [
|
||||||
{
|
{
|
||||||
@ -16,6 +17,10 @@ const AppRoutes = [
|
|||||||
path: "/game",
|
path: "/game",
|
||||||
element: <Game/>,
|
element: <Game/>,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: "/lobby",
|
||||||
|
element: <LobbyPage/>,
|
||||||
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
export default AppRoutes;
|
export default AppRoutes;
|
||||||
|
@ -1,11 +0,0 @@
|
|||||||
import React from "react";
|
|
||||||
import {NavMenu} from "./NavMenu";
|
|
||||||
|
|
||||||
export const Layout: Component<ChildProps> = ({children}) => (
|
|
||||||
<div>
|
|
||||||
<NavMenu/>
|
|
||||||
<main>
|
|
||||||
{children}
|
|
||||||
</main>
|
|
||||||
</div>
|
|
||||||
);
|
|
@ -1,26 +0,0 @@
|
|||||||
import React from "react";
|
|
||||||
import {Link} from "react-router-dom";
|
|
||||||
|
|
||||||
export const NavMenu = () => {
|
|
||||||
|
|
||||||
const [collapsed, setCollapsed] = React.useState(true);
|
|
||||||
|
|
||||||
function toggleNavbar() {
|
|
||||||
setCollapsed(!collapsed);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<header>
|
|
||||||
<nav className="navbar-expand-sm navbar-toggleable-sm ng-white border-bottom box-shadow mb-3">
|
|
||||||
<Link to="/">Pac-Man Board Game</Link>
|
|
||||||
<div onClick={toggleNavbar} className="mr-2"/>
|
|
||||||
<div className="d-sm-inline-flex flex-sm-row-reverse">
|
|
||||||
<ul className="navbar-nav flex-grow">
|
|
||||||
<Link className="text-dark" to="/">Home</Link>
|
|
||||||
<Link className="text-dark" to="/counter">Counter</Link>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
</header>
|
|
||||||
);
|
|
||||||
};
|
|
@ -11,7 +11,7 @@ import {diceAtom, ghostsAtom, playersAtom, rollDiceButtonAtom, selectedDiceAtom}
|
|||||||
import {CharacterType} from "../game/character";
|
import {CharacterType} from "../game/character";
|
||||||
import GameButton from "./gameButton";
|
import GameButton from "./gameButton";
|
||||||
|
|
||||||
const wsService = new WebSocketService(import.meta.env.VITE_API);
|
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 get to steal
|
||||||
// TODO bug, first player can sometimes roll dice twice (maybe only on firefox)
|
// TODO bug, first player can sometimes roll dice twice (maybe only on firefox)
|
||||||
|
13
pac-man-board-game/ClientApp/src/components/layout.tsx
Normal file
13
pac-man-board-game/ClientApp/src/components/layout.tsx
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import React from "react";
|
||||||
|
import NavMenu from "./navMenu";
|
||||||
|
|
||||||
|
const Layout: Component<ChildProps> = ({children}) => (
|
||||||
|
<div>
|
||||||
|
<NavMenu/>
|
||||||
|
<main>
|
||||||
|
{children}
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export default Layout;
|
28
pac-man-board-game/ClientApp/src/components/navMenu.tsx
Normal file
28
pac-man-board-game/ClientApp/src/components/navMenu.tsx
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import React from "react";
|
||||||
|
import {Link} from "react-router-dom";
|
||||||
|
|
||||||
|
const NavMenu: Component = () => {
|
||||||
|
|
||||||
|
const [collapsed, setCollapsed] = React.useState(true);
|
||||||
|
|
||||||
|
function toggleNavbar() {
|
||||||
|
setCollapsed(!collapsed);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<header>
|
||||||
|
<nav className="navbar-expand-sm navbar-toggleable-sm ng-white border-bottom box-shadow mb-3">
|
||||||
|
<Link to="/">Pac-Man Board Game</Link>
|
||||||
|
<div onClick={toggleNavbar} className="mr-2"/>
|
||||||
|
<div className="d-sm-inline-flex flex-sm-row-reverse">
|
||||||
|
<ul className="navbar-nav flex-grow">
|
||||||
|
<Link className="text-dark" to="/">Home</Link>
|
||||||
|
<Link className="text-dark" to="/counter">Counter</Link>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
</header>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default NavMenu;
|
52
pac-man-board-game/ClientApp/src/pages/lobby.tsx
Normal file
52
pac-man-board-game/ClientApp/src/pages/lobby.tsx
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import React, {Suspense} from "react";
|
||||||
|
import {atom, useAtomValue} from "jotai";
|
||||||
|
import {Button} from "../components/Button";
|
||||||
|
|
||||||
|
const fetchAtom = atom(async () => {
|
||||||
|
const response = await fetch(import.meta.env.VITE_API_HTTP + "/allGames");
|
||||||
|
return await response.json() as GameGroup[];
|
||||||
|
});
|
||||||
|
|
||||||
|
const LobbyPage: Component = () => (
|
||||||
|
<Suspense fallback={"Please wait"}>
|
||||||
|
<GameTable className={"mx-auto"}/>
|
||||||
|
</Suspense>
|
||||||
|
)
|
||||||
|
|
||||||
|
export default LobbyPage;
|
||||||
|
|
||||||
|
const GameTable: Component = ({className}) => {
|
||||||
|
|
||||||
|
const data = useAtomValue(fetchAtom);
|
||||||
|
|
||||||
|
function joinGame(gameId: string): void {
|
||||||
|
console.log("Joining game " + gameId); // TODO: Implement
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<table className={`rounded overflow-hidden ${className}`}>
|
||||||
|
<thead className={"bg-gray-500 text-white"}>
|
||||||
|
<tr className={"my-5"}>
|
||||||
|
<th>Id</th>
|
||||||
|
<th>Count</th>
|
||||||
|
<th className={"p-2"}>State</th>
|
||||||
|
<th>Join</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{data?.map(game =>
|
||||||
|
<tr key={game.id} className={"even:bg-gray-200"}>
|
||||||
|
<td className={"p-2"}>{game.id}</td>
|
||||||
|
<td className={"text-center"}>{game.count}</td>
|
||||||
|
<td>{game.isGameStarted ? "Closed" : "Open"}</td>
|
||||||
|
<td className={"p-2"}>
|
||||||
|
<Button disabled={game.isGameStarted} onClick={() => joinGame(game.id)}>
|
||||||
|
Join
|
||||||
|
</Button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
)}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
);
|
||||||
|
}
|
@ -34,3 +34,9 @@ type Path = {
|
|||||||
End: Position,
|
End: Position,
|
||||||
Direction: import("../game/direction").Direction
|
Direction: import("../game/direction").Direction
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type GameGroup = {
|
||||||
|
readonly id: string,
|
||||||
|
readonly count: number,
|
||||||
|
readonly isGameStarted: boolean,
|
||||||
|
}
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
/// <reference types="vite/client" />
|
/// <reference types="vite/client" />
|
||||||
|
|
||||||
interface ImportMetaEnv {
|
interface ImportMetaEnv {
|
||||||
readonly VITE_API: string;
|
readonly VITE_API_URI: string,
|
||||||
|
readonly VITE_API_HTTP: string,
|
||||||
|
readonly VITE_API_WS: string,
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ImportMeta {
|
interface ImportMeta {
|
||||||
|
@ -17,9 +17,17 @@ public class GameController : GenericController // TODO reconnect using player i
|
|||||||
base(logger, wsService) =>
|
base(logger, wsService) =>
|
||||||
_actionService = actionService;
|
_actionService = actionService;
|
||||||
|
|
||||||
[HttpGet]
|
[HttpGet("connect")]
|
||||||
public override async Task Accept() => await base.Accept();
|
public override async Task Accept() => await base.Accept();
|
||||||
|
|
||||||
|
[HttpGet("allGames")]
|
||||||
|
public IEnumerable<GameGroup> GetAllGames()
|
||||||
|
{
|
||||||
|
Logger.Log(LogLevel.Information, "Returning all games");
|
||||||
|
return WsService.Games;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data)
|
protected override ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data)
|
||||||
{
|
{
|
||||||
var stringResult = data.GetString(result.Count);
|
var stringResult = data.GetString(result.Count);
|
||||||
|
@ -1,28 +1,28 @@
|
|||||||
using System.Collections;
|
using System.Text.Json.Serialization;
|
||||||
using pacMan.Exceptions;
|
using pacMan.Exceptions;
|
||||||
using pacMan.Game;
|
using pacMan.Game;
|
||||||
using pacMan.Game.Items;
|
using pacMan.Game.Items;
|
||||||
|
|
||||||
namespace pacMan.Services;
|
namespace pacMan.Services;
|
||||||
|
|
||||||
public class GameGroup : IEnumerable<IPlayer> // TODO handle disconnects and reconnects
|
public class GameGroup // TODO handle disconnects and reconnects
|
||||||
{
|
{
|
||||||
private readonly Random _random = new();
|
private readonly Random _random = new();
|
||||||
private int _currentPlayerIndex;
|
private int _currentPlayerIndex;
|
||||||
|
|
||||||
public GameGroup(Queue<DirectionalPosition> spawns) => Spawns = spawns;
|
public GameGroup(Queue<DirectionalPosition> spawns) => Spawns = spawns;
|
||||||
|
|
||||||
public List<IPlayer> Players { get; } = new();
|
[JsonInclude] public Guid Id { get; } = Guid.NewGuid();
|
||||||
private Queue<DirectionalPosition> Spawns { get; }
|
|
||||||
|
|
||||||
public int Count => Players.Count;
|
[JsonIgnore] public List<IPlayer> Players { get; } = new();
|
||||||
|
|
||||||
|
[JsonIgnore] private Queue<DirectionalPosition> Spawns { get; }
|
||||||
|
|
||||||
|
[JsonInclude] public int Count => Players.Count;
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public bool IsGameStarted => Count > 0 && Players.All(player => player.State is State.InGame or State.Disconnected);
|
public bool IsGameStarted => Count > 0 && Players.All(player => player.State is State.InGame or State.Disconnected);
|
||||||
|
|
||||||
public IEnumerator<IPlayer> GetEnumerator() => Players.GetEnumerator();
|
|
||||||
|
|
||||||
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
|
||||||
|
|
||||||
public IPlayer NextPlayer()
|
public IPlayer NextPlayer()
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@ -33,6 +33,7 @@ public class GameGroup : IEnumerable<IPlayer> // TODO handle disconnects and rec
|
|||||||
{
|
{
|
||||||
throw new InvalidOperationException("There are no players in the game group.");
|
throw new InvalidOperationException("There are no players in the game group.");
|
||||||
}
|
}
|
||||||
|
|
||||||
return Players[_currentPlayerIndex];
|
return Players[_currentPlayerIndex];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user