4 Commits

9 changed files with 1027 additions and 342 deletions

103
README.md Normal file
View File

@@ -0,0 +1,103 @@
# Connect4 Moderator Observer UI
A website interface for connecting to the [Connect4 WebSocket server](https://github.com/joshuafhiggins/connect4-moderator-server) as an observer or player, watching live matches, and managing tournaments.
## Prerequisites
- Linux/macOS terminal or Windows PowerShell
- Git
- Docker (optional, for containerized runs)
## Run Locally (Node.js + npm)
### 1) Install Node.js (includes npm)
#### Option A: Install with `nvm` (recommended, including Windows support)
On Linux/macOS:
```bash
curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.40.3/install.sh | bash
source ~/.zshrc
nvm install --lts
nvm use --lts
node -v
npm -v
```
On Windows (PowerShell), use `nvm-windows`:
```powershell
winget install CoreyButler.NVMforWindows
nvm install lts
nvm use lts
node -v
npm -v
```
### 2) Install dependencies
From the project root:
```bash
npm i
```
### 3) Start the UI in development mode
```bash
npm run dev
```
Open [http://localhost:3000](http://localhost:3000).
## Run with Docker
This repository includes:
- `Dockerfile` (multi-stage production build)
- `docker-compose.yml`
- `docker_build.sh`
### Build image directly from `Dockerfile`
```bash
docker build . -t joshuafhiggins/connect4-ui
```
or use the provided script:
```bash
./docker_build.sh
```
### Run container directly
```bash
docker run --rm -p 3000:3000 --name connect4-ui joshuafhiggins/connect4-ui
```
### Run with Docker Compose
```bash
docker compose up --build
```
Run detached:
```bash
docker compose up -d --build
```
Stop:
```bash
docker compose down
```
## Useful npm Scripts
- `npm run dev` - start Next.js dev server
- `npm run build` - build for production
- `npm run start` - run production build
- `npm run lint` - lint the project

View File

@@ -7,7 +7,6 @@ import { ConnectionProvider } from "@/lib/connection";
import { CHIP_DROP_SOUND_PATHS } from "@/lib/sfx"; import { CHIP_DROP_SOUND_PATHS } from "@/lib/sfx";
export const metadata: Metadata = { export const metadata: Metadata = {
title: "Connect4 RPI Minds & Machines",
description: "Watch matches, track tournaments, and play Connect4", description: "Watch matches, track tournaments, and play Connect4",
icons: { icons: {
icon: "/favicon.ico", icon: "/favicon.ico",

View File

@@ -23,6 +23,8 @@ export default function PlayPage() {
role, role,
username, username,
status, status,
isInMatch,
shouldGoFirst,
send, send,
subscribe, subscribe,
reconnectAttempts, reconnectAttempts,
@@ -86,13 +88,25 @@ export default function PlayPage() {
} }
if (status === "connected" && gamePhase === "idle") { if (status === "connected" && gamePhase === "idle") {
setGamePhase("connected"); // Mid-match reconnect can remount with phase idle while still in a match; avoid
// the pre-queue "connected" / Ready Up state until we know we are not in-game.
setGamePhase(isInMatch ? "playing" : "connected");
if (isInMatch) {
const color: 1 | 2 = shouldGoFirst ? 1 : 2;
setMyColor(color);
myColorRef.current = color;
setIsMyTurn(shouldGoFirst);
isMyTurnRef.current = shouldGoFirst;
}
} }
}, [ }, [
role, role,
status, status,
router, router,
gamePhase, gamePhase,
isInMatch,
shouldRedirectToConnect, shouldRedirectToConnect,
clearRedirectFlag, clearRedirectFlag,
]); ]);
@@ -102,7 +116,11 @@ export default function PlayPage() {
switch (msg.type) { switch (msg.type) {
case "CONNECT_ACK": case "CONNECT_ACK":
case "RECONNECT_ACK": case "RECONNECT_ACK":
setGamePhase((prev) => (prev === "idle" ? "connected" : prev)); setGamePhase((prev) => {
if (prev !== "idle") return prev;
if (isInMatch) return "playing";
return "connected";
});
break; break;
case "ERROR": case "ERROR":
@@ -193,7 +211,7 @@ export default function PlayPage() {
}); });
return unsubscribe; return unsubscribe;
}, [resetGame, send, subscribe]); }, [resetGame, send, subscribe, isInMatch, username]);
const handleColumnClick = useCallback( const handleColumnClick = useCallback(
(col: number) => { (col: number) => {

View File

@@ -152,6 +152,7 @@ export default function AdminSettingsPanel() {
break; break;
case "RESERVATION_LIST": case "RESERVATION_LIST":
setReservations(message.reservations); setReservations(message.reservations);
setActionFeedback("Loaded reservations.");
break; break;
case "RESERVATION_ADD": case "RESERVATION_ADD":
setReservations((prev) => { setReservations((prev) => {

View File

@@ -40,6 +40,7 @@ interface ConnectionContextValue {
username: string; username: string;
status: ConnectionStatus; status: ConnectionStatus;
isInMatch: boolean; isInMatch: boolean;
shouldGoFirst: boolean;
isAdmin: boolean; isAdmin: boolean;
reconnectAttempts: number; reconnectAttempts: number;
shouldRedirectToConnect: boolean; shouldRedirectToConnect: boolean;
@@ -70,6 +71,7 @@ export function ConnectionProvider({
const [username, setUsername] = useState(""); const [username, setUsername] = useState("");
const [status, setStatus] = useState<ConnectionStatus>("idle"); const [status, setStatus] = useState<ConnectionStatus>("idle");
const [isInMatch, setIsInMatch] = useState(false); const [isInMatch, setIsInMatch] = useState(false);
const [shouldGoFirst, setShouldGoFirst] = useState(false);
const [isAdmin, setIsAdmin] = useState(false); const [isAdmin, setIsAdmin] = useState(false);
const [reconnectAttempts, setReconnectAttempts] = useState(0); const [reconnectAttempts, setReconnectAttempts] = useState(0);
const [shouldRedirectToConnect, setShouldRedirectToConnect] = useState(false); const [shouldRedirectToConnect, setShouldRedirectToConnect] = useState(false);
@@ -81,6 +83,7 @@ export function ConnectionProvider({
const reconnectDeadlineRef = useRef<number | null>(null); const reconnectDeadlineRef = useRef<number | null>(null);
const reconnectActiveRef = useRef(false); const reconnectActiveRef = useRef(false);
const isInMatchRef = useRef(false); const isInMatchRef = useRef(false);
const shouldGoFirstRef = useRef(false);
const sessionRef = useRef<SessionState | null>(null); const sessionRef = useRef<SessionState | null>(null);
const clearReconnectTimer = useCallback(() => { const clearReconnectTimer = useCallback(() => {
@@ -202,6 +205,8 @@ export function ConnectionProvider({
if (parsed.type === "GAME_START") { if (parsed.type === "GAME_START") {
isInMatchRef.current = true; isInMatchRef.current = true;
setIsInMatch(true); setIsInMatch(true);
shouldGoFirstRef.current = parsed.goesFirst;
setShouldGoFirst(parsed.goesFirst);
} }
if ( if (
@@ -377,6 +382,7 @@ export function ConnectionProvider({
username, username,
status, status,
isInMatch, isInMatch,
shouldGoFirst,
isAdmin, isAdmin,
reconnectAttempts, reconnectAttempts,
shouldRedirectToConnect, shouldRedirectToConnect,
@@ -394,6 +400,7 @@ export function ConnectionProvider({
username, username,
status, status,
isInMatch, isInMatch,
shouldGoFirst,
isAdmin, isAdmin,
reconnectAttempts, reconnectAttempts,
shouldRedirectToConnect, shouldRedirectToConnect,

1
next-env.d.ts vendored
View File

@@ -1,5 +1,6 @@
/// <reference types="next" /> /// <reference types="next" />
/// <reference types="next/image-types/global" /> /// <reference types="next/image-types/global" />
import "./.next/dev/types/routes.d.ts";
// NOTE: This file should not be edited // NOTE: This file should not be edited
// see https://nextjs.org/docs/app/api-reference/config/typescript for more information. // see https://nextjs.org/docs/app/api-reference/config/typescript for more information.

1192
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -12,21 +12,25 @@
}, },
"dependencies": { "dependencies": {
"lucide-react": "^1.8.0", "lucide-react": "^1.8.0",
"next": "15.2.4", "next": "16.2.4",
"react": "^19.0.0", "react": "19.2.5",
"react-confetti": "^6.4.0", "react-confetti": "^6.4.0",
"react-dom": "^19.0.0" "react-dom": "19.2.5"
}, },
"devDependencies": { "devDependencies": {
"@tailwindcss/postcss": "^4", "@tailwindcss/postcss": "^4",
"@types/node": "^20", "@types/node": "^20",
"@types/react": "^19", "@types/react": "19.2.14",
"@types/react-dom": "^19", "@types/react-dom": "19.2.3",
"eslint": "^9", "eslint": "^9",
"eslint-config-next": "15.2.4", "eslint-config-next": "16.2.4",
"postcss": "^8", "postcss": "^8",
"prettier": "^3.8.1", "prettier": "^3.8.1",
"tailwindcss": "^4", "tailwindcss": "^4",
"typescript": "^5" "typescript": "^5"
},
"overrides": {
"@types/react": "19.2.14",
"@types/react-dom": "19.2.3"
} }
} }

View File

@@ -1,6 +1,10 @@
{ {
"compilerOptions": { "compilerOptions": {
"lib": ["dom", "dom.iterable", "esnext"], "lib": [
"dom",
"dom.iterable",
"esnext"
],
"allowJs": true, "allowJs": true,
"skipLibCheck": true, "skipLibCheck": true,
"strict": true, "strict": true,
@@ -10,7 +14,7 @@
"moduleResolution": "bundler", "moduleResolution": "bundler",
"resolveJsonModule": true, "resolveJsonModule": true,
"isolatedModules": true, "isolatedModules": true,
"jsx": "preserve", "jsx": "react-jsx",
"incremental": true, "incremental": true,
"plugins": [ "plugins": [
{ {
@@ -18,10 +22,20 @@
} }
], ],
"paths": { "paths": {
"@/*": ["./*"] "@/*": [
"./*"
]
}, },
"target": "ES2017" "target": "ES2017"
}, },
"include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"], "include": [
"exclude": ["node_modules"] "next-env.d.ts",
"**/*.ts",
"**/*.tsx",
".next/types/**/*.ts",
".next/dev/types/**/*.ts"
],
"exclude": [
"node_modules"
]
} }