From 7548e2ad6948f1c6aa102fcac408bc6c9c0f9796 Mon Sep 17 00:00:00 2001 From: dujinkim Date: Wed, 27 Aug 2025 12:06:26 +0000 Subject: (대표님, 최겸) 기본계약, 입찰, 파일라우트, 계약서명라우트, 인포메이션, 메뉴설정, PQ(메일템플릿 관련) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../status/basic-contract-columns.tsx | 363 ++++++++++++++------- lib/basic-contract/status/basic-contract-table.tsx | 44 ++- .../status/basicContract-table-toolbar-actions.tsx | 4 +- 3 files changed, 263 insertions(+), 148 deletions(-) (limited to 'lib/basic-contract/status') diff --git a/lib/basic-contract/status/basic-contract-columns.tsx b/lib/basic-contract/status/basic-contract-columns.tsx index cc9d9bff..8ae8fa1e 100644 --- a/lib/basic-contract/status/basic-contract-columns.tsx +++ b/lib/basic-contract/status/basic-contract-columns.tsx @@ -8,26 +8,34 @@ import { formatDateTime } from "@/lib/utils" import { Badge } from "@/components/ui/badge" import { Checkbox } from "@/components/ui/checkbox" import { DataTableColumnHeaderSimple } from "@/components/data-table/data-table-column-simple-header" -import { - FileActionsDropdown, - FileNameLink -} from "@/components/ui/file-actions" +import { Button } from "@/components/ui/button" +import { MoreHorizontal, Eye, FileText, Calendar } from "lucide-react" +import { + DropdownMenu, + DropdownMenuContent, + DropdownMenuItem, + DropdownMenuTrigger, +} from "@/components/ui/dropdown-menu" +import { useRouter } from "next/navigation" +import { BasicContractTemplateStatsView } from "@/db/schema" + +type NextRouter = ReturnType; -import { basicContractColumnsConfig } from "@/config/basicContractColumnsConfig" -import { BasicContractView } from "@/db/schema" interface GetColumnsProps { - setRowAction: React.Dispatch | null>> + setRowAction: React.Dispatch | null>> + router: NextRouter; + } /** - * 공용 파일 다운로드 유틸리티를 사용하는 간소화된 컬럼 정의 + * BasicContractTemplateStatsView용 컬럼 정의 */ -export function getColumns({ setRowAction }: GetColumnsProps): ColumnDef[] { +export function getColumns({ setRowAction, router }: GetColumnsProps): ColumnDef[] { // ---------------------------------------------------------------- // 1) select 컬럼 (체크박스) // ---------------------------------------------------------------- - const selectColumn: ColumnDef = { + const selectColumn: ColumnDef = { id: "select", header: ({ table }) => ( = { - id: "download", - header: "", + const actionsColumn: ColumnDef = { + id: "actions", + header: "작업", cell: ({ row }) => { - const template = row.original; - - if (!template.filePath || !template.fileName) { - return null; - } + const template = row.original + const detailUrl = `/evcp/basic-contract/${template.templateId}`; return ( - - ); + + + + + + router.push(detailUrl)}> + + 상세 보기 + + setRowAction({ type: "view", row })}> + + 템플릿 보기 + + + + ) }, - maxSize: 30, enableSorting: false, + enableHiding: false, + maxSize: 80, } // ---------------------------------------------------------------- - // 3) 일반 컬럼들을 "그룹"별로 묶어 중첩 columns 생성 + // 3) 기본 정보 컬럼들 // ---------------------------------------------------------------- - const groupMap: Record[]> = {} - - basicContractColumnsConfig.forEach((cfg) => { - const groupName = cfg.group || "_noGroup" - - if (!groupMap[groupName]) { - groupMap[groupName] = [] - } + const basicInfoColumns: ColumnDef[] = [ + { + accessorKey: "templateName", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const name = row.getValue("templateName") as string + return ( +
+ {name} +
+ ) + }, + minSize: 200, + }, + { + accessorKey: "revision", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const revision = row.getValue("revision") as number + return ( + + v{revision} + + ) + }, + minSize: 80, + }, + ] - const childCol: ColumnDef = { - accessorKey: cfg.id, - enableResizing: true, + // ---------------------------------------------------------------- + // 4) 발송/처리 현황 컬럼들 + // ---------------------------------------------------------------- + const processStatusColumns: ColumnDef[] = [ + { + accessorKey: "totalSentCount", header: ({ column }) => ( - + ), - meta: { - excelHeader: cfg.excelHeader, - group: cfg.group, - type: cfg.type, + cell: ({ row }) => { + const count = row.getValue("totalSentCount") as number + return ( +
+ {count.toLocaleString()} +
+ ) }, - cell: ({ row, cell }) => { - // 날짜 형식 처리 - if (cfg.id === "createdAt" || cfg.id === "updatedAt") { - const dateVal = cell.getValue() as Date - return formatDateTime(dateVal, "KR") - } + minSize: 100, + }, + { + accessorKey: "overdueCount", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const count = row.getValue("overdueCount") as number + const total = row.getValue("totalSentCount") as number + const isHigh = total > 0 && (count / total) > 0.2 - // Status 컬럼에 Badge 적용 (확장) - if (cfg.id === "status") { - const status = row.getValue(cfg.id) as string - - let variant: "default" | "secondary" | "destructive" | "outline" = "secondary"; - let label = status; - - switch (status) { - case "ACTIVE": - variant = "default"; - label = "활성"; - break; - case "INACTIVE": - variant = "secondary"; - label = "비활성"; - break; - case "PENDING": - variant = "outline"; - label = "대기중"; - break; - case "COMPLETED": - variant = "default"; - label = "완료"; - break; - default: - variant = "secondary"; - label = status; - } - - return {label} - } - - // ✅ 파일 이름 컬럼 (공용 컴포넌트 사용) - if (cfg.id === "fileName") { - const fileName = cell.getValue() as string; - const filePath = row.original.filePath; - - if (fileName && filePath) { - return ( - - ); - } - return fileName || ""; - } - - // 나머지 컬럼은 그대로 값 표시 - return row.getValue(cfg.id) ?? "" + return ( +
+ {count.toLocaleString()} +
+ ) }, - minSize: 80, - } + minSize: 100, + }, + { + accessorKey: "unsignedCount", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const count = row.getValue("unsignedCount") as number + return ( +
+ {count.toLocaleString()} +
+ ) + }, + minSize: 120, + }, + ] - groupMap[groupName].push(childCol) - }) + // ---------------------------------------------------------------- + // 5) 법무검토 현황 컬럼들 + // ---------------------------------------------------------------- + const legalReviewColumns: ColumnDef[] = [ + { + accessorKey: "legalRequestCount", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const count = row.getValue("legalRequestCount") as number + return ( +
+ {count.toLocaleString()} +
+ ) + }, + minSize: 100, + }, + { + accessorKey: "legalCompletedCount", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const count = row.getValue("legalCompletedCount") as number + return ( +
+ {count.toLocaleString()} +
+ ) + }, + minSize: 100, + }, + { + accessorKey: "contractCompletedCount", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const count = row.getValue("contractCompletedCount") as number + return ( +
+ {count.toLocaleString()} +
+ ) + }, + minSize: 100, + }, + ] // ---------------------------------------------------------------- - // 4) groupMap에서 실제 상위 컬럼(그룹)을 만들기 + // 6) 성과 지표 컬럼들 // ---------------------------------------------------------------- - const nestedColumns: ColumnDef[] = [] + const performanceColumns: ColumnDef[] = [ + { + accessorKey: "avgProcessingDays", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const days = row.getValue("avgProcessingDays") as number | null + if (!days) return "-" + + const rounded = Math.round(days * 10) / 10 + const isGood = rounded <= 5 + const isWarning = rounded > 5 && rounded <= 10 + + let colorClass = "text-gray-900" + if (isGood) colorClass = "text-green-600" + else if (isWarning) colorClass = "text-orange-600" + else colorClass = "text-red-600" + + return ( +
+ {rounded}일 +
+ ) + }, + minSize: 100, + }, + ] - Object.entries(groupMap).forEach(([groupName, colDefs]) => { - if (groupName === "_noGroup") { - nestedColumns.push(...colDefs) - } else { - nestedColumns.push({ - id: groupName, - header: groupName, - columns: colDefs, - }) - } - }) + // ---------------------------------------------------------------- + // 7) 시간 정보 컬럼들 + // ---------------------------------------------------------------- + const timeInfoColumns: ColumnDef[] = [ + { + accessorKey: "templateCreatedAt", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const date = row.getValue("templateCreatedAt") as Date + return formatDateTime(date, "KR") + }, + minSize: 120, + }, + { + accessorKey: "lastActivityDate", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + const date = row.getValue("lastActivityDate") as Date | null + return date ? formatDateTime(date, "KR") : "-" + }, + minSize: 120, + }, + ] // ---------------------------------------------------------------- - // 5) 최종 컬럼 배열 + // 8) 최종 컬럼 배열 (평면 구조) // ---------------------------------------------------------------- return [ selectColumn, - downloadColumn, // ✅ 공용 파일 액션 컴포넌트 사용 - ...nestedColumns, + ...basicInfoColumns, + ...processStatusColumns, + ...legalReviewColumns, + ...performanceColumns, + ...timeInfoColumns, + actionsColumn, ] -} +} \ No newline at end of file diff --git a/lib/basic-contract/status/basic-contract-table.tsx b/lib/basic-contract/status/basic-contract-table.tsx index 22845144..07707ff0 100644 --- a/lib/basic-contract/status/basic-contract-table.tsx +++ b/lib/basic-contract/status/basic-contract-table.tsx @@ -14,9 +14,9 @@ import type { import { toast } from "sonner"; import { getColumns } from "./basic-contract-columns"; import { getBasicContracts } from "../service"; -import { BasicContractView } from "@/db/schema"; +import { BasicContractTemplateStatsView } from "@/db/schema"; import { BasicContractTableToolbarActions } from "./basicContract-table-toolbar-actions"; - +import { useRouter } from "next/navigation" interface BasicTemplateTableProps { promises: Promise< @@ -26,38 +26,36 @@ interface BasicTemplateTableProps { > } - export function BasicContractsTable({ promises }: BasicTemplateTableProps) { - const [rowAction, setRowAction] = - React.useState | null>(null) - + React.useState | null>(null) + const router = useRouter() const [{ data, pageCount }] = React.use(promises) // 컬럼 설정 - 외부 파일에서 가져옴 const columns = React.useMemo( - () => getColumns({ setRowAction }), - [setRowAction] + () => getColumns({ setRowAction,router }), + [setRowAction,router] ) // config 기반으로 필터 필드 설정 - const advancedFilterFields: DataTableAdvancedFilterField[] = [ + const advancedFilterFields: DataTableAdvancedFilterField[] = [ { id: "templateName", label: "템플릿명", type: "text" }, - { - id: "status", label: "상태", type: "select", options: [ - { label: "활성", value: "ACTIVE" }, - { label: "비활성", value: "INACTIVE" }, - ] - }, - { id: "userName", label: "요청자", type: "text" }, - { id: "vendorName", label: "업체명", type: "text" }, - { id: "vendorCode", label: "업체코드", type: "text" }, - { id: "vendorEmail", label: "업체대표이메일", type: "text" }, - { id: "createdAt", label: "생성일", type: "date" }, - { id: "updatedAt", label: "수정일", type: "date" }, + { id: "revision", label: "리비전", type: "number" }, + { id: "validityPeriod", label: "유효기간(개월)", type: "number" }, + { id: "totalSentCount", label: "발송건수", type: "number" }, + { id: "overdueCount", label: "지연건수", type: "number" }, + { id: "unsignedCount", label: "미서명건수", type: "number" }, + { id: "legalRequestCount", label: "법무요청건수", type: "number" }, + { id: "legalCompletedCount", label: "법무완료건수", type: "number" }, + { id: "contractCompletedCount", label: "계약완료건수", type: "number" }, + { id: "rejectedCount", label: "거절건수", type: "number" }, + { id: "avgProcessingDays", label: "평균처리일", type: "number" }, + { id: "templateCreatedAt", label: "템플릿 생성일", type: "date" }, + { id: "lastActivityDate", label: "최근활동일", type: "date" }, ]; const { table } = useDataTable({ @@ -68,10 +66,10 @@ export function BasicContractsTable({ promises }: BasicTemplateTableProps) { enablePinning: true, enableAdvancedFilter: true, initialState: { - sorting: [{ id: "createdAt", desc: true }], + sorting: [{ id: "lastActivityDate", desc: true }], columnPinning: { right: ["actions"] }, }, - getRowId: (originalRow) => String(originalRow.id), + getRowId: (originalRow) => String(originalRow.templateId), shallow: false, clearOnDefault: true, }) diff --git a/lib/basic-contract/status/basicContract-table-toolbar-actions.tsx b/lib/basic-contract/status/basicContract-table-toolbar-actions.tsx index cee94790..a74a6e4d 100644 --- a/lib/basic-contract/status/basicContract-table-toolbar-actions.tsx +++ b/lib/basic-contract/status/basicContract-table-toolbar-actions.tsx @@ -7,10 +7,10 @@ import { Download, Upload } from "lucide-react" import { exportTableToExcel } from "@/lib/export" import { Button } from "@/components/ui/button" -import { BasicContractView } from "@/db/schema" +import { BasicContractTemplateStatsView } from "@/db/schema" interface TemplateTableToolbarActionsProps { - table: Table + table: Table } export function BasicContractTableToolbarActions({ table }: TemplateTableToolbarActionsProps) { -- cgit v1.2.3