import { useQuery } from "@tanstack/react-query"; import { HardDrive, Plus, RotateCcw } from "lucide-react"; import { useState } from "react"; import { useNavigate } from "react-router"; import { EmptyState } from "~/client/components/empty-state"; import { StatusDot } from "~/client/components/status-dot"; import { Button } from "~/client/components/ui/button"; import { Card } from "~/client/components/ui/card"; import { Input } from "~/client/components/ui/input"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "~/client/components/ui/select"; import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "~/client/components/ui/table"; import { VolumeIcon } from "~/client/components/volume-icon"; import type { Route } from "./+types/volumes"; import { listVolumes } from "~/client/api-client"; import { listVolumesOptions } from "~/client/api-client/@tanstack/react-query.gen"; export const handle = { breadcrumb: () => [{ label: "Volumes" }], }; export function meta(_: Route.MetaArgs) { return [ { title: "Zerobyte - Volumes" }, { name: "description", content: "Create, manage, monitor, and automate your Docker volumes with ease.", }, ]; } export const clientLoader = async () => { const volumes = await listVolumes(); if (volumes.data) return volumes.data; return []; }; export default function Volumes({ loaderData }: Route.ComponentProps) { const [searchQuery, setSearchQuery] = useState(""); const [statusFilter, setStatusFilter] = useState(""); const [backendFilter, setBackendFilter] = useState(""); const clearFilters = () => { setSearchQuery(""); setStatusFilter(""); setBackendFilter(""); }; const navigate = useNavigate(); const { data } = useQuery({ ...listVolumesOptions(), initialData: loaderData, refetchInterval: 10000, refetchOnWindowFocus: true, }); const filteredVolumes = data.filter((volume) => { const matchesSearch = volume.name.toLowerCase().includes(searchQuery.toLowerCase()); const matchesStatus = !statusFilter || volume.status === statusFilter; const matchesBackend = !backendFilter || volume.type === backendFilter; return matchesSearch && matchesStatus && matchesBackend; }) || []; const hasNoVolumes = data.length === 0; const hasNoFilteredVolumes = filteredVolumes.length === 0 && !hasNoVolumes; if (hasNoVolumes) { return ( navigate("/volumes/create")}> Create Volume } /> ); } return (
setSearchQuery(e.target.value)} /> {(searchQuery || statusFilter || backendFilter) && ( )}
Name Backend Status {hasNoFilteredVolumes ? (

No volumes match your filters.

) : ( filteredVolumes.map((volume) => ( navigate(`/volumes/${volume.name}`)} > {volume.name} )) )}
{hasNoFilteredVolumes ? ( "No volumes match filters." ) : ( {filteredVolumes.length} volume {filteredVolumes.length > 1 ? "s" : ""} )}
); }