'use client' import { useState, useEffect, useCallback, useMemo } 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 { Badge } from '@/components/ui/badge' 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 { searchUnifiedProjects, UnifiedProject, ProjectSearchOptions } from './project-service' import { toast } from 'sonner' export interface UnifiedProjectSelectorProps { selectedProject?: UnifiedProject onProjectSelect: (project: UnifiedProject) => void disabled?: boolean searchOptions?: Partial placeholder?: string className?: string } export function UnifiedProjectSelector({ selectedProject, onProjectSelect, disabled, searchOptions = {}, placeholder = "프로젝트를 선택하세요", className }: UnifiedProjectSelectorProps) { const [open, setOpen] = useState(false) const [projects, setProjects] = useState([]) const [loading, setLoading] = useState(false) const [sorting, setSorting] = useState([]) const [columnFilters, setColumnFilters] = useState([]) const [columnVisibility, setColumnVisibility] = useState({}) const [rowSelection, setRowSelection] = useState({}) const [globalFilter, setGlobalFilter] = useState('') // 기본 검색 옵션 설정 const finalSearchOptions: ProjectSearchOptions = useMemo(() => ({ searchFrom: 'both', limit: 100, ...searchOptions }), [searchOptions]) // 테이블 컬럼 정의 const columns: ColumnDef[] = useMemo(() => [ { accessorKey: 'code', header: '프로젝트 코드', cell: ({ row }) => (
{row.getValue('code')}
), }, { accessorKey: 'name', header: '프로젝트명', cell: ({ row }) => (
{row.getValue('name')}
), }, { accessorKey: 'type', header: '타입', cell: ({ row }) => ( {row.getValue('type')} ), }, { accessorKey: 'source', header: '출처', cell: ({ row }) => { const source = row.getValue('source') as string return ( {source === 'projects' ? '프로젝트' : '견적프로젝트'} ) }, }, { id: 'actions', header: '선택', cell: ({ row }) => ( ), }, ], []) // 프로젝트 테이블 설정 const table = useReactTable({ data: projects, 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, }, }) // 프로젝트 목록 로드 const loadProjects = useCallback(async (searchTerm?: string) => { setLoading(true) try { const result = await searchUnifiedProjects({ ...finalSearchOptions, searchTerm: searchTerm || globalFilter }) if (result.success) { setProjects(result.data) } else { toast.error(result.error || '프로젝트를 불러오는데 실패했습니다.') setProjects([]) } } catch (error) { console.error('프로젝트 목록 로드 실패:', error) toast.error('프로젝트를 불러오는 중 오류가 발생했습니다.') setProjects([]) } finally { setLoading(false) } }, [finalSearchOptions, globalFilter]) // 프로젝트 선택 핸들러 const handleProjectSelect = useCallback((project: UnifiedProject) => { onProjectSelect(project) setOpen(false) }, [onProjectSelect]) // 다이얼로그 열릴 때 프로젝트 목록 로드 useEffect(() => { if (open && projects.length === 0) { loadProjects() } }, [open, projects.length, loadProjects]) // 검색어 변경 시 디바운스된 검색 useEffect(() => { if (!open) return const timeoutId = setTimeout(() => { loadProjects(globalFilter) }, 300) return () => clearTimeout(timeoutId) }, [globalFilter, open, loadProjects]) return ( 프로젝트 선택
{finalSearchOptions.searchFrom === 'both' && '프로젝트 및 견적프로젝트에서 검색'} {finalSearchOptions.searchFrom === 'projects' && '프로젝트에서 검색'} {finalSearchOptions.searchFrom === 'biddingProjects' && '견적프로젝트에서 검색'}
setGlobalFilter(e.target.value)} className="flex-1" />
{loading ? (
프로젝트를 불러오는 중...
) : (
{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) => ( handleProjectSelect(row.original)} > {row.getVisibleCells().map((cell) => ( {flexRender( cell.column.columnDef.cell, cell.getContext() )} ))} )) ) : ( 검색 결과가 없습니다. )}
)}
총 {table.getFilteredRowModel().rows.length}개 프로젝트
{table.getState().pagination.pageIndex + 1} / {table.getPageCount()}
) }