commit
fc2fff88a5
2
Makefile
2
Makefile
|
@ -6,5 +6,5 @@ build:
|
||||||
podman build . --tag=$(TAG)
|
podman build . --tag=$(TAG)
|
||||||
|
|
||||||
run: build
|
run: build
|
||||||
podman run -i -p 3000:3000 localhost/$(TAG):latest
|
podman run -i -p 80:3000 localhost/$(TAG):latest
|
||||||
|
|
||||||
|
|
6
package-lock.json
generated
Normal file
6
package-lock.json
generated
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
{
|
||||||
|
"name": "cccb-tanks-cs",
|
||||||
|
"lockfileVersion": 3,
|
||||||
|
"requires": true,
|
||||||
|
"packages": {}
|
||||||
|
}
|
|
@ -6,8 +6,9 @@ import Column from './components/Column.tsx';
|
||||||
import Row from './components/Row.tsx';
|
import Row from './components/Row.tsx';
|
||||||
import Scoreboard from './Scoreboard.tsx';
|
import Scoreboard from './Scoreboard.tsx';
|
||||||
import Button from './components/Button.tsx';
|
import Button from './components/Button.tsx';
|
||||||
|
import MapChooser from './MapChooser.tsx';
|
||||||
import './App.css';
|
import './App.css';
|
||||||
import {getRandomTheme, useStoredTheme} from './theme.ts';
|
import {getRandomTheme, useStoredTheme} from "./theme.ts";
|
||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
|
|
||||||
export default function App() {
|
export default function App() {
|
||||||
|
@ -20,6 +21,7 @@ export default function App() {
|
||||||
|
|
||||||
<Row>
|
<Row>
|
||||||
<h1 className="flex-grow">CCCB-Tanks!</h1>
|
<h1 className="flex-grow">CCCB-Tanks!</h1>
|
||||||
|
<MapChooser />
|
||||||
<Button text="change colors" onClick={() => setTheme(_ => getRandomTheme())}/>
|
<Button text="change colors" onClick={() => setTheme(_ => getRandomTheme())}/>
|
||||||
<Button
|
<Button
|
||||||
onClick={() => window.open('https://github.com/kaesaecracker/cccb-tanks-cs', '_blank')?.focus()}
|
onClick={() => window.open('https://github.com/kaesaecracker/cccb-tanks-cs', '_blank')?.focus()}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import {useEffect, useState} from 'react';
|
import {useEffect, useState} from 'react';
|
||||||
import './JoinForm.css';
|
import './JoinForm.css';
|
||||||
import {Player, postPlayer} from './serverCalls';
|
import {makeApiUrl, Player} from './serverCalls';
|
||||||
import Column from './components/Column.tsx';
|
import Column from './components/Column.tsx';
|
||||||
import Button from './components/Button.tsx';
|
import Button from './components/Button.tsx';
|
||||||
import TextInput from './components/TextInput.tsx';
|
import TextInput from './components/TextInput.tsx';
|
||||||
|
@ -16,19 +16,19 @@ export default function JoinForm({onDone}: {
|
||||||
if (!clicked || data)
|
if (!clicked || data)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
postPlayer(name).then(response => {
|
const url = makeApiUrl('/player');
|
||||||
if (response.ok && response.successResult) {
|
url.searchParams.set('name', name);
|
||||||
onDone(response.successResult!.trim());
|
|
||||||
setErrorText(null);
|
fetch(url, {method: 'POST'})
|
||||||
|
.then(async response => {
|
||||||
|
if (!response.ok) {
|
||||||
|
setErrorText(`${response.status} (${response.statusText}): ${await response.text()}`);
|
||||||
|
setClicked(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (response.additionalErrorText)
|
onDone((await response.json()).trim());
|
||||||
setErrorText(`${response.statusCode} (${response.statusText}): ${response.additionalErrorText}`);
|
setErrorText(null);
|
||||||
else
|
|
||||||
setErrorText(`${response.statusCode} (${response.statusText})`);
|
|
||||||
|
|
||||||
setClicked(false);
|
|
||||||
});
|
});
|
||||||
}, [clicked, setData, data, setClicked, onDone, errorText]);
|
}, [clicked, setData, data, setClicked, onDone, errorText]);
|
||||||
|
|
||||||
|
|
9
tank-frontend/src/MapChooser.css
Normal file
9
tank-frontend/src/MapChooser.css
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
|
||||||
|
.MapChooser-DropDown {
|
||||||
|
border: solid var(--border-size-thin);
|
||||||
|
padding: var(--padding-normal);
|
||||||
|
|
||||||
|
background: var(--color-background);
|
||||||
|
border-color: var(--color-primary);
|
||||||
|
color: var(--color-primary);
|
||||||
|
}
|
34
tank-frontend/src/MapChooser.tsx
Normal file
34
tank-frontend/src/MapChooser.tsx
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
import {ChangeEvent} from 'react';
|
||||||
|
import {makeApiUrl} from './serverCalls';
|
||||||
|
import './MapChooser.css';
|
||||||
|
import {useQuery} from '@tanstack/react-query';
|
||||||
|
|
||||||
|
export default function MapChooser() {
|
||||||
|
const query = useQuery({
|
||||||
|
queryKey: ['get-maps'],
|
||||||
|
queryFn: async () => {
|
||||||
|
const url = makeApiUrl('/map');
|
||||||
|
const response = await fetch(url, {method: 'GET'});
|
||||||
|
if (!response.ok)
|
||||||
|
throw new Error(`response failed with code ${response.status} (${response.status})${await response.text()}`);
|
||||||
|
return await response.json() as string[];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const onChange = (event: ChangeEvent<HTMLSelectElement>) => {
|
||||||
|
if (event.target.selectedIndex < 1)
|
||||||
|
return;
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
const url = makeApiUrl('/map');
|
||||||
|
url.searchParams.set('name', event.target.options[event.target.selectedIndex].value);
|
||||||
|
|
||||||
|
fetch(url, {method: 'POST'});
|
||||||
|
};
|
||||||
|
|
||||||
|
return <select value="maps" className="MapChooser-DropDown" onChange={onChange}>
|
||||||
|
<option value="" defaultValue={''}>Choose map</option>
|
||||||
|
{query.isSuccess && query.data.map(m =>
|
||||||
|
<option key={m} value={m}>{m}</option>)}
|
||||||
|
</select>;
|
||||||
|
}
|
|
@ -2,14 +2,6 @@ export function makeApiUrl(path: string, protocol: 'http' | 'ws' = 'http') {
|
||||||
return new URL(`${protocol}://${window.location.hostname}${path}`);
|
return new URL(`${protocol}://${window.location.hostname}${path}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
export type ServerResponse<T> = {
|
|
||||||
ok: boolean;
|
|
||||||
statusCode: number;
|
|
||||||
statusText: string;
|
|
||||||
additionalErrorText?: string;
|
|
||||||
successResult?: T;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type Scores = {
|
export type Scores = {
|
||||||
readonly kills: number;
|
readonly kills: number;
|
||||||
readonly deaths: number;
|
readonly deaths: number;
|
||||||
|
@ -22,25 +14,3 @@ export type Player = {
|
||||||
readonly name: string;
|
readonly name: string;
|
||||||
readonly scores: Scores;
|
readonly scores: Scores;
|
||||||
};
|
};
|
||||||
|
|
||||||
export async function fetchTyped<T>({url, method}: { url: URL; method: string; }): Promise<ServerResponse<T>> {
|
|
||||||
const response = await fetch(url, {method});
|
|
||||||
const result: ServerResponse<T> = {
|
|
||||||
ok: response.ok,
|
|
||||||
statusCode: response.status,
|
|
||||||
statusText: response.statusText
|
|
||||||
};
|
|
||||||
|
|
||||||
if (response.ok)
|
|
||||||
result.successResult = await response.json();
|
|
||||||
else
|
|
||||||
result.additionalErrorText = await response.text();
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function postPlayer(name: string) {
|
|
||||||
const url = makeApiUrl('/player');
|
|
||||||
url.searchParams.set('name', name);
|
|
||||||
|
|
||||||
return fetchTyped<string>({url, method: 'POST'});
|
|
||||||
}
|
|
||||||
|
|
|
@ -13,7 +13,7 @@ internal sealed class PlayerInfoConnection(
|
||||||
) : WebsocketServerConnection(logger, new ByteChannelWebSocket(rawSocket, logger, 0))
|
) : WebsocketServerConnection(logger, new ByteChannelWebSocket(rawSocket, logger, 0))
|
||||||
{
|
{
|
||||||
private readonly AppSerializerContext _context = new(new JsonSerializerOptions(JsonSerializerDefaults.Web));
|
private readonly AppSerializerContext _context = new(new JsonSerializerOptions(JsonSerializerDefaults.Web));
|
||||||
private bool _wantsInfoOnTick;
|
private bool _wantsInfoOnTick = true;
|
||||||
private byte[] _lastMessage = [];
|
private byte[] _lastMessage = [];
|
||||||
|
|
||||||
protected override ValueTask HandleMessageLockedAsync(Memory<byte> buffer)
|
protected override ValueTask HandleMessageLockedAsync(Memory<byte> buffer)
|
||||||
|
|
Loading…
Reference in a new issue