"use client" import * as React from "react" import { type DataTableRowAction } from "@/types/table" import { type ColumnDef } from "@tanstack/react-table" import { Ellipsis, Paperclip, Package } from "lucide-react" import { toast } from "sonner" import { formatDate } from "@/lib/utils" import { Badge } from "@/components/ui/badge" import { Button } from "@/components/ui/button" import { Checkbox } from "@/components/ui/checkbox" import { DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuRadioGroup, DropdownMenuRadioItem, DropdownMenuSeparator, DropdownMenuShortcut, DropdownMenuSub, DropdownMenuSubContent, DropdownMenuSubTrigger, DropdownMenuTrigger, } from "@/components/ui/dropdown-menu" import { getRFQStatusIcon } from "@/lib/tasks/utils" import { rfqsColumnsConfig } from "@/config/rfqsColumnsConfig" import { RfqWithItemCount } from "@/db/schema/rfq" import { DataTableColumnHeaderSimple } from "@/components/data-table/data-table-column-simple-header" import { useRouter } from "next/navigation" type NextRouter = ReturnType; interface GetColumnsProps { setRowAction: React.Dispatch< React.SetStateAction | null> > openItemsModal: (rfqId: number) => void openAttachmentsSheet: (rfqId: number) => void router: NextRouter } /** * tanstack table 컬럼 정의 (중첩 헤더 버전) */ export function getColumns({ setRowAction, openItemsModal, openAttachmentsSheet, router, }: GetColumnsProps): ColumnDef[] { // ---------------------------------------------------------------- // 1) select 컬럼 (체크박스) // ---------------------------------------------------------------- const selectColumn: ColumnDef = { id: "select", header: ({ table }) => ( table.toggleAllPageRowsSelected(!!value)} aria-label="Select all" className="translate-y-0.5" /> ), cell: ({ row }) => ( row.toggleSelected(!!value)} aria-label="Select row" className="translate-y-0.5" /> ), size: 40, enableSorting: false, enableHiding: false, } // ---------------------------------------------------------------- // 2) actions 컬럼 (Dropdown 메뉴) // ---------------------------------------------------------------- const actionsColumn: ColumnDef = { id: "actions", enableHiding: false, cell: function Cell({ row }) { // Proceed 버튼 클릭 시 호출되는 함수 const handleProceed = () => { const rfq = row.original const itemCount = Number(rfq.itemCount || 0) const attachCount = Number(rfq.attachCount || 0) // 아이템과 첨부파일이 모두 0보다 커야 진행 가능 if (itemCount > 0 && attachCount > 0) { router.push( `/evcp/rfq-tech/${rfq.rfqId}` ) } else { // 조건을 충족하지 않는 경우 토스트 알림 표시 if (itemCount === 0 && attachCount === 0) { toast.error("아이템과 첨부파일을 먼저 추가해주세요.") } else if (itemCount === 0) { toast.error("아이템을 먼저 추가해주세요.") } else { toast.error("첨부파일을 먼저 추가해주세요.") } } } return ( setRowAction({ row, type: "update" })} > Edit {row.original.status ==="DRAFT"?"Proceed":"View Detail"} setRowAction({ row, type: "delete" })} > Delete ⌘⌫ ) }, size: 40, } // ---------------------------------------------------------------- // 3) itemsColumn (아이템 개수 표시: 아이콘 + Badge) // ---------------------------------------------------------------- const itemsColumn: ColumnDef = { id: "items", header: "Items", cell: ({ row }) => { const rfq = row.original const itemCount = rfq.itemCount || 0 const handleClick = () => { openItemsModal(rfq.rfqId) } return ( ) }, enableSorting: false, size: 60, } // ---------------------------------------------------------------- // 4) attachmentsColumn (첨부파일 개수 표시: 아이콘 + Badge) // ---------------------------------------------------------------- const attachmentsColumn: ColumnDef = { id: "attachments", header: "Attachments", cell: ({ row }) => { const fileCount = row.original.attachCount ?? 0 const handleClick = () => { openAttachmentsSheet(row.original.rfqId) } return ( ) }, enableSorting: false, size: 60, } // ---------------------------------------------------------------- // 5) 일반 컬럼들을 "그룹"별로 묶어 중첩 columns 생성 // ---------------------------------------------------------------- const groupMap: Record[]> = {} rfqsColumnsConfig.forEach((cfg) => { const groupName = cfg.group || "_noGroup" if (!groupMap[groupName]) { groupMap[groupName] = [] } // child column 정의 const childCol: ColumnDef = { accessorKey: cfg.id, enableResizing: true, header: ({ column }) => ( ), meta: { excelHeader: cfg.excelHeader, group: cfg.group, type: cfg.type, }, cell: ({ row, cell }) => { if (cfg.id === "status") { const statusVal = row.original.status if (!statusVal) return null const Icon = getRFQStatusIcon( statusVal as "DRAFT" | "PUBLISHED" | "EVALUATION" | "AWARDED" ) return (
) } if (cfg.id === "createdAt" || cfg.id === "updatedAt") { const dateVal = cell.getValue() as Date return formatDate(dateVal) } return row.getValue(cfg.id) ?? "" }, } groupMap[groupName].push(childCol) }) // groupMap -> nestedColumns const nestedColumns: ColumnDef[] = [] Object.entries(groupMap).forEach(([groupName, colDefs]) => { if (groupName === "_noGroup") { nestedColumns.push(...colDefs) } else { nestedColumns.push({ id: groupName, header: groupName, columns: colDefs, }) } }) // ---------------------------------------------------------------- // 6) 최종 컬럼 배열 // ---------------------------------------------------------------- return [ selectColumn, ...nestedColumns, attachmentsColumn, // 첨부파일 actionsColumn, itemsColumn, // 아이템 ] }