Files
decap-stream/src/components/StreamCard.tsx
T

348 lines
16 KiB
TypeScript
Raw Normal View History

2026-04-23 23:40:34 -03:00
"use client"
import { useState, useEffect, useRef } from "react"
import { MoreHorizontal, Play, Globe, Monitor, Pencil, RotateCcw, Square, Trash2, Circle, Copy, Check, Video, ImageUp, GripVertical, Wrench, RefreshCw } from "lucide-react"
2026-04-23 23:40:34 -03:00
import { cn } from "@/lib/utils"
import { Toggle } from "@/components/Toggle"
2026-04-23 23:40:34 -03:00
import type { Stream } from "@/types/stream"
import type { SyntheticListenerMap } from "@dnd-kit/core/dist/hooks/utilities"
import type { DraggableAttributes } from "@dnd-kit/core"
2026-04-23 23:40:34 -03:00
interface Props {
stream: Stream
status?: Record<string, string>
localStatus?: string | null
cardSize?: "mini" | "sm" | "md" | "lg"
2026-04-23 23:40:34 -03:00
onRefresh: () => void
onLocalStatus: (id: string, s: string | null) => void
dragHandleListeners?: SyntheticListenerMap
dragHandleAttributes?: DraggableAttributes
isDragging?: boolean
globalPrefs: { pureMode: boolean; newTab: boolean }
2026-04-23 23:40:34 -03:00
}
function StatusBadge({ status, localStatus, cardSize = "md" }: { status?: Record<string, string>; localStatus?: string | null; cardSize?: "mini" | "sm" | "md" | "lg" }) {
const sc = SCALE[cardSize]
const label = localStatus ?? (
status?.ffmpeg === "RUNNING" ? "running" :
status?.ffmpeg === "STARTING" ? "starting" :
status?.ffmpeg === "FATAL" ? "error" :
status?.ffmpeg === "STOPPED" ? "stopped" : "..."
)
2026-04-23 23:40:34 -03:00
const color =
label === "running" ? "bg-green-500" :
label === "starting" ? "bg-yellow-500" :
label === "restarting" ? "bg-yellow-500" :
label === "stopping" ? "bg-orange-500" :
label === "error" ? "bg-red-500" : "bg-zinc-500"
2026-04-23 23:40:34 -03:00
return (
<span className={cn("flex items-center gap-1.5 text-muted-foreground whitespace-nowrap", sc.meta)}>
<Circle className={cn("fill-current shrink-0", color, sc.dot)} />
2026-04-23 23:40:34 -03:00
{label}
</span>
)
}
function copyToClipboard(text: string) {
if (navigator.clipboard && window.isSecureContext) {
return navigator.clipboard.writeText(text)
}
const el = document.createElement("textarea")
el.value = text
el.style.position = "fixed"
el.style.opacity = "0"
document.body.appendChild(el)
el.focus()
el.select()
document.execCommand("copy")
document.body.removeChild(el)
return Promise.resolve()
}
const CARD_WIDTHS = { mini: "sm:max-w-[200px]", sm: "sm:max-w-[240px]", md: "sm:max-w-[300px]", lg: "sm:max-w-[380px]" }
const SCALE = {
mini: { card: "p-2 gap-2", name: "text-xs", meta: "text-[10px]", btn: "text-[10px] px-2 py-1 gap-1.5", btnIcon: "w-2.5 h-2.5", menuIcon: "w-3.5 h-3.5", dot: "w-1.5 h-1.5" },
sm: { card: "p-2.5 gap-2", name: "text-xs", meta: "text-[10px]", btn: "text-xs px-2.5 py-1.5 gap-1.5", btnIcon: "w-2.5 h-2.5", menuIcon: "w-3.5 h-3.5", dot: "w-1.5 h-1.5" },
md: { card: "p-3 gap-2.5", name: "text-sm", meta: "text-xs", btn: "text-xs px-3 py-2 gap-2", btnIcon: "w-3 h-3", menuIcon: "w-4 h-4", dot: "w-2 h-2" },
lg: { card: "p-4 gap-3", name: "text-base", meta: "text-sm", btn: "text-sm px-4 py-2.5 gap-2", btnIcon: "w-4 h-4", menuIcon: "w-4 h-4", dot: "w-2 h-2" },
} satisfies Record<string, { card: string; name: string; meta: string; btn: string; btnIcon: string; menuIcon: string; dot: string }>
function ConfirmDeleteModal({ name, onConfirm, onCancel }: { name: string; onConfirm: () => void; onCancel: () => void }) {
return (
<div className="fixed inset-0 z-50 flex items-center justify-center">
<div className="absolute inset-0 bg-black/60" onClick={onCancel} />
<div className="relative z-10 w-80 rounded-xl border border-border shadow-2xl p-6 flex flex-col gap-4" style={{ background: "#1c1c1c" }}>
<div className="flex flex-col gap-1">
<p className="font-semibold text-sm">Delete stream</p>
<p className="text-sm text-muted-foreground">
Are you sure you want to delete <span className="text-foreground font-medium">{name}</span>? This action cannot be undone.
</p>
</div>
<div className="flex gap-2 justify-end">
<button
onClick={onCancel}
className="px-4 py-1.5 rounded border border-border text-sm hover:bg-[#2a2a2a] transition-colors cursor-pointer"
>
Cancel
</button>
<button
onClick={onConfirm}
className="px-4 py-1.5 rounded border border-destructive bg-destructive/10 text-destructive text-sm hover:bg-destructive hover:text-white transition-colors cursor-pointer"
>
Delete
</button>
</div>
</div>
</div>
)
}
export function StreamCard({ stream, status, localStatus, cardSize = "md", onRefresh, onLocalStatus, dragHandleListeners, dragHandleAttributes, isDragging, globalPrefs }: Props) {
const sc = SCALE[cardSize]
2026-04-23 23:40:34 -03:00
const [menuOpen, setMenuOpen] = useState(false)
const [copied, setCopied] = useState(false)
const [thumbKey, setThumbKey] = useState(0)
const [thumbError, setThumbError] = useState(false)
const [thumbCapturing, setThumbCapturing] = useState(false)
const [confirmDelete, setConfirmDelete] = useState(false)
const [autoReload, setAutoReload] = useState(stream.autoReload ?? false)
const [autoReloadMins, setAutoReloadMins] = useState(Math.round((stream.autoReloadInterval ?? 3600) / 60))
const pollRef = useRef<ReturnType<typeof setInterval> | null>(null)
2026-04-23 23:40:34 -03:00
useEffect(() => {
if (!thumbError || thumbCapturing) return
const interval = setInterval(() => setThumbKey((k) => k + 1), 15000)
return () => clearInterval(interval)
}, [thumbError, thumbCapturing])
useEffect(() => () => { if (pollRef.current) clearInterval(pollRef.current) }, [])
function navigate(url: string) {
if (globalPrefs.newTab) window.open(url, "_blank")
else window.location.href = url
}
async function action(act: string, optimisticStatus: string) {
onLocalStatus(stream.id, optimisticStatus)
setMenuOpen(false)
2026-04-23 23:40:34 -03:00
await fetch(`/api/streams/${stream.id}/${act}`, { method: "POST" })
onRefresh()
setTimeout(() => onLocalStatus(stream.id, null), 15000)
2026-04-23 23:40:34 -03:00
}
async function remove() {
setMenuOpen(false)
setConfirmDelete(true)
}
async function confirmRemove() {
setConfirmDelete(false)
2026-04-23 23:40:34 -03:00
await fetch(`/api/streams/${stream.id}`, { method: "DELETE" })
onRefresh()
}
function openVNC() {
navigate(`/vnc/${stream.id}`)
}
function handlePlayStream() {
navigate(globalPrefs.pureMode
? `/api/hls/live/${stream.id}/index.m3u8`
: `/player/${stream.id}?mode=hls`)
}
function handleRunHtml() {
navigate(globalPrefs.pureMode
? `/player.html?id=${stream.id}`
: `/static/${stream.id}`)
}
function copyRTMP() {
const url = `rtmp://${window.location.hostname}:1935/live/${stream.id}`
copyToClipboard(url).then(() => {
setCopied(true)
setTimeout(() => setCopied(false), 2000)
})
}
async function toggleAutoReload() {
const next = !autoReload
setAutoReload(next)
await fetch(`/api/streams/${stream.id}/autoreload`, {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ enabled: next, interval: autoReloadMins * 60 }),
})
}
async function saveAutoReloadInterval(mins: number) {
await fetch(`/api/streams/${stream.id}/autoreload`, {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ enabled: autoReload, interval: mins * 60 }),
})
}
async function refreshThumb() {
setMenuOpen(false)
setThumbError(false)
setThumbCapturing(true)
if (pollRef.current) clearInterval(pollRef.current)
await fetch(`/api/streams/${stream.id}/thumb`, { method: "POST" })
const deadline = Date.now() + 30000
pollRef.current = setInterval(async () => {
const res = await fetch(`/api/streams/${stream.id}/thumb?t=${Date.now()}`, { cache: "no-store" })
if (res.ok) {
clearInterval(pollRef.current!); pollRef.current = null
setThumbKey((k) => k + 1)
setThumbCapturing(false)
} else if (Date.now() >= deadline) {
clearInterval(pollRef.current!); pollRef.current = null
setThumbCapturing(false)
}
}, 2000)
2026-04-23 23:40:34 -03:00
}
const playBtn = `w-full flex items-center rounded border border-border bg-muted hover:bg-[#2a2a2a] active:bg-[#333] transition-colors cursor-pointer ${sc.btn}`
const menuItem = "w-full flex items-center gap-2 px-3 py-2 text-sm hover:bg-[#2a2a2a] active:bg-[#333] transition-colors cursor-pointer"
2026-04-23 23:40:34 -03:00
return (
<>
{confirmDelete && (
<ConfirmDeleteModal
name={stream.name}
onConfirm={confirmRemove}
onCancel={() => setConfirmDelete(false)}
/>
)}
<div className={cn("relative rounded-lg border border-border bg-card flex flex-col w-full transition-opacity", sc.card, CARD_WIDTHS[cardSize], isDragging && "opacity-40")}>
{/* Drag handle strip */}
{(dragHandleListeners || dragHandleAttributes) && (
<div
{...dragHandleListeners}
{...dragHandleAttributes}
className="-mx-3 -mt-3 h-7 flex items-center justify-center rounded-t-lg cursor-grab active:cursor-grabbing hover:bg-white/[0.05] transition-colors border-b border-border/40 group"
>
<GripVertical className="w-4 h-4 text-muted-foreground/30 group-hover:text-muted-foreground/65 transition-colors" />
</div>
)}
{/* Thumbnail */}
<div className="w-full aspect-video rounded overflow-hidden bg-muted flex items-center justify-center relative">
{thumbCapturing ? (
<span className="text-xs text-muted-foreground animate-pulse">Capturing...</span>
) : (
<>
{thumbError && (
<div className="absolute inset-0 flex items-center justify-center">
<Video className="w-5 h-5 text-muted-foreground/25" />
</div>
)}
<img
key={thumbKey}
src={`/api/streams/${stream.id}/thumb?t=${thumbKey}`}
className={cn("w-full h-full object-cover", thumbError && "invisible")}
onError={() => setThumbError(true)}
onLoad={() => setThumbError(false)}
/>
</>
)}
</div>
2026-04-23 23:40:34 -03:00
<div className="flex items-start justify-between gap-2">
<div className="min-w-0">
<p className={cn("font-semibold truncate", sc.name)}>{stream.name}</p>
<p className={cn("text-muted-foreground font-mono truncate", sc.meta)}>{stream.id}</p>
2026-04-23 23:40:34 -03:00
</div>
<div className="flex items-center gap-1.5 shrink-0">
<StatusBadge status={status} localStatus={localStatus} cardSize={cardSize} />
<div className="relative">
<button onClick={() => setMenuOpen((v) => !v)} className="p-1 rounded hover:bg-[#2a2a2a] transition-colors cursor-pointer">
<MoreHorizontal className={sc.menuIcon} />
</button>
{menuOpen && (
<>
<div className="fixed inset-0 z-40" onClick={() => setMenuOpen(false)} />
<div className="absolute top-full right-0 mt-1 z-50 min-w-[180px] rounded-lg border border-border shadow-2xl overflow-hidden"
style={{ background: "#1c1c1c" }}>
<button onClick={() => { setMenuOpen(false); window.location.href = `/streams/${stream.id}/edit` }} className={menuItem}>
<Pencil className="w-3.5 h-3.5" /> Edit
</button>
<button onClick={() => action("restart", "restarting")} className={menuItem}>
<RotateCcw className="w-3.5 h-3.5" /> Restart
</button>
<button onClick={() => action("recreate", "restarting")} className={menuItem}>
<Wrench className="w-3.5 h-3.5" /> Recreate
</button>
{status?.ffmpeg === "RUNNING" || localStatus === "restarting" ? (
<button onClick={() => action("stop", "stopping")} className={menuItem}>
<Square className="w-3.5 h-3.5" /> Stop
</button>
) : (
<button onClick={() => action("start", "starting")} className={menuItem}>
<Play className="w-3.5 h-3.5" /> Start
</button>
)}
<button onClick={() => { setMenuOpen(false); copyRTMP() }} className={menuItem}>
{copied ? <Check className="w-3.5 h-3.5 text-green-500" /> : <Copy className="w-3.5 h-3.5" />}
{copied ? "Copied!" : "Copy RTMP"}
</button>
<div className="border-t border-border" />
<button onClick={refreshThumb} disabled={thumbCapturing} className={cn(menuItem, thumbCapturing && "opacity-50")}>
<ImageUp className="w-3.5 h-3.5" />
{thumbCapturing ? "Capturing..." : "Refresh thumbnail"}
</button>
<div className="border-t border-border" />
<div className="px-3 py-2 flex flex-col gap-2">
<div className="flex items-center justify-between gap-3">
<span className="text-sm flex items-center gap-2">
<RefreshCw className="w-3.5 h-3.5 shrink-0" /> Auto-reload
</span>
<button
onClick={toggleAutoReload}
className={cn("relative w-9 h-5 rounded-full transition-colors shrink-0 overflow-hidden", autoReload ? "bg-blue-600" : "bg-zinc-600")}
>
<span className={cn("absolute top-0.5 w-4 h-4 rounded-full bg-white transition-all", autoReload ? "left-[18px]" : "left-0.5")} />
</button>
</div>
{autoReload && (
<div className="flex items-center gap-2">
<span className="text-xs text-muted-foreground whitespace-nowrap">Every</span>
<input
type="number"
min={1}
value={autoReloadMins}
onChange={e => setAutoReloadMins(Math.max(1, Number(e.target.value) || 1))}
onBlur={() => saveAutoReloadInterval(autoReloadMins)}
onKeyDown={e => { if (e.key === "Enter") saveAutoReloadInterval(autoReloadMins) }}
className="w-16 text-xs bg-[#2a2a2a] border border-border rounded px-2 py-0.5 text-center"
/>
<span className="text-xs text-muted-foreground whitespace-nowrap">min</span>
</div>
)}
</div>
<div className="border-t border-border" />
<button onClick={remove} className={cn(menuItem, "text-destructive")}>
<Trash2 className="w-3.5 h-3.5" /> Delete
</button>
</div>
</>
)}
</div>
2026-04-23 23:40:34 -03:00
</div>
</div>
<p className={cn("text-muted-foreground truncate", sc.meta)} title={stream.url}>{stream.url}</p>
2026-04-23 23:40:34 -03:00
<div className="flex flex-col gap-1.5">
<button onClick={handlePlayStream} className={playBtn}><Play className={cn("shrink-0", sc.btnIcon)} /> Play Stream</button>
<button onClick={handleRunHtml} className={playBtn}><Globe className={cn("shrink-0", sc.btnIcon)} /> Run HTML</button>
<button onClick={openVNC} className={playBtn}><Monitor className={cn("shrink-0", sc.btnIcon)} /> Open VNC</button>
2026-04-23 23:40:34 -03:00
</div>
</div>
</>
2026-04-23 23:40:34 -03:00
)
}