Add src/App.tsx
This commit is contained in:
parent
1ed95ff52a
commit
3dac7a5577
1 changed files with 86 additions and 0 deletions
86
src/App.tsx
Normal file
86
src/App.tsx
Normal file
|
|
@ -0,0 +1,86 @@
|
||||||
|
import { useState } from "react";
|
||||||
|
import { useStatus } from "./hooks/useStatus";
|
||||||
|
import { StatusBar } from "./components/StatusBar";
|
||||||
|
import { HostCard } from "./components/HostCard";
|
||||||
|
import { AddHostModal } from "./components/AddHostModal";
|
||||||
|
|
||||||
|
export default function App() {
|
||||||
|
const { status, hosts, error, loading, refresh } = useStatus(3000);
|
||||||
|
const [showAddModal, setShowAddModal] = useState(false);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col h-screen bg-surface select-none">
|
||||||
|
{/* Title bar / drag region for Tauri */}
|
||||||
|
<div
|
||||||
|
data-tauri-drag-region
|
||||||
|
className="flex items-center justify-between px-4 py-3 bg-surface-raised"
|
||||||
|
style={{ WebkitAppRegion: "drag" } as React.CSSProperties}
|
||||||
|
>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="text-accent font-bold text-lg">🌙 Moonlight Relay</span>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={refresh}
|
||||||
|
style={{ WebkitAppRegion: "no-drag" } as React.CSSProperties}
|
||||||
|
className="text-gray-400 hover:text-white text-xs transition-colors"
|
||||||
|
title="Refresh hosts"
|
||||||
|
>
|
||||||
|
↻ Refresh
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* VPN + moonlight status */}
|
||||||
|
<StatusBar status={status} error={error} loading={loading} />
|
||||||
|
|
||||||
|
{/* Main content */}
|
||||||
|
<main className="flex-1 overflow-y-auto p-4">
|
||||||
|
{/* Host grid */}
|
||||||
|
{hosts.length === 0 && !loading ? (
|
||||||
|
<div className="flex flex-col items-center justify-center h-full gap-3 text-gray-500">
|
||||||
|
<span className="text-5xl">🔍</span>
|
||||||
|
<p className="text-sm">No hosts found yet.</p>
|
||||||
|
<p className="text-xs text-center max-w-xs">
|
||||||
|
Make sure Apollo/Artemis is running on your gaming PC and it's
|
||||||
|
reachable over Tailscale (or your local network).
|
||||||
|
</p>
|
||||||
|
<button
|
||||||
|
onClick={() => setShowAddModal(true)}
|
||||||
|
className="mt-2 px-4 py-2 rounded-lg bg-accent text-sm font-medium"
|
||||||
|
>
|
||||||
|
+ Add host manually
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<div className="grid grid-cols-2 gap-3 sm:grid-cols-3">
|
||||||
|
{hosts.map((h) => (
|
||||||
|
<HostCard key={h.name + h.ip} host={h} onRefresh={refresh} />
|
||||||
|
))}
|
||||||
|
{/* Add host card */}
|
||||||
|
<button
|
||||||
|
onClick={() => setShowAddModal(true)}
|
||||||
|
className="
|
||||||
|
flex flex-col items-center justify-center gap-2
|
||||||
|
p-5 rounded-xl border border-dashed border-surface-elevated
|
||||||
|
text-gray-500 hover:text-gray-300 hover:border-gray-500
|
||||||
|
transition-colors duration-150
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<span className="text-2xl">+</span>
|
||||||
|
<span className="text-xs">Add host</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</main>
|
||||||
|
|
||||||
|
{/* Add host modal */}
|
||||||
|
{showAddModal && (
|
||||||
|
<AddHostModal
|
||||||
|
onClose={() => setShowAddModal(false)}
|
||||||
|
onAdded={refresh}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
Loading…
Reference in a new issue