init repo
This commit is contained in:
410
components/data-table.tsx
Normal file
410
components/data-table.tsx
Normal file
@@ -0,0 +1,410 @@
|
||||
"use client"
|
||||
|
||||
import * as React from "react"
|
||||
import {
|
||||
type ColumnDef,
|
||||
type ColumnFiltersState,
|
||||
type SortingState,
|
||||
type VisibilityState,
|
||||
flexRender,
|
||||
getCoreRowModel,
|
||||
getFilteredRowModel,
|
||||
getPaginationRowModel,
|
||||
getSortedRowModel,
|
||||
useReactTable,
|
||||
} from "@tanstack/react-table"
|
||||
import { ArrowUpDown, ChevronDown, MoreHorizontal } from "lucide-react"
|
||||
|
||||
import { Button } from "@/components/ui/button"
|
||||
import { Checkbox } from "@/components/ui/checkbox"
|
||||
import {
|
||||
DropdownMenu,
|
||||
DropdownMenuCheckboxItem,
|
||||
DropdownMenuContent,
|
||||
DropdownMenuItem,
|
||||
DropdownMenuLabel,
|
||||
DropdownMenuSeparator,
|
||||
DropdownMenuTrigger,
|
||||
} from "@/components/ui/dropdown-menu"
|
||||
import { Input } from "@/components/ui/input"
|
||||
import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table"
|
||||
import { Badge } from "@/components/ui/badge"
|
||||
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar"
|
||||
import { useTranslation } from "@/lib/i18n"
|
||||
|
||||
export type Deal = {
|
||||
id: string
|
||||
dealName: string
|
||||
client: string
|
||||
stage: "Lead" | "Qualified" | "Proposal" | "Negotiation" | "Closed Won" | "Closed Lost"
|
||||
value: number
|
||||
probability: number
|
||||
owner: string
|
||||
ownerAvatar: string
|
||||
lastContacted: string
|
||||
createdAt: string
|
||||
expectedClose: string
|
||||
}
|
||||
|
||||
const data: Deal[] = [
|
||||
{
|
||||
id: "DEAL-001",
|
||||
dealName: "Enterprise Software License",
|
||||
client: "TechCorp Inc.",
|
||||
stage: "Negotiation",
|
||||
value: 45000,
|
||||
probability: 75,
|
||||
owner: "Jane Doe",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1494790108755-2616b612b786?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-15",
|
||||
createdAt: "2023-12-01",
|
||||
expectedClose: "2024-02-15",
|
||||
},
|
||||
{
|
||||
id: "DEAL-002",
|
||||
dealName: "Marketing Automation Setup",
|
||||
client: "StartupXYZ",
|
||||
stage: "Proposal",
|
||||
value: 12500,
|
||||
probability: 60,
|
||||
owner: "Mike Roberts",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-14",
|
||||
createdAt: "2023-12-15",
|
||||
expectedClose: "2024-01-30",
|
||||
},
|
||||
{
|
||||
id: "DEAL-003",
|
||||
dealName: "Cloud Migration Project",
|
||||
client: "Global Solutions",
|
||||
stage: "Qualified",
|
||||
value: 78000,
|
||||
probability: 40,
|
||||
owner: "Sarah Johnson",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1438761681033-6461ffad8d80?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-12",
|
||||
createdAt: "2023-11-20",
|
||||
expectedClose: "2024-03-01",
|
||||
},
|
||||
{
|
||||
id: "DEAL-004",
|
||||
dealName: "CRM Implementation",
|
||||
client: "Retail Chain Co.",
|
||||
stage: "Closed Won",
|
||||
value: 25000,
|
||||
probability: 100,
|
||||
owner: "Alex Lee",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1507003211169-0a1dd7228f2d?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-08",
|
||||
createdAt: "2023-10-15",
|
||||
expectedClose: "2024-01-15",
|
||||
},
|
||||
{
|
||||
id: "DEAL-005",
|
||||
dealName: "Security Audit Services",
|
||||
client: "FinanceFirst Bank",
|
||||
stage: "Lead",
|
||||
value: 35000,
|
||||
probability: 20,
|
||||
owner: "Emily Martinez",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1544005313-94ddf0286df2?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-10",
|
||||
createdAt: "2024-01-05",
|
||||
expectedClose: "2024-04-01",
|
||||
},
|
||||
{
|
||||
id: "DEAL-006",
|
||||
dealName: "Data Analytics Platform",
|
||||
client: "Healthcare Plus",
|
||||
stage: "Negotiation",
|
||||
value: 95000,
|
||||
probability: 80,
|
||||
owner: "David Chen",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1506794778202-cad84cf45f1d?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-16",
|
||||
createdAt: "2023-11-30",
|
||||
expectedClose: "2024-02-28",
|
||||
},
|
||||
{
|
||||
id: "DEAL-007",
|
||||
dealName: "Mobile App Development",
|
||||
client: "E-commerce Co.",
|
||||
stage: "Proposal",
|
||||
value: 55000,
|
||||
probability: 50,
|
||||
owner: "Lisa Wang",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1487412720507-e7ab37603c6f?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-13",
|
||||
createdAt: "2023-12-20",
|
||||
expectedClose: "2024-03-15",
|
||||
},
|
||||
{
|
||||
id: "DEAL-008",
|
||||
dealName: "IT Infrastructure Upgrade",
|
||||
client: "Manufacturing Corp",
|
||||
stage: "Closed Lost",
|
||||
value: 120000,
|
||||
probability: 0,
|
||||
owner: "Tom Wilson",
|
||||
ownerAvatar: "https://images.unsplash.com/photo-1500648767791-00dcc994a43e?w=150&h=150&fit=crop&crop=face",
|
||||
lastContacted: "2024-01-05",
|
||||
createdAt: "2023-09-15",
|
||||
expectedClose: "2024-01-01",
|
||||
},
|
||||
]
|
||||
|
||||
const getStageColor = (stage: Deal["stage"]) => {
|
||||
const colors = {
|
||||
Lead: "bg-gray-100 text-gray-800",
|
||||
Qualified: "bg-blue-100 text-blue-800",
|
||||
Proposal: "bg-yellow-100 text-yellow-800",
|
||||
Negotiation: "bg-orange-100 text-orange-800",
|
||||
"Closed Won": "bg-green-100 text-green-800",
|
||||
"Closed Lost": "bg-red-100 text-red-800",
|
||||
}
|
||||
return colors[stage]
|
||||
}
|
||||
|
||||
function useColumns() {
|
||||
const { t } = useTranslation()
|
||||
|
||||
const columns: ColumnDef<Deal>[] = React.useMemo(() => [
|
||||
{
|
||||
id: "select",
|
||||
header: ({ table }) => (
|
||||
<Checkbox
|
||||
checked={table.getIsAllPageRowsSelected() || (table.getIsSomePageRowsSelected() && "indeterminate")}
|
||||
onCheckedChange={(value) => table.toggleAllPageRowsSelected(!!value)}
|
||||
aria-label="Select all"
|
||||
/>
|
||||
),
|
||||
cell: ({ row }) => (
|
||||
<Checkbox
|
||||
checked={row.getIsSelected()}
|
||||
onCheckedChange={(value) => row.toggleSelected(!!value)}
|
||||
aria-label="Select row"
|
||||
/>
|
||||
),
|
||||
enableSorting: false,
|
||||
enableHiding: false,
|
||||
},
|
||||
{
|
||||
accessorKey: "dealName",
|
||||
header: ({ column }) => (
|
||||
<Button variant="ghost" onClick={() => column.toggleSorting(column.getIsSorted() === "asc")}>
|
||||
{t("dataTable.dealName")}
|
||||
<ArrowUpDown className="ml-2 h-4 w-4" />
|
||||
</Button>
|
||||
),
|
||||
cell: ({ row }) => <div className="font-medium">{row.getValue("dealName")}</div>,
|
||||
},
|
||||
{
|
||||
accessorKey: "client",
|
||||
header: ({ column }) => (
|
||||
<Button variant="ghost" onClick={() => column.toggleSorting(column.getIsSorted() === "asc")}>
|
||||
{t("dataTable.client")}
|
||||
<ArrowUpDown className="ml-2 h-4 w-4" />
|
||||
</Button>
|
||||
),
|
||||
cell: ({ row }) => <div>{row.getValue("client")}</div>,
|
||||
},
|
||||
{
|
||||
accessorKey: "stage",
|
||||
header: t("dataTable.stage"),
|
||||
cell: ({ row }) => {
|
||||
const stage = row.getValue("stage") as Deal["stage"]
|
||||
return <Badge className={getStageColor(stage)}>{stage}</Badge>
|
||||
},
|
||||
},
|
||||
{
|
||||
accessorKey: "value",
|
||||
header: ({ column }) => (
|
||||
<Button variant="ghost" onClick={() => column.toggleSorting(column.getIsSorted() === "asc")}>
|
||||
{t("dataTable.value")}
|
||||
<ArrowUpDown className="ml-2 h-4 w-4" />
|
||||
</Button>
|
||||
),
|
||||
cell: ({ row }) => {
|
||||
const amount = Number.parseFloat(row.getValue("value"))
|
||||
const formatted = new Intl.NumberFormat("en-US", { style: "currency", currency: "USD" }).format(amount)
|
||||
return <div className="font-medium">{formatted}</div>
|
||||
},
|
||||
},
|
||||
{
|
||||
accessorKey: "probability",
|
||||
header: ({ column }) => (
|
||||
<Button variant="ghost" onClick={() => column.toggleSorting(column.getIsSorted() === "asc")}>
|
||||
{t("dataTable.probability")}
|
||||
<ArrowUpDown className="ml-2 h-4 w-4" />
|
||||
</Button>
|
||||
),
|
||||
cell: ({ row }) => <div>{(row.getValue("probability") as number)}%</div>,
|
||||
},
|
||||
{
|
||||
accessorKey: "owner",
|
||||
header: t("dataTable.owner"),
|
||||
cell: ({ row }) => {
|
||||
const owner = row.getValue("owner") as string
|
||||
const ownerAvatar = row.original.ownerAvatar
|
||||
return (
|
||||
<div className="flex items-center gap-2">
|
||||
<Avatar className="h-8 w-8">
|
||||
<AvatarImage src={ownerAvatar || "/placeholder.svg"} alt={owner} className="object-cover" />
|
||||
<AvatarFallback className="text-xs">
|
||||
{owner.split(" ").map((n) => n[0]).join("")}
|
||||
</AvatarFallback>
|
||||
</Avatar>
|
||||
<span className="font-medium">{owner}</span>
|
||||
</div>
|
||||
)
|
||||
},
|
||||
},
|
||||
{
|
||||
accessorKey: "expectedClose",
|
||||
header: ({ column }) => (
|
||||
<Button variant="ghost" onClick={() => column.toggleSorting(column.getIsSorted() === "asc")}>
|
||||
{t("dataTable.expectedClose")}
|
||||
<ArrowUpDown className="ml-2 h-4 w-4" />
|
||||
</Button>
|
||||
),
|
||||
cell: ({ row }) => {
|
||||
const date = new Date(row.getValue("expectedClose"))
|
||||
return <div>{date.toLocaleDateString()}</div>
|
||||
},
|
||||
},
|
||||
{
|
||||
id: "actions",
|
||||
enableHiding: false,
|
||||
cell: ({ row }) => {
|
||||
const deal = row.original
|
||||
return (
|
||||
<DropdownMenu>
|
||||
<DropdownMenuTrigger asChild>
|
||||
<Button variant="ghost" className="h-8 w-8 p-0">
|
||||
<span className="sr-only">Open menu</span>
|
||||
<MoreHorizontal className="h-4 w-4" />
|
||||
</Button>
|
||||
</DropdownMenuTrigger>
|
||||
<DropdownMenuContent align="end">
|
||||
<DropdownMenuLabel>{t("dataTable.actions")}</DropdownMenuLabel>
|
||||
<DropdownMenuItem onClick={() => navigator.clipboard.writeText(deal.id)}>
|
||||
{t("dataTable.copyDealId")}
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuSeparator />
|
||||
<DropdownMenuItem>{t("dataTable.viewDealDetails")}</DropdownMenuItem>
|
||||
<DropdownMenuItem>{t("dataTable.editDeal")}</DropdownMenuItem>
|
||||
<DropdownMenuItem>{t("dataTable.updateStage")}</DropdownMenuItem>
|
||||
<DropdownMenuSeparator />
|
||||
<DropdownMenuItem className="text-red-600">{t("dataTable.deleteDeal")}</DropdownMenuItem>
|
||||
</DropdownMenuContent>
|
||||
</DropdownMenu>
|
||||
)
|
||||
},
|
||||
},
|
||||
], [t])
|
||||
|
||||
return columns
|
||||
}
|
||||
|
||||
export function DataTable() {
|
||||
const { t } = useTranslation()
|
||||
const columns = useColumns()
|
||||
const [sorting, setSorting] = React.useState<SortingState>([])
|
||||
const [columnFilters, setColumnFilters] = React.useState<ColumnFiltersState>([])
|
||||
const [columnVisibility, setColumnVisibility] = React.useState<VisibilityState>({})
|
||||
const [rowSelection, setRowSelection] = React.useState({})
|
||||
|
||||
const table = useReactTable({
|
||||
data,
|
||||
columns,
|
||||
onSortingChange: setSorting,
|
||||
onColumnFiltersChange: setColumnFilters,
|
||||
getCoreRowModel: getCoreRowModel(),
|
||||
getPaginationRowModel: getPaginationRowModel(),
|
||||
getSortedRowModel: getSortedRowModel(),
|
||||
getFilteredRowModel: getFilteredRowModel(),
|
||||
onColumnVisibilityChange: setColumnVisibility,
|
||||
onRowSelectionChange: setRowSelection,
|
||||
state: { sorting, columnFilters, columnVisibility, rowSelection },
|
||||
})
|
||||
|
||||
return (
|
||||
<div className="w-full">
|
||||
<div className="flex items-center py-4">
|
||||
<Input
|
||||
placeholder={t("dataTable.filterDeals")}
|
||||
value={(table.getColumn("dealName")?.getFilterValue() as string) ?? ""}
|
||||
onChange={(event) => table.getColumn("dealName")?.setFilterValue(event.target.value)}
|
||||
className="max-w-sm"
|
||||
/>
|
||||
<DropdownMenu>
|
||||
<DropdownMenuTrigger asChild>
|
||||
<Button variant="outline" className="ml-auto">
|
||||
{t("dataTable.columns")} <ChevronDown className="ml-2 h-4 w-4" />
|
||||
</Button>
|
||||
</DropdownMenuTrigger>
|
||||
<DropdownMenuContent align="end">
|
||||
{table
|
||||
.getAllColumns()
|
||||
.filter((column) => column.getCanHide())
|
||||
.map((column) => (
|
||||
<DropdownMenuCheckboxItem
|
||||
key={column.id}
|
||||
className="capitalize"
|
||||
checked={column.getIsVisible()}
|
||||
onCheckedChange={(value) => column.toggleVisibility(!!value)}
|
||||
>
|
||||
{column.id}
|
||||
</DropdownMenuCheckboxItem>
|
||||
))}
|
||||
</DropdownMenuContent>
|
||||
</DropdownMenu>
|
||||
</div>
|
||||
<div className="rounded-md border">
|
||||
<Table>
|
||||
<TableHeader>
|
||||
{table.getHeaderGroups().map((headerGroup) => (
|
||||
<TableRow key={headerGroup.id}>
|
||||
{headerGroup.headers.map((header) => (
|
||||
<TableHead key={header.id}>
|
||||
{header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())}
|
||||
</TableHead>
|
||||
))}
|
||||
</TableRow>
|
||||
))}
|
||||
</TableHeader>
|
||||
<TableBody>
|
||||
{table.getRowModel().rows?.length ? (
|
||||
table.getRowModel().rows.map((row) => (
|
||||
<TableRow key={row.id} data-state={row.getIsSelected() && "selected"}>
|
||||
{row.getVisibleCells().map((cell) => (
|
||||
<TableCell key={cell.id}>{flexRender(cell.column.columnDef.cell, cell.getContext())}</TableCell>
|
||||
))}
|
||||
</TableRow>
|
||||
))
|
||||
) : (
|
||||
<TableRow>
|
||||
<TableCell colSpan={columns.length} className="h-24 text-center">
|
||||
{t("dataTable.noResults")}
|
||||
</TableCell>
|
||||
</TableRow>
|
||||
)}
|
||||
</TableBody>
|
||||
</Table>
|
||||
</div>
|
||||
<div className="flex items-center justify-end space-x-2 py-4">
|
||||
<div className="flex-1 text-sm text-muted-foreground">
|
||||
{table.getFilteredSelectedRowModel().rows.length} / {table.getFilteredRowModel().rows.length} {t("dataTable.rowsSelected")}
|
||||
</div>
|
||||
<div className="space-x-2">
|
||||
<Button variant="outline" size="sm" onClick={() => table.previousPage()} disabled={!table.getCanPreviousPage()}>
|
||||
{t("dataTable.previous")}
|
||||
</Button>
|
||||
<Button variant="outline" size="sm" onClick={() => table.nextPage()} disabled={!table.getCanNextPage()}>
|
||||
{t("dataTable.next")}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
Reference in New Issue
Block a user