2026-05-21 12:03:17 +00:00
|
|
|
import { Camera, Radio } from 'lucide-react'
|
|
|
|
|
import { useSSE } from './hooks/useSSE'
|
|
|
|
|
import { useCameraStore } from './store/useCameraStore'
|
|
|
|
|
import { CameraCard } from './components'
|
2026-05-19 07:31:23 -04:00
|
|
|
|
|
|
|
|
function App() {
|
2026-05-21 12:03:17 +00:00
|
|
|
// Connect to SSE endpoint — auto-updates the camera store
|
|
|
|
|
useSSE()
|
|
|
|
|
|
|
|
|
|
// Subscribe to the camera store for reactivity.
|
|
|
|
|
// getCameras / getOnlineCount / getRecordingCount pull from live state.
|
|
|
|
|
const { getCameras, getOnlineCount, getRecordingCount } = useCameraStore()
|
|
|
|
|
const cameras = getCameras()
|
|
|
|
|
const onlineCount = getOnlineCount()
|
|
|
|
|
const recordingCount = getRecordingCount()
|
|
|
|
|
|
2026-05-19 07:31:23 -04:00
|
|
|
return (
|
|
|
|
|
<div className="min-h-screen bg-rig-dark-900">
|
|
|
|
|
{/* Header */}
|
|
|
|
|
<header className="border-b border-rig-dark-700 bg-rig-dark-800/50 backdrop-blur-sm">
|
|
|
|
|
<div className="mx-auto max-w-7xl px-4 py-4 sm:px-6 lg:px-8">
|
|
|
|
|
<div className="flex items-center gap-3">
|
|
|
|
|
<Camera className="h-7 w-7 text-rig-accent" />
|
|
|
|
|
<h1 className="text-xl font-bold tracking-tight text-rig-dark-50">
|
|
|
|
|
RemoteRig
|
|
|
|
|
</h1>
|
|
|
|
|
<span className="ml-2 rounded-full bg-rig-accent/10 px-2.5 py-0.5 text-xs font-medium text-rig-accent">
|
|
|
|
|
Dashboard
|
|
|
|
|
</span>
|
2026-05-21 12:03:17 +00:00
|
|
|
|
|
|
|
|
{/* Stats badges */}
|
|
|
|
|
<div className="ml-auto flex items-center gap-4">
|
|
|
|
|
{/* Online count */}
|
|
|
|
|
<span
|
|
|
|
|
className="inline-flex items-center gap-1.5 rounded-full bg-rig-dark-700/60 px-3 py-1 text-xs font-medium text-rig-dark-200"
|
|
|
|
|
title="Cameras online"
|
|
|
|
|
>
|
|
|
|
|
<span className="h-2 w-2 rounded-full bg-rig-success" />
|
|
|
|
|
{onlineCount} online
|
|
|
|
|
</span>
|
|
|
|
|
|
|
|
|
|
{/* Recording count */}
|
|
|
|
|
<span
|
|
|
|
|
className="inline-flex items-center gap-1.5 rounded-full bg-rig-dark-700/60 px-3 py-1 text-xs font-medium text-rig-dark-200"
|
|
|
|
|
title="Cameras recording"
|
|
|
|
|
>
|
|
|
|
|
<span className="relative flex h-2 w-2">
|
|
|
|
|
<span className="absolute inline-flex h-full w-full animate-ping rounded-full bg-rig-danger opacity-75" />
|
|
|
|
|
<span className="relative inline-flex h-2 w-2 rounded-full bg-rig-danger" />
|
|
|
|
|
</span>
|
|
|
|
|
{recordingCount} recording
|
|
|
|
|
</span>
|
|
|
|
|
</div>
|
2026-05-19 07:31:23 -04:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</header>
|
|
|
|
|
|
|
|
|
|
{/* Main Content */}
|
|
|
|
|
<main className="mx-auto max-w-7xl px-4 py-8 sm:px-6 lg:px-8">
|
2026-05-21 12:03:17 +00:00
|
|
|
{cameras.length === 0 ? (
|
|
|
|
|
/* Empty state */
|
|
|
|
|
<div className="flex flex-col items-center justify-center rounded-xl border border-dashed border-rig-dark-600 bg-rig-dark-800/30 py-24 text-center">
|
|
|
|
|
<span className="relative mb-4 inline-flex">
|
|
|
|
|
<Radio className="h-12 w-12 animate-pulse text-rig-accent" />
|
|
|
|
|
</span>
|
|
|
|
|
<h2 className="text-lg font-semibold text-rig-dark-200">
|
|
|
|
|
Waiting for cameras…
|
|
|
|
|
</h2>
|
|
|
|
|
<p className="mt-2 max-w-sm text-sm text-rig-dark-400">
|
|
|
|
|
Connect cameras to your RemoteRig server and they will appear here
|
|
|
|
|
automatically.
|
|
|
|
|
</p>
|
|
|
|
|
</div>
|
|
|
|
|
) : (
|
|
|
|
|
/* Camera grid */
|
|
|
|
|
<div className="grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4">
|
|
|
|
|
{cameras.map((camera) => (
|
|
|
|
|
<CameraCard key={camera.camera_id} camera={camera} />
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2026-05-19 07:31:23 -04:00
|
|
|
</main>
|
|
|
|
|
|
|
|
|
|
{/* Footer */}
|
|
|
|
|
<footer className="border-t border-rig-dark-700 bg-rig-dark-800/30">
|
|
|
|
|
<div className="mx-auto max-w-7xl px-4 py-3 sm:px-6 lg:px-8">
|
|
|
|
|
<p className="text-center text-xs text-rig-dark-500">
|
|
|
|
|
RemoteRig v0.1.0 — Multi-Camera Remote Monitoring System
|
|
|
|
|
</p>
|
|
|
|
|
</div>
|
|
|
|
|
</footer>
|
|
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default App
|