'use client' /** * WBS 코드 선택기 * * @description * - 오라클에서 WBS 코드들을 조회 * - PROJ_NO: 프로젝트 번호 * - WBS_ELMT: WBS 요소 * - WBS_ELMT_NM: WBS 요소명 * - WBS_LVL: WBS 레벨 */ import { useState, useCallback, useMemo, useTransition } from 'react' import { Button } from '@/components/ui/button' import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogTrigger } from '@/components/ui/dialog' import { Input } from '@/components/ui/input' import { Search, Check } from 'lucide-react' import { ColumnDef, flexRender, getCoreRowModel, getFilteredRowModel, getPaginationRowModel, getSortedRowModel, useReactTable, SortingState, ColumnFiltersState, VisibilityState, RowSelectionState, } from '@tanstack/react-table' import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow, } from '@/components/ui/table' import { getWbsCodes, WbsCode } from './wbs-code-service' import { toast } from 'sonner' export interface WbsCodeSelectorProps { selectedCode?: WbsCode onCodeSelect: (code: WbsCode) => void disabled?: boolean placeholder?: string className?: string projNo?: string // 프로젝트 번호 필터 } export interface WbsCodeItem { code: string // WBS 코드 (PROJ_NO + WBS_ELMT 조합) projNo: string // 프로젝트 번호 wbsElmt: string // WBS 요소 wbsElmtNm: string // WBS 요소명 wbsLvl: string // WBS 레벨 displayText: string // 표시용 텍스트 } export function WbsCodeSelector({ selectedCode, onCodeSelect, disabled, placeholder = "WBS 코드를 선택하세요", className, projNo }: WbsCodeSelectorProps) { const [open, setOpen] = useState(false) const [codes, setCodes] = useState([]) const [sorting, setSorting] = useState([]) const [columnFilters, setColumnFilters] = useState([]) const [columnVisibility, setColumnVisibility] = useState({}) const [rowSelection, setRowSelection] = useState({}) const [globalFilter, setGlobalFilter] = useState('') const [isPending, startTransition] = useTransition() // WBS 코드 선택 핸들러 const handleCodeSelect = useCallback(async (code: WbsCode) => { onCodeSelect(code) setOpen(false) }, [onCodeSelect]) // 테이블 컬럼 정의 const columns: ColumnDef[] = useMemo(() => [ { accessorKey: 'PROJ_NO', header: '프로젝트 번호', cell: ({ row }) => (
{row.getValue('PROJ_NO')}
), }, { accessorKey: 'WBS_ELMT', header: 'WBS 요소', cell: ({ row }) => (
{row.getValue('WBS_ELMT')}
), }, { accessorKey: 'WBS_ELMT_NM', header: 'WBS 요소명', cell: ({ row }) => (
{row.getValue('WBS_ELMT_NM')}
), }, { accessorKey: 'WBS_LVL', header: '레벨', cell: ({ row }) => (
{row.getValue('WBS_LVL')}
), }, { id: 'actions', header: '선택', cell: ({ row }) => ( ), }, ], [handleCodeSelect]) // WBS 코드 테이블 설정 const table = useReactTable({ data: codes, columns, onSortingChange: setSorting, onColumnFiltersChange: setColumnFilters, onColumnVisibilityChange: setColumnVisibility, onRowSelectionChange: setRowSelection, onGlobalFilterChange: setGlobalFilter, getCoreRowModel: getCoreRowModel(), getPaginationRowModel: getPaginationRowModel(), getSortedRowModel: getSortedRowModel(), getFilteredRowModel: getFilteredRowModel(), state: { sorting, columnFilters, columnVisibility, rowSelection, globalFilter, }, }) // 서버에서 WBS 코드 전체 목록 로드 (한 번만) const loadCodes = useCallback(async () => { startTransition(async () => { try { const result = await getWbsCodes(projNo) if (result.success) { setCodes(result.data) // 폴백 데이터를 사용하는 경우 알림 if (result.isUsingFallback) { toast.info('Oracle 연결 실패', { description: '테스트 데이터를 사용합니다.', duration: 4000, }) } } else { toast.error(result.error || 'WBS 코드를 불러오는데 실패했습니다.') setCodes([]) } } catch (error) { console.error('WBS 코드 목록 로드 실패:', error) toast.error('WBS 코드를 불러오는 중 오류가 발생했습니다.') setCodes([]) } }) }, [projNo]) // 다이얼로그 열기 핸들러 const handleDialogOpenChange = useCallback((newOpen: boolean) => { setOpen(newOpen) if (newOpen && codes.length === 0) { loadCodes() } }, [loadCodes, codes.length]) // 검색어 변경 핸들러 (클라이언트 사이드 필터링) const handleSearchChange = useCallback((value: string) => { setGlobalFilter(value) }, []) return ( WBS 코드 선택
WBS 코드 조회
handleSearchChange(e.target.value)} className="flex-1" />
{isPending ? (
WBS 코드를 불러오는 중...
) : (
{table.getHeaderGroups().map((headerGroup) => ( {headerGroup.headers.map((header) => ( {header.isPlaceholder ? null : flexRender( header.column.columnDef.header, header.getContext() )} ))} ))} {table.getRowModel().rows?.length ? ( table.getRowModel().rows.map((row) => ( handleCodeSelect(row.original)} > {row.getVisibleCells().map((cell) => ( {flexRender( cell.column.columnDef.cell, cell.getContext() )} ))} )) ) : ( 검색 결과가 없습니다. )}
)}
총 {table.getFilteredRowModel().rows.length}개 WBS 코드
{table.getState().pagination.pageIndex + 1} / {table.getPageCount()}
) }