separated server calls

This commit is contained in:
Ronja Spiegelberg 2024-04-13 17:56:33 +02:00
parent 7cc159edd7
commit 45b4c7f7fc
6 changed files with 99 additions and 31 deletions

View file

@ -13,5 +13,4 @@
.JoinElems {
padding: 8px 8px;
margin: 8px 8px;
color: rgb(255, 255, 255);
}

View file

@ -1,22 +1,6 @@
import { useEffect, useState } from 'react';
import './JoinForm.css';
type PlayerResponse = {
readonly name: string;
readonly id: string;
};
export async function fetchPlayer(name: string, options: RequestInit) {
const url = new URL(import.meta.env.VITE_TANK_PLAYER_URL);
url.searchParams.set('name', name);
const response = await fetch(url, options);
if (!response.ok)
return null;
const json = await response.json() as PlayerResponse;
return json.id;
}
import { PlayerResponse, postPlayer } from './serverCalls';
export default function JoinForm({ onDone }: { onDone: (id: string) => void }) {
const [name, setName] = useState('');
@ -28,9 +12,10 @@ export default function JoinForm({ onDone }: { onDone: (id: string) => void }) {
return;
try {
fetchPlayer(name, {}).then((value: string | null) => {
postPlayer(name)
.then((value: PlayerResponse | null) => {
if (value)
onDone(value);
onDone(value.id);
else
setClicked(false);
});

View file

@ -0,0 +1,21 @@
.Player {
display: flex;
flex-direction: column;
border: 2px solid rgb(76, 76, 76);
border-radius: 4px;
}
.ScoreForm {
display: flex;
flex-direction: column;
}
.ElemGroup {
display: flex;
flex-direction: row;
}
.Elems {
padding: 8px 8px;
margin: 8px 8px;
}

View file

@ -0,0 +1,38 @@
import { useEffect, useState } from 'react';
import './PlayerInfo.css'
import { PlayerResponse, getPlayer } from './serverCalls';
export default function PlayerInfo({ playerId }: { playerId: string }) {
const [player, setPlayer] = useState<PlayerResponse | null>();
const refresh = () => {
getPlayer(playerId).then(setPlayer);
};
useEffect(() => {
const timer = setInterval(refresh, 5000);
return () => clearInterval(timer);
});
return <div className='TankWelcome'>
<h1 className='Elems' style={{ "color": "white" }}>
Tanks
</h1>
<div className="ScoreForm">
<div className='ElemGroup'>
<p className='Elems' style={{ "color": "white" }}>
name
</p>
<p className='Elems' style={{ "color": "white" }}>
{player?.name}
</p>
</div>
<div className='ElemGroup'></div>
<p className='Elems' style={{ "color": "white" }}>
{JSON.stringify(player)}
</p>
</div>
</div >;
}

View file

@ -4,12 +4,14 @@ import ClientScreen from './ClientScreen';
import Controls from './Controls.tsx';
import JoinForm from './JoinForm.tsx';
import { createRoot } from 'react-dom/client';
import PlayerInfo from './PlayerInfo.tsx';
function App() {
const [id, setId] = useState<string | null>(null);
return <>
{id === null && <JoinForm onDone={name => setId(name)} />}
{id == null || <PlayerInfo playerId={id} />}
<ClientScreen />
{id == null || <Controls playerId={id} />}
</>;

View file

@ -0,0 +1,23 @@
export type PlayerResponse = {
readonly name: string;
readonly id: string;
};
export async function fetchTyped<T>({ url, method }: { url: URL; method: string; }) {
const response = await fetch(url, { method });
if (!response.ok)
return null;
return await response.json() as T;
}
export function postPlayer(name: string) {
const url = new URL(import.meta.env.VITE_TANK_PLAYER_URL);
url.searchParams.set('name', name);
return fetchTyped<PlayerResponse>({ url, method: 'POST' });
}
export function getPlayer(id: string) {
const url = new URL(import.meta.env.VITE_TANK_PLAYER_URL);
url.searchParams.set('id', id);
return fetchTyped<PlayerResponse>({ url, method: 'GET' });
}