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 { .JoinElems {
padding: 8px 8px; padding: 8px 8px;
margin: 8px 8px; margin: 8px 8px;
color: rgb(255, 255, 255);
} }

View file

@ -1,22 +1,6 @@
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
import './JoinForm.css'; import './JoinForm.css';
import { PlayerResponse, postPlayer } from './serverCalls';
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;
}
export default function JoinForm({ onDone }: { onDone: (id: string) => void }) { export default function JoinForm({ onDone }: { onDone: (id: string) => void }) {
const [name, setName] = useState(''); const [name, setName] = useState('');
@ -28,12 +12,13 @@ export default function JoinForm({ onDone }: { onDone: (id: string) => void }) {
return; return;
try { try {
fetchPlayer(name, {}).then((value: string | null) => { postPlayer(name)
if (value) .then((value: PlayerResponse | null) => {
onDone(value); if (value)
else onDone(value.id);
setClicked(false); else
}); setClicked(false);
});
} catch (e) { } catch (e) {
console.log(e); console.log(e);
alert(e); alert(e);
@ -47,7 +32,7 @@ export default function JoinForm({ onDone }: { onDone: (id: string) => void }) {
</h1> </h1>
<p className='JoinElems' style={{ "color": "white" }}> Welcome and have fun!</p> <p className='JoinElems' style={{ "color": "white" }}> Welcome and have fun!</p>
<div className="JoinForm"> <div className="JoinForm">
<p className='JoinElems' style={{ "color": "white" }}> <p className='JoinElems' style={{ "color": "white" }}>
Enter your name to join the game! Enter your name to join the game!
</p> </p>
<input className="JoinElems" <input className="JoinElems"

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

@ -1,22 +1,24 @@
import React, {useState} from 'react'; import React, { useState } from 'react';
import './index.css'; import './index.css';
import ClientScreen from './ClientScreen'; import ClientScreen from './ClientScreen';
import Controls from './Controls.tsx'; import Controls from './Controls.tsx';
import JoinForm from './JoinForm.tsx'; import JoinForm from './JoinForm.tsx';
import {createRoot} from 'react-dom/client'; import { createRoot } from 'react-dom/client';
import PlayerInfo from './PlayerInfo.tsx';
function App() { function App() {
const [id, setId] = useState<string | null>(null); const [id, setId] = useState<string | null>(null);
return <> return <>
{id === null && <JoinForm onDone={name => setId(name)}/>} {id === null && <JoinForm onDone={name => setId(name)} />}
<ClientScreen/> {id == null || <PlayerInfo playerId={id} />}
{id == null || <Controls playerId={id}/>} <ClientScreen />
{id == null || <Controls playerId={id} />}
</>; </>;
} }
createRoot(document.getElementById('root')!).render( createRoot(document.getElementById('root')!).render(
<React.StrictMode> <React.StrictMode>
<App/> <App />
</React.StrictMode> </React.StrictMode>
); );

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' });
}