'use client' /** * Cost Center 단일 선택 다이얼로그 * * @description * - Cost Center를 하나만 선택할 수 있는 다이얼로그 * - 트리거 버튼과 다이얼로그가 분리된 구조 * - 외부에서 open 상태를 제어 가능 */ import { useState, useCallback, useMemo, useTransition, useEffect } from 'react' import { Button } from '@/components/ui/button' import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter } from '@/components/ui/dialog' import { Input } from '@/components/ui/input' import { Search, Check, X } 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 { getCostCenters, CostCenter } from './cost-center-service' import { toast } from 'sonner' export interface CostCenterSingleSelectorProps { open: boolean onOpenChange: (open: boolean) => void selectedCode?: CostCenter onCodeSelect: (code: CostCenter) => void onConfirm?: (code: CostCenter | undefined) => void onCancel?: () => void title?: string description?: string showConfirmButtons?: boolean } export function CostCenterSingleSelector({ open, onOpenChange, selectedCode, onCodeSelect, onConfirm, onCancel, title = "코스트센터 선택", description = "코스트센터를 선택하세요", showConfirmButtons = false }: CostCenterSingleSelectorProps) { 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() const [tempSelectedCode, setTempSelectedCode] = useState(selectedCode) // 날짜 포맷 함수 (YYYYMMDD -> YYYY-MM-DD) const formatDate = (dateStr: string) => { if (!dateStr || dateStr.length !== 8) return dateStr return `${dateStr.substring(0, 4)}-${dateStr.substring(4, 6)}-${dateStr.substring(6, 8)}` } // Cost Center 선택 핸들러 const handleCodeSelect = useCallback((code: CostCenter) => { if (showConfirmButtons) { setTempSelectedCode(code) } else { onCodeSelect(code) onOpenChange(false) } }, [onCodeSelect, onOpenChange, showConfirmButtons]) // 확인 버튼 핸들러 const handleConfirm = useCallback(() => { if (tempSelectedCode) { onCodeSelect(tempSelectedCode) } onConfirm?.(tempSelectedCode) onOpenChange(false) }, [tempSelectedCode, onCodeSelect, onConfirm, onOpenChange]) // 취소 버튼 핸들러 const handleCancel = useCallback(() => { setTempSelectedCode(selectedCode) onCancel?.() onOpenChange(false) }, [selectedCode, onCancel, onOpenChange]) // 테이블 컬럼 정의 const columns: ColumnDef[] = useMemo(() => [ { accessorKey: 'KOSTL', header: '코스트센터', cell: ({ row }) => (
{row.getValue('KOSTL')}
), }, { accessorKey: 'KTEXT', header: '단축명', cell: ({ row }) => (
{row.getValue('KTEXT')}
), }, { accessorKey: 'LTEXT', header: '설명', cell: ({ row }) => (
{row.getValue('LTEXT')}
), }, { accessorKey: 'DATAB', header: '시작일', cell: ({ row }) => (
{formatDate(row.getValue('DATAB'))}
), }, { accessorKey: 'DATBI', header: '종료일', cell: ({ row }) => (
{formatDate(row.getValue('DATBI'))}
), }, { id: 'actions', header: '선택', cell: ({ row }) => { const isSelected = showConfirmButtons ? tempSelectedCode?.KOSTL === row.original.KOSTL : selectedCode?.KOSTL === row.original.KOSTL return ( ) }, }, ], [handleCodeSelect, selectedCode, tempSelectedCode, showConfirmButtons]) // Cost Center 테이블 설정 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, }, }) // 서버에서 Cost Center 전체 목록 로드 (한 번만) const loadCodes = useCallback(async () => { startTransition(async () => { try { const result = await getCostCenters() if (result.success) { setCodes(result.data) // 폴백 데이터를 사용하는 경우 알림 if (result.isUsingFallback) { toast.info('Oracle 연결 실패', { description: '테스트 데이터를 사용합니다.', duration: 4000, }) } } else { toast.error(result.error || '코스트센터를 불러오는데 실패했습니다.') setCodes([]) } } catch (error) { console.error('코스트센터 목록 로드 실패:', error) toast.error('코스트센터를 불러오는 중 오류가 발생했습니다.') setCodes([]) } }) }, []) // 다이얼로그 열릴 때 코드 로드 (open prop 변화 감지) useEffect(() => { if (open) { setTempSelectedCode(selectedCode) if (codes.length === 0) { console.log('🚀 [CostCenterSingleSelector] 다이얼로그 열림 - loadCodes 호출') loadCodes() } else { console.log('📦 [CostCenterSingleSelector] 다이얼로그 열림 - 기존 데이터 사용 (' + codes.length + '건)') } } }, [open, selectedCode, loadCodes, codes.length]) // 검색어 변경 핸들러 (클라이언트 사이드 필터링) const handleSearchChange = useCallback((value: string) => { setGlobalFilter(value) }, []) const currentSelectedCode = showConfirmButtons ? tempSelectedCode : selectedCode return ( {title}
{description}
{/* 현재 선택된 코스트센터 표시 */} {currentSelectedCode && (
선택된 코스트센터:
[{currentSelectedCode.KOSTL}] {currentSelectedCode.KTEXT} - {currentSelectedCode.LTEXT}
)}
handleSearchChange(e.target.value)} className="flex-1" />
{isPending ? (
코스트센터를 불러오는 중...
) : (
{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) => { const isRowSelected = currentSelectedCode?.KOSTL === row.original.KOSTL return ( handleCodeSelect(row.original)} > {row.getVisibleCells().map((cell) => ( {flexRender( cell.column.columnDef.cell, cell.getContext() )} ))} ) }) ) : ( 검색 결과가 없습니다. )}
)}
총 {table.getFilteredRowModel().rows.length}개 코스트센터
{table.getState().pagination.pageIndex + 1} / {table.getPageCount()}
{showConfirmButtons && ( )}
) }