"use client" import * as React from "react" import { type DataTableRowAction } from "@/types/table" import { type ColumnDef } from "@tanstack/react-table" import { Ellipsis } from "lucide-react" import { toast } from "sonner" import { getErrorMessage } from "@/lib/handle-error" 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 { DataTableColumnHeader } from "@/components/data-table/data-table-column-header" import { useRouter } from "next/navigation" import { vendors } from "@/db/schema/vendors" import { DataTableColumnHeaderSimple } from "@/components/data-table/data-table-column-simple-header" import { vendorColumnsConfig } from "@/config/vendorColumnsConfig" import { Separator } from "@/components/ui/separator" import { MatchedVendorRow, vendorRfqColumnsConfig } from "@/config/vendorRfbColumnsConfig" type NextRouter = ReturnType; interface GetColumnsProps { setRowAction: React.Dispatch | null>>; router: NextRouter; openCommentSheet: (rfqId: number) => void; } /** * tanstack table 컬럼 정의 (중첩 헤더 버전) */ export function getColumns({ setRowAction, router, openCommentSheet }: 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 메뉴) // ---------------------------------------------------------------- // ---------------------------------------------------------------- // 3) 일반 컬럼들을 "그룹"별로 묶어 중첩 columns 생성 // ---------------------------------------------------------------- // 3-1) groupMap: { [groupName]: ColumnDef[] } const groupMap: Record[]> = {} vendorRfqColumnsConfig.forEach((cfg) => { // 만약 group가 없으면 "_noGroup" 처리 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 === "vendorStatus") { const statusVal = row.original.vendorStatus if (!statusVal) return null // const Icon = getStatusIcon(statusVal) return ( {statusVal} ) } if (cfg.id === "rfqVendorStatus") { const statusVal = row.original.rfqVendorStatus if (!statusVal) return null // const Icon = getStatusIcon(statusVal) const variant = statusVal === "INVITED" ? "default" : statusVal === "REJECTED" ? "destructive" : statusVal === "ACCEPTED" ? "secondary" : "outline" return ( {statusVal} ) } if (cfg.id === "rfqVendorUpdated") { const dateVal = cell.getValue() as Date if (!dateVal) return null return formatDate(dateVal) } // code etc... return row.getValue(cfg.id) ?? "" }, } groupMap[groupName].push(childCol) }) const actionsColumn: ColumnDef = { id: "actions", // header: "Actions", cell: ({ row }) => { const rfq = row.original const commCount = rfq.comments?.length ?? 0 const status = row.original.rfqVendorStatus // 공통 코멘트 핸들러 function handleCommentClick() { setRowAction({ row, type: "comments" }) openCommentSheet(Number(row.original.id)) } return ( {/* 기존 기능: status가 INVITED일 때만 표시 */} {(!status || status === 'INVITED') && ( setRowAction({ row, type: "invite" })}> 발행하기 )} {/* 두 기능 사이 구분선 */} {/* 코멘트 메뉴 항목 */} {commCount > 0 ? `${commCount} Comments` : "Add Comment"} ) }, size: 40, enableSorting: false, enableHiding: false, } // const commentsColumn: ColumnDef = { // id: "comments", // header: "Comments", // cell: ({ row }) => { // const rfq = row.original // const commCount = rfq.comments?.length ?? 0 // // 공통 클릭 핸들러 // function handleClick() { // setRowAction({ row, type: "comments" }) // openCommentSheet(Number(row.original.id)) // } // return commCount > 0 ? ( // { // e.preventDefault() // handleClick() // }} // > // {commCount} Comments // // ) : ( // // ) // }, // } // ---------------------------------------------------------------- // 3-2) groupMap에서 실제 상위 컬럼(그룹)을 만들기 // ---------------------------------------------------------------- const nestedColumns: ColumnDef[] = [] // 순서를 고정하고 싶다면 group 순서를 미리 정의하거나 sort해야 함 // 여기서는 그냥 Object.entries 순서 Object.entries(groupMap).forEach(([groupName, colDefs]) => { if (groupName === "_noGroup") { // 그룹 없음 → 그냥 최상위 레벨 컬럼 nestedColumns.push(...colDefs) } else { // 상위 컬럼 nestedColumns.push({ id: groupName, header: groupName, // "Basic Info", "Metadata" 등 columns: colDefs, }) } }) // ---------------------------------------------------------------- // 4) 최종 컬럼 배열: select, nestedColumns, actions // ---------------------------------------------------------------- return [ selectColumn, ...nestedColumns, // commentsColumn, actionsColumn ] }