Files
gg-backend/frontend/src/pages/Appointment.tsx
T

392 lines
11 KiB
TypeScript
Raw Normal View History

2026-04-08 16:30:50 +05:30
import { useState, useEffect, useCallback } from "react";
2026-03-19 13:12:04 +05:30
2026-04-08 16:30:50 +05:30
import { getAppointmentsApi, deleteAppointmentApi } from "@/api/appointment";
import { exportToExcel } from "@/utils/exportToExcel";
2026-03-19 13:12:04 +05:30
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "@/components/ui/table";
2026-04-08 16:30:50 +05:30
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
import { Button } from "@/components/ui/button";
import { Input } from "@/components/ui/input";
import {
Dialog,
DialogContent,
DialogHeader,
DialogTitle,
DialogFooter,
} from "@/components/ui/dialog";
2026-03-19 13:12:04 +05:30
2026-04-08 16:30:50 +05:30
import {
Loader2,
Trash,
RefreshCw,
Download,
ChevronLeft,
ChevronRight,
Eye,
} from "lucide-react";
2026-03-19 13:12:04 +05:30
export default function AppointmentPage() {
const [appointments, setAppointments] = useState<any[]>([]);
const [loading, setLoading] = useState(true);
const [searchText, setSearchText] = useState("");
const [filterDoctor, setFilterDoctor] = useState("");
const [filterDate, setFilterDate] = useState("");
2026-04-08 16:30:50 +05:30
const [viewOpen, setViewOpen] = useState(false);
const [viewData, setViewData] = useState<any>(null);
const [currentPage, setCurrentPage] = useState(1);
const [totalPages, setTotalPages] = useState(1);
const [totalItems, setTotalItems] = useState(0);
const [itemsPerPage, setItemsPerPage] = useState(10);
2026-04-08 16:30:50 +05:30
2026-03-19 13:12:04 +05:30
const fetchAll = useCallback(async () => {
setLoading(true);
try {
const res = await getAppointmentsApi(
currentPage,
itemsPerPage,
filterDate,
searchText,
);
2026-03-19 13:12:04 +05:30
setAppointments(res?.data || []);
setTotalPages(res?.pagination?.totalPages || 1);
setTotalItems(res?.pagination?.total || 0);
2026-03-19 13:12:04 +05:30
} catch (err) {
console.error(err);
} finally {
setLoading(false);
}
}, [currentPage, itemsPerPage, filterDate, searchText]);
2026-03-19 13:12:04 +05:30
useEffect(() => {
fetchAll();
}, [fetchAll]);
const filteredAppointments = appointments.filter((item) => {
const matchesDoctor = filterDoctor
? item.doctor?.name?.toLowerCase().includes(filterDoctor.toLowerCase())
: true;
return matchesDoctor;
2026-03-19 13:12:04 +05:30
});
2026-04-08 16:30:50 +05:30
useEffect(() => {
setCurrentPage(1);
}, [searchText, filterDoctor, filterDate]);
const indexOfFirstItem = (currentPage - 1) * itemsPerPage;
2026-04-08 16:30:50 +05:30
function openView(item: any) {
setViewData(item);
setViewOpen(true);
}
2026-03-19 13:12:04 +05:30
async function handleDelete(id: number) {
if (!confirm("Delete appointment?")) return;
await deleteAppointmentApi(id);
fetchAll();
}
const handleExport = () => {
const exportData = filteredAppointments.map((item) => ({
ID: item.id,
Name: item.name,
Phone: item.mobileNumber,
Email: item.email,
Doctor: item.doctor?.name,
Department: item.department?.name,
Date: new Date(item.date).toLocaleDateString(),
Message: item.message,
}));
exportToExcel(exportData, "appointments");
};
return (
<div className="p-6 space-y-6">
2026-04-08 16:30:50 +05:30
<div className="flex flex-col md:flex-row md:justify-between md:items-center gap-4">
<h1 className="text-3xl font-bold">Appointments</h1>
2026-03-19 13:12:04 +05:30
2026-04-08 16:30:50 +05:30
<div className="flex flex-wrap gap-3">
2026-03-19 13:12:04 +05:30
<Input
2026-04-08 16:30:50 +05:30
placeholder="Search name / phone..."
2026-03-19 13:12:04 +05:30
value={searchText}
onChange={(e) => {
setSearchText(e.target.value);
setCurrentPage(1);
}}
2026-04-08 16:30:50 +05:30
className="w-[220px] text-base"
2026-03-19 13:12:04 +05:30
/>
<Input
type="date"
value={filterDate}
onChange={(e) => {
setFilterDate(e.target.value);
setCurrentPage(1);
}}
2026-04-08 16:30:50 +05:30
className="w-[160px] text-base"
2026-03-19 13:12:04 +05:30
/>
<select
value={itemsPerPage}
onChange={(e) => {
setItemsPerPage(Number(e.target.value));
setCurrentPage(1);
}}
className="flex h-10 rounded-md border border-input bg-background px-3 py-2 text-sm focus:ring-2 focus:ring-primary"
>
<option value={5}>5 / page</option>
<option value={10}>10 / page</option>
<option value={20}>20 / page</option>
</select>
2026-04-08 16:30:50 +05:30
<Button
variant="outline"
onClick={fetchAll}
disabled={loading}
className="text-base"
>
<RefreshCw className="mr-2 h-5 w-5" />
2026-03-19 13:12:04 +05:30
Refresh
</Button>
2026-04-08 16:30:50 +05:30
<Button onClick={handleExport} className="text-base">
<Download className="mr-2 h-5 w-5" />
2026-03-19 13:12:04 +05:30
Export
</Button>
</div>
</div>
<Card>
<CardHeader>
2026-04-08 16:30:50 +05:30
<CardTitle className="text-xl">Appointment List</CardTitle>
2026-03-19 13:12:04 +05:30
</CardHeader>
2026-04-08 16:30:50 +05:30
<CardContent className="p-0 sm:p-6 space-y-4">
<div className="rounded-md border overflow-x-auto overflow-y-auto max-h-[650px] relative">
<Table className="w-full min-w-[1000px] table-fixed border-separate border-spacing-0">
<TableHeader className="sticky top-0 z-20 bg-background shadow-sm">
2026-03-19 13:12:04 +05:30
<TableRow>
2026-04-08 16:30:50 +05:30
<TableHead className="w-[60px] bg-background font-bold text-sm">
ID
</TableHead>
<TableHead className="w-[200px] bg-background font-bold text-sm">
Patient
</TableHead>
<TableHead className="w-[180px] bg-background font-bold text-sm">
Doctor
</TableHead>
<TableHead className="w-[150px] bg-background font-bold text-sm">
Date
</TableHead>
<TableHead className="w-[250px] bg-background font-bold text-sm">
Message
</TableHead>
<TableHead className="w-[120px] bg-background font-bold text-right text-sm">
Actions
</TableHead>
2026-03-19 13:12:04 +05:30
</TableRow>
</TableHeader>
<TableBody>
{loading ? (
<TableRow>
2026-04-08 16:30:50 +05:30
<TableCell colSpan={6} className="text-center py-10">
<Loader2 className="h-8 w-8 animate-spin mx-auto" />
2026-03-19 13:12:04 +05:30
</TableCell>
</TableRow>
) : filteredAppointments.length === 0 ? (
2026-03-19 13:12:04 +05:30
<TableRow>
2026-04-08 16:30:50 +05:30
<TableCell
colSpan={6}
className="text-center text-muted-foreground py-10 text-base"
>
2026-03-19 13:12:04 +05:30
No appointments found
</TableCell>
</TableRow>
) : (
filteredAppointments.map((item) => (
2026-04-08 16:30:50 +05:30
<TableRow key={item.id} className="hover:bg-muted/50">
<TableCell className="font-mono text-xs">
{item.id}
</TableCell>
2026-03-19 13:12:04 +05:30
<TableCell>
2026-04-08 16:30:50 +05:30
<div className="font-semibold text-base truncate">
{item.name}
</div>
<div className="text-xs text-muted-foreground">
{item.mobileNumber}
</div>
2026-03-19 13:12:04 +05:30
</TableCell>
2026-04-08 16:30:50 +05:30
<TableCell>
<div className="text-sm font-medium">
{item.doctor?.name || "-"}
</div>
<div className="text-[10px] text-muted-foreground truncate">
{item.department?.name}
</div>
2026-03-19 13:12:04 +05:30
</TableCell>
<TableCell>
2026-04-08 16:30:50 +05:30
<div className="text-sm">
{new Date(item.date).toLocaleDateString()}
</div>
2026-03-19 13:12:04 +05:30
</TableCell>
<TableCell>
2026-04-08 16:30:50 +05:30
<div className="text-sm line-clamp-2 text-muted-foreground italic">
{item.message || "-"}
</div>
</TableCell>
<TableCell className="text-right">
<div className="flex justify-end gap-2">
<Button
size="icon"
variant="ghost"
className="h-9 w-9"
onClick={() => openView(item)}
>
<Eye className="h-4 w-4" />
</Button>
<Button
size="icon"
variant="ghost"
className="h-9 w-9 text-destructive hover:text-destructive hover:bg-destructive/10"
onClick={() => handleDelete(item.id)}
>
<Trash className="h-4 w-4" />
</Button>
</div>
2026-03-19 13:12:04 +05:30
</TableCell>
</TableRow>
))
)}
</TableBody>
</Table>
</div>
2026-04-08 16:30:50 +05:30
{!loading && totalItems > 0 && (
2026-04-08 16:30:50 +05:30
<div className="flex items-center justify-between px-2 py-6 border-t">
<div className="text-base text-muted-foreground">
Showing{" "}
<span className="font-semibold">{indexOfFirstItem + 1}</span> to{" "}
<span className="font-semibold">
{Math.min(currentPage * itemsPerPage, totalItems)}
2026-04-08 16:30:50 +05:30
</span>{" "}
of <span className="font-semibold">{totalItems}</span>
2026-04-08 16:30:50 +05:30
</div>
<div className="flex items-center gap-6">
<div className="text-base font-semibold">
Page {currentPage} of {totalPages}
</div>
<div className="flex gap-2">
<Button
variant="outline"
size="icon"
className="h-10 w-10"
onClick={() =>
setCurrentPage((prev) => Math.max(prev - 1, 1))
}
disabled={currentPage === 1}
>
<ChevronLeft className="h-5 w-5" />
</Button>
<Button
variant="outline"
size="icon"
className="h-10 w-10"
onClick={() =>
setCurrentPage((prev) => Math.min(prev + 1, totalPages))
}
disabled={currentPage === totalPages || totalPages === 0}
>
<ChevronRight className="h-5 w-5" />
</Button>
</div>
</div>
</div>
)}
2026-03-19 13:12:04 +05:30
</CardContent>
</Card>
2026-04-08 16:30:50 +05:30
<Dialog open={viewOpen} onOpenChange={setViewOpen}>
<DialogContent className="w-full !max-w-3xl max-h-[85vh] overflow-y-auto">
<DialogHeader>
<DialogTitle className="text-2xl border-b pb-2">
Appointment Details
</DialogTitle>
</DialogHeader>
{viewData && (
<div className="space-y-6 py-4">
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<div className="space-y-4">
<div>
<p className="text-xs uppercase font-bold text-muted-foreground">
Patient Information
</p>
<p className="text-lg font-bold text-primary">
{viewData.name}
</p>
<p className="text-sm">{viewData.mobileNumber}</p>
<p className="text-sm">
{viewData.email || "No email provided"}
</p>
</div>
<div>
<p className="text-xs uppercase font-bold text-muted-foreground">
Appointment Date
</p>
<p className="text-base font-semibold">
{new Date(viewData.date).toLocaleDateString()}
</p>
<p className="text-[10px] text-muted-foreground">
Booked on: {new Date(viewData.createdAt).toLocaleString()}
</p>
</div>
</div>
<div className="space-y-4">
<div>
<p className="text-xs uppercase font-bold text-muted-foreground">
Doctor / Department
</p>
<p className="text-base font-bold">
{viewData.doctor?.name || "Not Assigned"}
</p>
<p className="text-sm text-muted-foreground">
{viewData.department?.name || "General"}
</p>
</div>
<div className="p-4 bg-muted/30 rounded-lg">
<p className="text-xs uppercase font-bold text-muted-foreground mb-2">
Message from Patient
</p>
<p className="text-sm italic leading-relaxed whitespace-pre-wrap">
{viewData.message || "No message provided."}
</p>
</div>
</div>
</div>
</div>
)}
<DialogFooter>
<Button
onClick={() => setViewOpen(false)}
className="w-full md:w-auto"
>
Close
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
2026-03-19 13:12:04 +05:30
</div>
);
}