add items per page
This commit is contained in:
@@ -64,6 +64,46 @@ interface StatusConfig {
|
||||
|
||||
type OrderStatus = "paid" | "unpaid" | "acknowledged" | "shipped" | "completed" | "cancelled" | "confirming";
|
||||
|
||||
// Create a StatusFilter component to replace the missing component
|
||||
const StatusFilter = ({ currentStatus, onChange }: { currentStatus: string, onChange: (value: string) => void }) => {
|
||||
return (
|
||||
<Select value={currentStatus} onValueChange={onChange}>
|
||||
<SelectTrigger className="w-40">
|
||||
<SelectValue placeholder="Filter Status" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="all">All Statuses</SelectItem>
|
||||
{["paid", "unpaid", "acknowledged", "shipped", "completed", "cancelled", "confirming"].map(status => (
|
||||
<SelectItem key={status} value={status}>
|
||||
{status.charAt(0).toUpperCase() + status.slice(1)}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
);
|
||||
};
|
||||
|
||||
// Create a PageSizeSelector component
|
||||
const PageSizeSelector = ({ currentSize, onChange, options }: { currentSize: number, onChange: (value: string) => void, options: number[] }) => {
|
||||
return (
|
||||
<div className="flex items-center gap-2">
|
||||
<div className="text-sm font-medium text-muted-foreground">Show:</div>
|
||||
<Select value={currentSize.toString()} onValueChange={onChange}>
|
||||
<SelectTrigger className="w-24">
|
||||
<SelectValue placeholder="Page Size" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
{options.map(size => (
|
||||
<SelectItem key={size} value={size.toString()}>
|
||||
{size}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default function OrderTable() {
|
||||
const [orders, setOrders] = useState<Order[]>([]);
|
||||
const [loading, setLoading] = useState(true);
|
||||
@@ -77,7 +117,8 @@ export default function OrderTable() {
|
||||
}>({ column: "orderDate", direction: "desc" });
|
||||
const [selectedOrders, setSelectedOrders] = useState<Set<string>>(new Set());
|
||||
const [isShipping, setIsShipping] = useState(false);
|
||||
const ITEMS_PER_PAGE = 10;
|
||||
const [itemsPerPage, setItemsPerPage] = useState<number>(20);
|
||||
const pageSizeOptions = [5, 10, 15, 20, 25, 50, 75, 100];
|
||||
|
||||
// Fetch orders with server-side pagination
|
||||
const fetchOrders = useCallback(async () => {
|
||||
@@ -85,7 +126,7 @@ export default function OrderTable() {
|
||||
setLoading(true);
|
||||
const queryParams = new URLSearchParams({
|
||||
page: currentPage.toString(),
|
||||
limit: ITEMS_PER_PAGE.toString(),
|
||||
limit: itemsPerPage.toString(),
|
||||
...(statusFilter !== "all" && { status: statusFilter }),
|
||||
});
|
||||
|
||||
@@ -100,7 +141,7 @@ export default function OrderTable() {
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
}, [currentPage, statusFilter]);
|
||||
}, [currentPage, statusFilter, itemsPerPage]);
|
||||
|
||||
useEffect(() => {
|
||||
fetchOrders();
|
||||
@@ -115,6 +156,12 @@ export default function OrderTable() {
|
||||
setCurrentPage(newPage);
|
||||
};
|
||||
|
||||
const handleItemsPerPageChange = (e: React.ChangeEvent<HTMLSelectElement>) => {
|
||||
const newSize = parseInt(e.target.value, 10);
|
||||
setItemsPerPage(newSize);
|
||||
setCurrentPage(1); // Reset to first page when changing items per page
|
||||
};
|
||||
|
||||
// Derived data calculations
|
||||
const filteredOrders = orders.filter(
|
||||
(order) => statusFilter === "all" || order.status === statusFilter
|
||||
@@ -138,8 +185,8 @@ export default function OrderTable() {
|
||||
});
|
||||
|
||||
const paginatedOrders = sortedOrders.slice(
|
||||
(currentPage - 1) * ITEMS_PER_PAGE,
|
||||
currentPage * ITEMS_PER_PAGE
|
||||
(currentPage - 1) * itemsPerPage,
|
||||
currentPage * itemsPerPage
|
||||
);
|
||||
|
||||
// Handlers
|
||||
@@ -249,28 +296,25 @@ export default function OrderTable() {
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="rounded-lg border shadow-sm overflow-hidden">
|
||||
{/* Toolbar */}
|
||||
<div className="p-4 flex justify-between items-center border-b">
|
||||
<div className="flex items-center gap-4">
|
||||
<Select value={statusFilter} onValueChange={setStatusFilter}>
|
||||
<SelectTrigger className="w-40">
|
||||
<SelectValue placeholder="Filter Status" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="all">All Statuses</SelectItem>
|
||||
{Object.keys(statusConfig).map(status => (
|
||||
<SelectItem key={status} value={status}>
|
||||
{status.charAt(0).toUpperCase() + status.slice(1)}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
<div className="text-sm text-muted-foreground">
|
||||
Total Orders: {totalOrders}
|
||||
</div>
|
||||
<div className="space-y-4">
|
||||
<div className="border border-zinc-800 rounded-md bg-black/40 overflow-hidden">
|
||||
{/* Filters header */}
|
||||
<div className="p-4 border-b border-zinc-800 bg-black/60">
|
||||
<div className="flex flex-col sm:flex-row justify-between items-start sm:items-center gap-4">
|
||||
<div className="flex flex-col sm:flex-row gap-2 sm:items-center">
|
||||
<StatusFilter
|
||||
currentStatus={statusFilter}
|
||||
onChange={setStatusFilter}
|
||||
/>
|
||||
|
||||
<PageSizeSelector
|
||||
currentSize={itemsPerPage}
|
||||
onChange={(value) => handleItemsPerPageChange({ target: { value } } as React.ChangeEvent<HTMLSelectElement>)}
|
||||
options={pageSizeOptions}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="flex items-center gap-2 self-end sm:self-auto">
|
||||
<AlertDialog>
|
||||
<AlertDialogTrigger asChild>
|
||||
<Button disabled={selectedOrders.size === 0 || isShipping}>
|
||||
@@ -299,6 +343,8 @@ export default function OrderTable() {
|
||||
</AlertDialogContent>
|
||||
</AlertDialog>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Table */}
|
||||
<div className="relative">
|
||||
@@ -307,8 +353,9 @@ export default function OrderTable() {
|
||||
<Loader2 className="h-6 w-6 animate-spin" />
|
||||
</div>
|
||||
)}
|
||||
<Table>
|
||||
<TableHeader className="bg-muted">
|
||||
<div className="max-h-[calc(100vh-300px)] overflow-auto">
|
||||
<Table className="[&_tr]:border-b [&_tr]:border-zinc-800 [&_tr:last-child]:border-b-0 [&_td]:border-r [&_td]:border-zinc-800 [&_td:last-child]:border-r-0 [&_th]:border-r [&_th]:border-zinc-800 [&_th:last-child]:border-r-0 [&_tr:hover]:bg-zinc-900/70">
|
||||
<TableHeader className="bg-black/60 sticky top-0 z-10">
|
||||
<TableRow>
|
||||
<TableHead className="w-12">
|
||||
<Checkbox
|
||||
@@ -329,7 +376,7 @@ export default function OrderTable() {
|
||||
Date <ArrowUpDown className="ml-2 inline h-4 w-4" />
|
||||
</TableHead>
|
||||
<TableHead>Buyer</TableHead>
|
||||
<TableHead className="w-24">Actions</TableHead>
|
||||
<TableHead className="w-24 text-center">Actions</TableHead>
|
||||
</TableRow>
|
||||
</TableHeader>
|
||||
<TableBody>
|
||||
@@ -363,8 +410,8 @@ export default function OrderTable() {
|
||||
<TableCell>
|
||||
{order.telegramUsername ? `@${order.telegramUsername}` : "-"}
|
||||
</TableCell>
|
||||
<TableCell className="text-right">
|
||||
<Button variant="ghost" size="sm" asChild>
|
||||
<TableCell className="text-center">
|
||||
<Button variant="ghost" size="sm" className="mx-auto" asChild>
|
||||
<Link href={`/dashboard/orders/${order._id}`}>
|
||||
<Eye className="h-4 w-4" />
|
||||
</Link>
|
||||
@@ -376,11 +423,12 @@ export default function OrderTable() {
|
||||
</TableBody>
|
||||
</Table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Pagination */}
|
||||
<div className="flex items-center justify-between px-4 py-4 border-t">
|
||||
<div className="flex items-center justify-between px-4 py-4 border-t border-zinc-800 bg-black/40">
|
||||
<div className="text-sm text-muted-foreground">
|
||||
Page {currentPage} of {totalPages}
|
||||
Page {currentPage} of {totalPages} ({totalOrders} total)
|
||||
</div>
|
||||
<div className="flex gap-2">
|
||||
<Button
|
||||
@@ -404,5 +452,6 @@ export default function OrderTable() {
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -9,7 +9,7 @@ const Table = React.forwardRef<
|
||||
<div className="relative w-full overflow-auto">
|
||||
<table
|
||||
ref={ref}
|
||||
className={cn("w-full caption-bottom text-sm", className)}
|
||||
className={cn("w-full caption-bottom text-sm border-collapse", className)}
|
||||
{...props}
|
||||
/>
|
||||
</div>
|
||||
@@ -58,7 +58,7 @@ const TableRow = React.forwardRef<
|
||||
<tr
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"border-b transition-colors hover:bg-muted/50 data-[state=selected]:bg-muted",
|
||||
"transition-colors hover:bg-muted/20 data-[state=selected]:bg-muted",
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
|
||||
Reference in New Issue
Block a user