summaryrefslogtreecommitdiff
path: root/components/common
diff options
context:
space:
mode:
Diffstat (limited to 'components/common')
-rw-r--r--components/common/selectors/procurement-manager/index.ts21
-rw-r--r--components/common/selectors/procurement-manager/procurement-manager-multi-selector.tsx466
-rw-r--r--components/common/selectors/procurement-manager/procurement-manager-selector.tsx310
-rw-r--r--components/common/selectors/procurement-manager/procurement-manager-service.ts199
-rw-r--r--components/common/selectors/procurement-manager/procurement-manager-single-selector.tsx374
5 files changed, 1370 insertions, 0 deletions
diff --git a/components/common/selectors/procurement-manager/index.ts b/components/common/selectors/procurement-manager/index.ts
new file mode 100644
index 00000000..8b53adfa
--- /dev/null
+++ b/components/common/selectors/procurement-manager/index.ts
@@ -0,0 +1,21 @@
+// 조달담당자 선택기 관련 컴포넌트와 타입 내보내기
+
+export { ProcurementManagerSelector } from './procurement-manager-selector'
+export type { ProcurementManagerSelectorProps } from './procurement-manager-selector'
+
+export { ProcurementManagerSingleSelector } from './procurement-manager-single-selector'
+export type { ProcurementManagerSingleSelectorProps } from './procurement-manager-single-selector'
+
+export { ProcurementManagerMultiSelector } from './procurement-manager-multi-selector'
+export type { ProcurementManagerMultiSelectorProps } from './procurement-manager-multi-selector'
+
+export {
+ getProcurementManagers,
+ addUsersToManagersAsync
+} from './procurement-manager-service'
+export type {
+ ProcurementManager,
+ ProcurementManagerWithUser
+} from './procurement-manager-service'
+
+
diff --git a/components/common/selectors/procurement-manager/procurement-manager-multi-selector.tsx b/components/common/selectors/procurement-manager/procurement-manager-multi-selector.tsx
new file mode 100644
index 00000000..8d2f2afa
--- /dev/null
+++ b/components/common/selectors/procurement-manager/procurement-manager-multi-selector.tsx
@@ -0,0 +1,466 @@
+'use client'
+
+/**
+ * 조달담당자 다중 선택 다이얼로그
+ *
+ * @description
+ * - 여러 조달담당자를 선택할 수 있는 다이얼로그
+ * - 체크박스를 통한 다중 선택
+ * - 선택된 조달담당자들을 상단에 표시
+ * - 확인/취소 버튼으로 선택 확정
+ * - 선택 시 각 담당자에 대한 사용자 정보도 함께 반환
+ */
+
+import { useState, useCallback, useMemo, useTransition } 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 { Checkbox } from '@/components/ui/checkbox'
+import { Badge } from '@/components/ui/badge'
+import { Search, Check, X, Trash2 } 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 {
+ getProcurementManagers,
+ addUsersToManagersAsync,
+ ProcurementManager,
+ ProcurementManagerWithUser
+} from './procurement-manager-service'
+import { toast } from 'sonner'
+
+// 간단한 디바운스 함수
+function debounce<T extends (...args: unknown[]) => void>(func: T, delay: number): T {
+ let timeoutId: NodeJS.Timeout
+ return ((...args: Parameters<T>) => {
+ clearTimeout(timeoutId)
+ timeoutId = setTimeout(() => func(...args), delay)
+ }) as T
+}
+
+export interface ProcurementManagerMultiSelectorProps {
+ open: boolean
+ onOpenChange: (open: boolean) => void
+ selectedManagers?: ProcurementManagerWithUser[]
+ onManagersSelect: (managers: ProcurementManagerWithUser[]) => void
+ onConfirm?: (managers: ProcurementManagerWithUser[]) => void
+ onCancel?: () => void
+ title?: string
+ description?: string
+ maxSelection?: number
+}
+
+export function ProcurementManagerMultiSelector({
+ open,
+ onOpenChange,
+ selectedManagers = [],
+ onManagersSelect,
+ onConfirm,
+ onCancel,
+ title = "조달담당자 다중 선택",
+ description = "여러 조달담당자를 선택하세요",
+ maxSelection
+}: ProcurementManagerMultiSelectorProps) {
+ const [managers, setManagers] = useState<ProcurementManager[]>([])
+ const [sorting, setSorting] = useState<SortingState>([])
+ const [columnFilters, setColumnFilters] = useState<ColumnFiltersState>([])
+ const [columnVisibility, setColumnVisibility] = useState<VisibilityState>({})
+ const [rowSelection, setRowSelection] = useState<RowSelectionState>({})
+ const [globalFilter, setGlobalFilter] = useState('')
+ const [isPending, startTransition] = useTransition()
+ const [tempSelectedManagers, setTempSelectedManagers] = useState<ProcurementManager[]>(
+ selectedManagers.map(m => ({
+ PROCUREMENT_MANAGER_CODE: m.PROCUREMENT_MANAGER_CODE,
+ DISPLAY_NAME: m.DISPLAY_NAME,
+ DEPARTMENT_NAME: m.DEPARTMENT_NAME,
+ EMPLOYEE_NUMBER: m.EMPLOYEE_NUMBER,
+ DEPARTMENT_CODE: m.DEPARTMENT_CODE,
+ IS_ACTIVE: m.IS_ACTIVE
+ }))
+ )
+
+ // 조달담당자 선택/해제 핸들러
+ const handleManagerToggle = useCallback((manager: ProcurementManager, checked: boolean) => {
+ setTempSelectedManagers(prev => {
+ if (checked) {
+ // 최대 선택 수 제한 확인
+ if (maxSelection && prev.length >= maxSelection) {
+ toast.warning(`최대 ${maxSelection}개까지 선택할 수 있습니다.`)
+ return prev
+ }
+ // 이미 선택된 담당자인지 확인
+ if (prev.some(m => m.PROCUREMENT_MANAGER_CODE === manager.PROCUREMENT_MANAGER_CODE)) {
+ return prev
+ }
+ return [...prev, manager]
+ } else {
+ return prev.filter(m => m.PROCUREMENT_MANAGER_CODE !== manager.PROCUREMENT_MANAGER_CODE)
+ }
+ })
+ }, [maxSelection])
+
+ // 개별 담당자 제거 핸들러
+ const handleRemoveManager = useCallback((procurementManagerCode: string) => {
+ setTempSelectedManagers(prev => prev.filter(m => m.PROCUREMENT_MANAGER_CODE !== procurementManagerCode))
+ }, [])
+
+ // 모든 선택 해제 핸들러
+ const handleClearAll = useCallback(() => {
+ setTempSelectedManagers([])
+ }, [])
+
+ // 확인 버튼 핸들러 (사용자 정보 포함하여 반환)
+ const handleConfirm = useCallback(async () => {
+ try {
+ // 선택된 담당자들에 대해 사용자 정보 추가
+ const managersWithUsers = await addUsersToManagersAsync(tempSelectedManagers)
+
+ onManagersSelect(managersWithUsers)
+ onConfirm?.(managersWithUsers)
+ onOpenChange(false)
+ } catch (error) {
+ console.error('조달담당자 정보 조회 오류:', error)
+ toast.error('조달담당자 정보를 가져오는 중 오류가 발생했습니다.')
+ }
+ }, [tempSelectedManagers, onManagersSelect, onConfirm, onOpenChange])
+
+ // 취소 버튼 핸들러
+ const handleCancel = useCallback(() => {
+ setTempSelectedManagers(
+ selectedManagers.map(m => ({
+ PROCUREMENT_MANAGER_CODE: m.PROCUREMENT_MANAGER_CODE,
+ DISPLAY_NAME: m.DISPLAY_NAME,
+ DEPARTMENT_NAME: m.DEPARTMENT_NAME,
+ EMPLOYEE_NUMBER: m.EMPLOYEE_NUMBER,
+ DEPARTMENT_CODE: m.DEPARTMENT_CODE,
+ IS_ACTIVE: m.IS_ACTIVE
+ }))
+ )
+ onCancel?.()
+ onOpenChange(false)
+ }, [selectedManagers, onCancel, onOpenChange])
+
+ // 테이블 컬럼 정의
+ const columns: ColumnDef<ProcurementManager>[] = useMemo(() => [
+ {
+ id: 'select',
+ header: ({ table }) => (
+ <Checkbox
+ checked={table.getIsAllPageRowsSelected()}
+ onCheckedChange={(value) => {
+ if (value) {
+ // 페이지의 모든 행을 선택하되, 최대 선택 수 제한 확인
+ const currentPageRows = table.getRowModel().rows
+ const newSelections = currentPageRows
+ .map(row => row.original)
+ .filter(manager => !tempSelectedManagers.some(m => m.PROCUREMENT_MANAGER_CODE === manager.PROCUREMENT_MANAGER_CODE))
+
+ if (maxSelection) {
+ const remainingSlots = maxSelection - tempSelectedManagers.length
+ if (newSelections.length > remainingSlots) {
+ toast.warning(`최대 ${maxSelection}개까지 선택할 수 있습니다.`)
+ return
+ }
+ }
+
+ setTempSelectedManagers(prev => [...prev, ...newSelections])
+ } else {
+ // 페이지의 모든 행 선택 해제
+ const currentPageCodes = table.getRowModel().rows.map(row => row.original.PROCUREMENT_MANAGER_CODE)
+ setTempSelectedManagers(prev => prev.filter(m => !currentPageCodes.includes(m.PROCUREMENT_MANAGER_CODE)))
+ }
+ }}
+ aria-label="모든 행 선택"
+ />
+ ),
+ cell: ({ row }) => {
+ const isSelected = tempSelectedManagers.some(m => m.PROCUREMENT_MANAGER_CODE === row.original.PROCUREMENT_MANAGER_CODE)
+ return (
+ <Checkbox
+ checked={isSelected}
+ onCheckedChange={(value) => handleManagerToggle(row.original, !!value)}
+ aria-label="행 선택"
+ />
+ )
+ },
+ enableSorting: false,
+ enableHiding: false,
+ },
+ {
+ accessorKey: 'PROCUREMENT_MANAGER_CODE',
+ header: '조달담당자코드',
+ cell: ({ row }) => (
+ <div className="font-mono text-sm">{row.getValue('PROCUREMENT_MANAGER_CODE')}</div>
+ ),
+ },
+ {
+ accessorKey: 'DISPLAY_NAME',
+ header: '담당자명',
+ cell: ({ row }) => (
+ <div>{row.getValue('DISPLAY_NAME')}</div>
+ ),
+ },
+ {
+ accessorKey: 'DEPARTMENT_NAME',
+ header: '부서명',
+ cell: ({ row }) => (
+ <div className="text-sm text-muted-foreground">{row.getValue('DEPARTMENT_NAME')}</div>
+ ),
+ },
+ ], [handleManagerToggle, tempSelectedManagers, maxSelection])
+
+ // 조달담당자 테이블 설정
+ const table = useReactTable({
+ data: managers,
+ 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 loadManagers = useCallback(async () => {
+ startTransition(async () => {
+ try {
+ const result = await getProcurementManagers()
+
+ if (result.success) {
+ setManagers(result.data)
+ } else {
+ toast.error(result.error || '조달담당자를 불러오는데 실패했습니다.')
+ setManagers([])
+ }
+ } catch (error) {
+ console.error('조달담당자 목록 로드 실패:', error)
+ toast.error('조달담당자를 불러오는 중 오류가 발생했습니다.')
+ setManagers([])
+ }
+ })
+ }, [])
+
+ // 다이얼로그 열기/닫기 핸들러
+ const handleDialogOpenChange = useCallback((newOpen: boolean) => {
+ onOpenChange(newOpen)
+ if (newOpen) {
+ setTempSelectedManagers(
+ selectedManagers.map(m => ({
+ PROCUREMENT_MANAGER_CODE: m.PROCUREMENT_MANAGER_CODE,
+ DISPLAY_NAME: m.DISPLAY_NAME,
+ DEPARTMENT_NAME: m.DEPARTMENT_NAME,
+ EMPLOYEE_NUMBER: m.EMPLOYEE_NUMBER,
+ DEPARTMENT_CODE: m.DEPARTMENT_CODE,
+ IS_ACTIVE: m.IS_ACTIVE
+ }))
+ )
+ if (managers.length === 0) {
+ loadManagers()
+ }
+ }
+ }, [onOpenChange, selectedManagers, loadManagers, managers.length])
+
+ // 검색어 변경 핸들러 (클라이언트 사이드 필터링)
+ const handleSearchChange = useCallback((value: string) => {
+ setGlobalFilter(value)
+ }, [])
+
+ return (
+ <Dialog open={open} onOpenChange={handleDialogOpenChange}>
+ <DialogContent className="max-w-6xl max-h-[85vh]">
+ <DialogHeader>
+ <DialogTitle>{title}</DialogTitle>
+ <div className="text-sm text-muted-foreground">
+ {description}
+ {maxSelection && ` (최대 ${maxSelection}개)`}
+ </div>
+ </DialogHeader>
+
+ <div className="space-y-4">
+ {/* 선택된 조달담당자들 표시 */}
+ {tempSelectedManagers.length > 0 && (
+ <div className="space-y-2">
+ <div className="flex items-center justify-between">
+ <div className="text-sm font-medium">
+ 선택된 조달담당자 ({tempSelectedManagers.length}개)
+ {maxSelection && ` / ${maxSelection}`}
+ </div>
+ <Button
+ variant="outline"
+ size="sm"
+ onClick={handleClearAll}
+ disabled={tempSelectedManagers.length === 0}
+ >
+ <Trash2 className="h-4 w-4 mr-2" />
+ 모두 제거
+ </Button>
+ </div>
+ <div className="flex flex-wrap gap-2 max-h-24 overflow-y-auto p-2 border rounded-md bg-muted/30">
+ {tempSelectedManagers.map((manager) => (
+ <Badge
+ key={manager.PROCUREMENT_MANAGER_CODE}
+ variant="secondary"
+ className="text-xs"
+ >
+ [{manager.PROCUREMENT_MANAGER_CODE}] {manager.DISPLAY_NAME}
+ {manager.DEPARTMENT_NAME && ` - ${manager.DEPARTMENT_NAME}`}
+ <button
+ onClick={() => handleRemoveManager(manager.PROCUREMENT_MANAGER_CODE)}
+ className="ml-1 hover:text-destructive"
+ >
+ <X className="h-3 w-3" />
+ </button>
+ </Badge>
+ ))}
+ </div>
+ </div>
+ )}
+
+ <div className="flex items-center space-x-2">
+ <Search className="h-4 w-4" />
+ <Input
+ placeholder="조달담당자코드, 담당자명, 부서명으로 검색..."
+ value={globalFilter}
+ onChange={(e) => handleSearchChange(e.target.value)}
+ className="flex-1"
+ />
+ </div>
+
+ {isPending ? (
+ <div className="flex justify-center py-8">
+ <div className="text-sm text-muted-foreground">조달담당자를 불러오는 중...</div>
+ </div>
+ ) : (
+ <div className="border rounded-md">
+ <Table>
+ <TableHeader>
+ {table.getHeaderGroups().map((headerGroup) => (
+ <TableRow key={headerGroup.id}>
+ {headerGroup.headers.map((header) => (
+ <TableHead key={header.id}>
+ {header.isPlaceholder
+ ? null
+ : flexRender(
+ header.column.columnDef.header,
+ header.getContext()
+ )}
+ </TableHead>
+ ))}
+ </TableRow>
+ ))}
+ </TableHeader>
+ <TableBody>
+ {table.getRowModel().rows?.length ? (
+ table.getRowModel().rows.map((row) => {
+ const isSelected = tempSelectedManagers.some(m => m.PROCUREMENT_MANAGER_CODE === row.original.PROCUREMENT_MANAGER_CODE)
+ return (
+ <TableRow
+ key={row.id}
+ data-state={isSelected && "selected"}
+ className={`cursor-pointer hover:bg-muted/50 ${
+ isSelected ? 'bg-muted/30' : ''
+ }`}
+ onClick={() => {
+ const isCurrentlySelected = tempSelectedManagers.some(m => m.PROCUREMENT_MANAGER_CODE === row.original.PROCUREMENT_MANAGER_CODE)
+ handleManagerToggle(row.original, !isCurrentlySelected)
+ }}
+ >
+ {row.getVisibleCells().map((cell) => (
+ <TableCell key={cell.id}>
+ {flexRender(
+ cell.column.columnDef.cell,
+ cell.getContext()
+ )}
+ </TableCell>
+ ))}
+ </TableRow>
+ )
+ })
+ ) : (
+ <TableRow>
+ <TableCell
+ colSpan={columns.length}
+ className="h-24 text-center"
+ >
+ 검색 결과가 없습니다.
+ </TableCell>
+ </TableRow>
+ )}
+ </TableBody>
+ </Table>
+ </div>
+ )}
+
+ <div className="flex items-center justify-between">
+ <div className="text-sm text-muted-foreground">
+ 총 {table.getFilteredRowModel().rows.length}개 조달담당자
+ </div>
+ <div className="flex items-center space-x-2">
+ <Button
+ variant="outline"
+ size="sm"
+ onClick={() => table.previousPage()}
+ disabled={!table.getCanPreviousPage()}
+ >
+ 이전
+ </Button>
+ <div className="text-sm">
+ {table.getState().pagination.pageIndex + 1} / {table.getPageCount()}
+ </div>
+ <Button
+ variant="outline"
+ size="sm"
+ onClick={() => table.nextPage()}
+ disabled={!table.getCanNextPage()}
+ >
+ 다음
+ </Button>
+ </div>
+ </div>
+ </div>
+
+ <DialogFooter>
+ <Button variant="outline" onClick={handleCancel}>
+ <X className="h-4 w-4 mr-2" />
+ 취소
+ </Button>
+ <Button onClick={handleConfirm}>
+ <Check className="h-4 w-4 mr-2" />
+ 확인 ({tempSelectedManagers.length}개 선택)
+ </Button>
+ </DialogFooter>
+ </DialogContent>
+ </Dialog>
+ )
+}
+
+
diff --git a/components/common/selectors/procurement-manager/procurement-manager-selector.tsx b/components/common/selectors/procurement-manager/procurement-manager-selector.tsx
new file mode 100644
index 00000000..60a28515
--- /dev/null
+++ b/components/common/selectors/procurement-manager/procurement-manager-selector.tsx
@@ -0,0 +1,310 @@
+'use client'
+
+/**
+ * 조달담당자 선택기
+ *
+ * @description
+ * - 오라클에서 CMCTB_CDNM, CMCTB_CD 테이블에서 CD_CLF = 'MMK010' 인 건들을 조회
+ * - PROCUREMENT_MANAGER_CODE: 조달담당자 코드
+ * - DISPLAY_NAME: 담당자명
+ * - DEPARTMENT_NAME: 부서명
+ * - EMPLOYEE_NUMBER: 사번
+ * - 선택 시 사용자 정보도 함께 반환
+ */
+
+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 {
+ getProcurementManagers,
+ addUsersToManagersAsync,
+ ProcurementManager,
+ ProcurementManagerWithUser
+} from './procurement-manager-service'
+import { toast } from 'sonner'
+
+export interface ProcurementManagerSelectorProps {
+ selectedManager?: ProcurementManagerWithUser
+ onManagerSelect: (manager: ProcurementManagerWithUser) => void
+ disabled?: boolean
+ placeholder?: string
+ className?: string
+}
+
+export function ProcurementManagerSelector({
+ selectedManager,
+ onManagerSelect,
+ disabled,
+ placeholder = "조달담당자를 선택하세요",
+ className
+}: ProcurementManagerSelectorProps) {
+ const [open, setOpen] = useState(false)
+ const [managers, setManagers] = useState<ProcurementManager[]>([])
+ const [sorting, setSorting] = useState<SortingState>([])
+ const [columnFilters, setColumnFilters] = useState<ColumnFiltersState>([])
+ const [columnVisibility, setColumnVisibility] = useState<VisibilityState>({})
+ const [rowSelection, setRowSelection] = useState<RowSelectionState>({})
+ const [globalFilter, setGlobalFilter] = useState('')
+ const [isPending, startTransition] = useTransition()
+
+ // 조달담당자 선택 핸들러
+ const handleManagerSelect = useCallback(async (manager: ProcurementManager) => {
+ try {
+ // 사용자 정보 추가
+ const [managerWithUser] = await addUsersToManagersAsync(manager)
+ onManagerSelect(managerWithUser)
+ setOpen(false)
+ } catch (error) {
+ console.error('조달담당자 선택 오류:', error)
+ toast.error('조달담당자 선택 중 오류가 발생했습니다.')
+ }
+ }, [onManagerSelect])
+
+ // 테이블 컬럼 정의
+ const columns: ColumnDef<ProcurementManager>[] = useMemo(() => [
+ {
+ accessorKey: 'PROCUREMENT_MANAGER_CODE',
+ header: '조달담당자코드',
+ cell: ({ row }) => (
+ <div className="font-mono text-sm">{row.getValue('PROCUREMENT_MANAGER_CODE')}</div>
+ ),
+ },
+ {
+ accessorKey: 'DISPLAY_NAME',
+ header: '담당자명',
+ cell: ({ row }) => (
+ <div>{row.getValue('DISPLAY_NAME')}</div>
+ ),
+ },
+ {
+ accessorKey: 'DEPARTMENT_NAME',
+ header: '부서명',
+ cell: ({ row }) => (
+ <div className="text-sm text-muted-foreground">{row.getValue('DEPARTMENT_NAME')}</div>
+ ),
+ },
+ {
+ id: 'actions',
+ header: '선택',
+ cell: ({ row }) => (
+ <Button
+ variant="ghost"
+ size="sm"
+ onClick={(e) => {
+ e.stopPropagation()
+ handleManagerSelect(row.original)
+ }}
+ >
+ <Check className="h-4 w-4" />
+ </Button>
+ ),
+ },
+ ], [handleManagerSelect])
+
+ // 조달담당자 테이블 설정
+ const table = useReactTable({
+ data: managers,
+ 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 loadManagers = useCallback(async () => {
+ startTransition(async () => {
+ try {
+ const result = await getProcurementManagers()
+
+ if (result.success) {
+ setManagers(result.data)
+ } else {
+ toast.error(result.error || '조달담당자를 불러오는데 실패했습니다.')
+ setManagers([])
+ }
+ } catch (error) {
+ console.error('조달담당자 목록 로드 실패:', error)
+ toast.error('조달담당자를 불러오는 중 오류가 발생했습니다.')
+ setManagers([])
+ }
+ })
+ }, [])
+
+ // 다이얼로그 열기 핸들러
+ const handleDialogOpenChange = useCallback((newOpen: boolean) => {
+ setOpen(newOpen)
+ if (newOpen && managers.length === 0) {
+ loadManagers()
+ }
+ }, [loadManagers, managers.length])
+
+ // 검색어 변경 핸들러 (클라이언트 사이드 필터링)
+ const handleSearchChange = useCallback((value: string) => {
+ setGlobalFilter(value)
+ }, [])
+
+ return (
+ <Dialog open={open} onOpenChange={handleDialogOpenChange}>
+ <DialogTrigger asChild>
+ <Button
+ variant="outline"
+ disabled={disabled}
+ className={`w-full justify-start ${className || ''}`}
+ >
+ {selectedManager ? (
+ <div className="flex items-center gap-2 w-full">
+ <span className="font-mono text-sm">[{selectedManager.PROCUREMENT_MANAGER_CODE}]</span>
+ <span className="truncate flex-1 text-left">{selectedManager.DISPLAY_NAME}</span>
+ {selectedManager.DEPARTMENT_NAME && (
+ <span className="text-xs text-muted-foreground">({selectedManager.DEPARTMENT_NAME})</span>
+ )}
+ </div>
+ ) : (
+ <span className="text-muted-foreground">{placeholder}</span>
+ )}
+ </Button>
+ </DialogTrigger>
+ <DialogContent className="max-w-5xl max-h-[80vh]">
+ <DialogHeader>
+ <DialogTitle>조달담당자 선택</DialogTitle>
+ <div className="text-sm text-muted-foreground">
+ 조달담당자(CD_CLF=MMK010) 조회
+ </div>
+ </DialogHeader>
+
+ <div className="space-y-4">
+ <div className="flex items-center space-x-2">
+ <Search className="h-4 w-4" />
+ <Input
+ placeholder="조달담당자코드, 담당자명, 부서명으로 검색..."
+ value={globalFilter}
+ onChange={(e) => handleSearchChange(e.target.value)}
+ className="flex-1"
+ />
+ </div>
+
+ {isPending ? (
+ <div className="flex justify-center py-8">
+ <div className="text-sm text-muted-foreground">조달담당자를 불러오는 중...</div>
+ </div>
+ ) : (
+ <div className="border rounded-md">
+ <Table>
+ <TableHeader>
+ {table.getHeaderGroups().map((headerGroup) => (
+ <TableRow key={headerGroup.id}>
+ {headerGroup.headers.map((header) => (
+ <TableHead key={header.id}>
+ {header.isPlaceholder
+ ? null
+ : flexRender(
+ header.column.columnDef.header,
+ header.getContext()
+ )}
+ </TableHead>
+ ))}
+ </TableRow>
+ ))}
+ </TableHeader>
+ <TableBody>
+ {table.getRowModel().rows?.length ? (
+ table.getRowModel().rows.map((row) => (
+ <TableRow
+ key={row.id}
+ data-state={row.getIsSelected() && "selected"}
+ className="cursor-pointer hover:bg-muted/50"
+ onClick={() => handleManagerSelect(row.original)}
+ >
+ {row.getVisibleCells().map((cell) => (
+ <TableCell key={cell.id}>
+ {flexRender(
+ cell.column.columnDef.cell,
+ cell.getContext()
+ )}
+ </TableCell>
+ ))}
+ </TableRow>
+ ))
+ ) : (
+ <TableRow>
+ <TableCell
+ colSpan={columns.length}
+ className="h-24 text-center"
+ >
+ 검색 결과가 없습니다.
+ </TableCell>
+ </TableRow>
+ )}
+ </TableBody>
+ </Table>
+ </div>
+ )}
+
+ <div className="flex items-center justify-between">
+ <div className="text-sm text-muted-foreground">
+ 총 {table.getFilteredRowModel().rows.length}개 조달담당자
+ </div>
+ <div className="flex items-center space-x-2">
+ <Button
+ variant="outline"
+ size="sm"
+ onClick={() => table.previousPage()}
+ disabled={!table.getCanPreviousPage()}
+ >
+ 이전
+ </Button>
+ <div className="text-sm">
+ {table.getState().pagination.pageIndex + 1} / {table.getPageCount()}
+ </div>
+ <Button
+ variant="outline"
+ size="sm"
+ onClick={() => table.nextPage()}
+ disabled={!table.getCanNextPage()}
+ >
+ 다음
+ </Button>
+ </div>
+ </div>
+ </div>
+ </DialogContent>
+ </Dialog>
+ )
+}
+
diff --git a/components/common/selectors/procurement-manager/procurement-manager-service.ts b/components/common/selectors/procurement-manager/procurement-manager-service.ts
new file mode 100644
index 00000000..e10b58b3
--- /dev/null
+++ b/components/common/selectors/procurement-manager/procurement-manager-service.ts
@@ -0,0 +1,199 @@
+"use server"
+
+import { oracleKnex } from '@/lib/oracle-db/db'
+import { getUserByEmployeeNumber } from '@/lib/users/service'
+import { updateUser } from '@/lib/users/repository'
+
+// 조달담당자 타입 정의
+export interface ProcurementManager {
+ PROCUREMENT_MANAGER_CODE: string // 조달담당자 코드
+ DISPLAY_NAME: string // 담당자명
+ DEPARTMENT_NAME: string // 부서명
+ EMPLOYEE_NUMBER: string // 사번
+ DEPARTMENT_CODE: string // 부서코드
+ IS_ACTIVE: string // 사용 여부
+}
+
+// 조달담당자 + 사용자 정보 타입
+export interface ProcurementManagerWithUser extends ProcurementManager {
+ user?: {
+ id: number
+ name: string
+ email: string
+ employeeNumber: string | null
+ // 필요한 다른 사용자 필드들...
+ } | null
+}
+
+// 테스트 환경용 폴백 데이터
+const FALLBACK_TEST_DATA: ProcurementManager[] = [
+ {
+ PROCUREMENT_MANAGER_CODE: 'PM001',
+ DISPLAY_NAME: '홍길동',
+ DEPARTMENT_NAME: '조달팀',
+ EMPLOYEE_NUMBER: '2001234',
+ DEPARTMENT_CODE: 'D001',
+ IS_ACTIVE: 'Y'
+ },
+ {
+ PROCUREMENT_MANAGER_CODE: 'PM002',
+ DISPLAY_NAME: '김영수',
+ DEPARTMENT_NAME: '구매지원팀',
+ EMPLOYEE_NUMBER: '2002345',
+ DEPARTMENT_CODE: 'D002',
+ IS_ACTIVE: 'Y'
+ },
+ {
+ PROCUREMENT_MANAGER_CODE: 'PM003',
+ DISPLAY_NAME: '이민지',
+ DEPARTMENT_NAME: '조달관리팀',
+ EMPLOYEE_NUMBER: '2003456',
+ DEPARTMENT_CODE: 'D003',
+ IS_ACTIVE: 'Y'
+ },
+ {
+ PROCUREMENT_MANAGER_CODE: 'PM004',
+ DISPLAY_NAME: '박서준',
+ DEPARTMENT_NAME: '자재조달팀',
+ EMPLOYEE_NUMBER: '2004567',
+ DEPARTMENT_CODE: 'D004',
+ IS_ACTIVE: 'Y'
+ },
+ {
+ PROCUREMENT_MANAGER_CODE: 'PM005',
+ DISPLAY_NAME: '최지우',
+ DEPARTMENT_NAME: '조달기획팀',
+ EMPLOYEE_NUMBER: '2005678',
+ DEPARTMENT_CODE: 'D005',
+ IS_ACTIVE: 'Y'
+ },
+]
+
+/**
+ * 모든 조달담당자 목록 조회 (Oracle에서 전체 조회, 실패 시 폴백 데이터 사용)
+ * CMCTB_CDNM, CMCTB_CD 테이블에서 CD_CLF = 'MMK010' 조건으로 조회
+ * 클라이언트에서 검색/필터링 수행
+ */
+export async function getProcurementManagers(): Promise<{
+ success: boolean
+ data: ProcurementManager[]
+ error?: string
+ isUsingFallback?: boolean
+}> {
+ try {
+ console.log('📋 [getProcurementManagers] Oracle 쿼리 시작...')
+
+ const result = await oracleKnex.raw(`
+ SELECT
+ NM.CD AS PROCUREMENT_MANAGER_CODE,
+ NM.CDNM AS DISPLAY_NAME,
+ NM.GRP_DSC AS DEPARTMENT_NAME,
+ CD.USR_DF_CHAR_1 AS EMPLOYEE_NUMBER,
+ CD.USR_DF_CHAR_2 AS DEPARTMENT_CODE,
+ CD.USR_DF_CHK_1 AS IS_ACTIVE
+ FROM CMCTB_CD CD
+ JOIN CMCTB_CDNM NM
+ ON CD.CD_CLF = NM.CD_CLF
+ AND CD.CD = NM.CD
+ AND CD.CD2 = NM.CD2
+ AND CD.CD3 = NM.CD3
+ WHERE CD.CD_CLF = 'MMK010'
+ AND CD.DEL_YN != 'Y'
+ ORDER BY NM.CD
+ `)
+
+ // Oracle raw query의 결과는 rows 배열에 들어있음
+ const rows = (result.rows || result) as Array<Record<string, unknown>>
+
+ console.log(`✅ [getProcurementManagers] Oracle 쿼리 성공 - ${rows.length}건 조회`)
+
+ // null 값 필터링
+ const cleanedResult = rows
+ .filter((item) =>
+ item.PROCUREMENT_MANAGER_CODE &&
+ item.DISPLAY_NAME &&
+ item.EMPLOYEE_NUMBER
+ )
+ .map((item) => ({
+ PROCUREMENT_MANAGER_CODE: String(item.PROCUREMENT_MANAGER_CODE),
+ DISPLAY_NAME: String(item.DISPLAY_NAME),
+ DEPARTMENT_NAME: String(item.DEPARTMENT_NAME || ''),
+ EMPLOYEE_NUMBER: String(item.EMPLOYEE_NUMBER),
+ DEPARTMENT_CODE: String(item.DEPARTMENT_CODE || ''),
+ IS_ACTIVE: String(item.IS_ACTIVE || 'Y')
+ }))
+
+ console.log(`✅ [getProcurementManagers] 필터링 후 ${cleanedResult.length}건`)
+
+ return {
+ success: true,
+ data: cleanedResult,
+ isUsingFallback: false
+ }
+ } catch (error) {
+ console.error('❌ [getProcurementManagers] Oracle 오류:', error)
+ console.log('🔄 [getProcurementManagers] 폴백 테스트 데이터 사용 (' + FALLBACK_TEST_DATA.length + '건)')
+ return {
+ success: true,
+ data: FALLBACK_TEST_DATA,
+ isUsingFallback: true
+ }
+ }
+}
+
+/**
+ * 조달담당자(들)에 사용자 정보 추가 및 procurementManagerCode 업데이트 (1개 또는 여러 개 처리)
+ * 선택 시 해당 사용자의 procurementManagerCode를 조달담당자 코드로 업데이트
+ * @param managers - 단일 또는 배열 형태의 조달담당자
+ * @returns 사용자 정보가 포함된 조달담당자 배열
+ */
+export async function addUsersToManagersAsync(
+ managers: ProcurementManager | ProcurementManager[]
+): Promise<ProcurementManagerWithUser[]> {
+ try {
+ // 배열로 정규화
+ const managersArray = Array.isArray(managers) ? managers : [managers]
+
+ console.log(`👤 [addUsersToManagersAsync] ${managersArray.length}개 조달담당자에 대한 사용자 정보 조회 및 procurementManagerCode 업데이트 시작`)
+
+ // 각 조달담당자에 대해 사용자 정보 조회 및 procurementManagerCode 업데이트
+ const withUsers = await Promise.all(
+ managersArray.map(async (manager) => {
+ console.log(` 🔍 [addUsersToManagersAsync] 사번 ${manager.EMPLOYEE_NUMBER} 조회 중...`)
+ const user = await getUserByEmployeeNumber(manager.EMPLOYEE_NUMBER)
+
+ if (user) {
+ console.log(` ✅ [addUsersToManagersAsync] 사번 ${manager.EMPLOYEE_NUMBER} → 사용자 찾음: ${user.name}`)
+
+ // 사용자의 procurementManagerCode를 조달담당자 코드로 업데이트
+ try {
+ console.log(` 📝 [addUsersToManagersAsync] 사용자 ${user.id}의 procurementManagerCode를 ${manager.PROCUREMENT_MANAGER_CODE}로 업데이트 중...`)
+ await updateUser(user.id, { procurementManagerCode: manager.PROCUREMENT_MANAGER_CODE })
+ console.log(` ✅ [addUsersToManagersAsync] 사용자 ${user.id}의 procurementManagerCode 업데이트 완료`)
+ } catch (updateError) {
+ console.error(` ❌ [addUsersToManagersAsync] 사용자 ${user.id}의 procurementManagerCode 업데이트 실패:`, updateError)
+ }
+ } else {
+ console.log(` ⚠️ [addUsersToManagersAsync] 사번 ${manager.EMPLOYEE_NUMBER} → 사용자 없음`)
+ }
+
+ return {
+ ...manager,
+ user: user ? {
+ id: user.id,
+ name: user.name,
+ email: user.email,
+ employeeNumber: user.employeeNumber,
+ } : null
+ }
+ })
+ )
+
+ return withUsers
+ } catch (error) {
+ console.error('❌ [addUsersToManagersAsync] 오류:', error)
+ const managersArray = Array.isArray(managers) ? managers : [managers]
+ return managersArray.map(manager => ({ ...manager, user: null }))
+ }
+}
+
diff --git a/components/common/selectors/procurement-manager/procurement-manager-single-selector.tsx b/components/common/selectors/procurement-manager/procurement-manager-single-selector.tsx
new file mode 100644
index 00000000..4667385a
--- /dev/null
+++ b/components/common/selectors/procurement-manager/procurement-manager-single-selector.tsx
@@ -0,0 +1,374 @@
+'use client'
+
+/**
+ * 조달담당자 단일 선택 다이얼로그
+ *
+ * @description
+ * - 조달담당자를 하나만 선택할 수 있는 다이얼로그
+ * - 트리거 버튼과 다이얼로그가 분리된 구조
+ * - 외부에서 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 {
+ getProcurementManagers,
+ addUsersToManagersAsync,
+ ProcurementManager,
+ ProcurementManagerWithUser
+} from './procurement-manager-service'
+import { toast } from 'sonner'
+
+export interface ProcurementManagerSingleSelectorProps {
+ open: boolean
+ onOpenChange: (open: boolean) => void
+ selectedManager?: ProcurementManagerWithUser
+ onManagerSelect: (manager: ProcurementManagerWithUser) => void
+ onConfirm?: (manager: ProcurementManagerWithUser | undefined) => void
+ onCancel?: () => void
+ title?: string
+ description?: string
+ showConfirmButtons?: boolean
+}
+
+export function ProcurementManagerSingleSelector({
+ open,
+ onOpenChange,
+ selectedManager,
+ onManagerSelect,
+ onConfirm,
+ onCancel,
+ title = "조달담당자 선택",
+ description = "조달담당자를 선택하세요",
+ showConfirmButtons = false
+}: ProcurementManagerSingleSelectorProps) {
+ const [managers, setManagers] = useState<ProcurementManager[]>([])
+ const [sorting, setSorting] = useState<SortingState>([])
+ const [columnFilters, setColumnFilters] = useState<ColumnFiltersState>([])
+ const [columnVisibility, setColumnVisibility] = useState<VisibilityState>({})
+ const [rowSelection, setRowSelection] = useState<RowSelectionState>({})
+ const [globalFilter, setGlobalFilter] = useState('')
+ const [isPending, startTransition] = useTransition()
+ const [tempSelectedManager, setTempSelectedManager] = useState<ProcurementManagerWithUser | undefined>(selectedManager)
+
+ // 조달담당자 선택 핸들러
+ const handleManagerSelect = useCallback(async (manager: ProcurementManager) => {
+ try {
+ // 사용자 정보 추가
+ const [managerWithUser] = await addUsersToManagersAsync(manager)
+ if (showConfirmButtons) {
+ setTempSelectedManager(managerWithUser)
+ } else {
+ onManagerSelect(managerWithUser)
+ onOpenChange(false)
+ }
+ } catch (error) {
+ console.error('조달담당자 선택 오류:', error)
+ toast.error('조달담당자 선택 중 오류가 발생했습니다.')
+ }
+ }, [onManagerSelect, onOpenChange, showConfirmButtons])
+
+ // 확인 버튼 핸들러
+ const handleConfirm = useCallback(() => {
+ if (tempSelectedManager) {
+ onManagerSelect(tempSelectedManager)
+ }
+ onConfirm?.(tempSelectedManager)
+ onOpenChange(false)
+ }, [tempSelectedManager, onManagerSelect, onConfirm, onOpenChange])
+
+ // 취소 버튼 핸들러
+ const handleCancel = useCallback(() => {
+ setTempSelectedManager(selectedManager)
+ onCancel?.()
+ onOpenChange(false)
+ }, [selectedManager, onCancel, onOpenChange])
+
+ // 테이블 컬럼 정의
+ const columns: ColumnDef<ProcurementManager>[] = useMemo(() => [
+ {
+ accessorKey: 'PROCUREMENT_MANAGER_CODE',
+ header: '조달담당자코드',
+ cell: ({ row }) => (
+ <div className="font-mono text-sm">{row.getValue('PROCUREMENT_MANAGER_CODE')}</div>
+ ),
+ },
+ {
+ accessorKey: 'DISPLAY_NAME',
+ header: '담당자명',
+ cell: ({ row }) => (
+ <div>{row.getValue('DISPLAY_NAME')}</div>
+ ),
+ },
+ {
+ accessorKey: 'DEPARTMENT_NAME',
+ header: '부서명',
+ cell: ({ row }) => (
+ <div className="text-sm text-muted-foreground">{row.getValue('DEPARTMENT_NAME')}</div>
+ ),
+ },
+ {
+ id: 'actions',
+ header: '선택',
+ cell: ({ row }) => {
+ const isSelected = showConfirmButtons
+ ? tempSelectedManager?.PROCUREMENT_MANAGER_CODE === row.original.PROCUREMENT_MANAGER_CODE
+ : selectedManager?.PROCUREMENT_MANAGER_CODE === row.original.PROCUREMENT_MANAGER_CODE
+
+ return (
+ <Button
+ variant={isSelected ? "default" : "ghost"}
+ size="sm"
+ onClick={(e) => {
+ e.stopPropagation()
+ handleManagerSelect(row.original)
+ }}
+ >
+ <Check className="h-4 w-4" />
+ </Button>
+ )
+ },
+ },
+ ], [handleManagerSelect, selectedManager, tempSelectedManager, showConfirmButtons])
+
+ // 조달담당자 테이블 설정
+ const table = useReactTable({
+ data: managers,
+ 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 loadManagers = useCallback(async () => {
+ startTransition(async () => {
+ try {
+ const result = await getProcurementManagers()
+
+ if (result.success) {
+ setManagers(result.data)
+
+ // 폴백 데이터를 사용하는 경우 알림
+ if (result.isUsingFallback) {
+ toast.info('Oracle 연결 실패', {
+ description: '테스트 데이터를 사용합니다.',
+ duration: 4000,
+ })
+ }
+ } else {
+ toast.error(result.error || '조달담당자를 불러오는데 실패했습니다.')
+ setManagers([])
+ }
+ } catch (error) {
+ console.error('조달담당자 목록 로드 실패:', error)
+ toast.error('조달담당자를 불러오는 중 오류가 발생했습니다.')
+ setManagers([])
+ }
+ })
+ }, [])
+
+ // 다이얼로그 열릴 때 코드 로드 (open prop 변화 감지)
+ useEffect(() => {
+ if (open) {
+ setTempSelectedManager(selectedManager)
+ if (managers.length === 0) {
+ console.log('🚀 [ProcurementManagerSingleSelector] 다이얼로그 열림 - loadManagers 호출')
+ loadManagers()
+ } else {
+ console.log('📦 [ProcurementManagerSingleSelector] 다이얼로그 열림 - 기존 데이터 사용 (' + managers.length + '건)')
+ }
+ }
+ }, [open, selectedManager, loadManagers, managers.length])
+
+ // 검색어 변경 핸들러 (클라이언트 사이드 필터링)
+ const handleSearchChange = useCallback((value: string) => {
+ setGlobalFilter(value)
+ }, [])
+
+ const currentSelectedManager = showConfirmButtons ? tempSelectedManager : selectedManager
+
+ return (
+ <Dialog open={open} onOpenChange={onOpenChange}>
+ <DialogContent className="max-w-5xl max-h-[80vh]">
+ <DialogHeader>
+ <DialogTitle>{title}</DialogTitle>
+ <div className="text-sm text-muted-foreground">
+ {description}
+ </div>
+ </DialogHeader>
+
+ <div className="space-y-4">
+ {/* 현재 선택된 조달담당자 표시 */}
+ {currentSelectedManager && (
+ <div className="p-3 bg-muted rounded-md">
+ <div className="text-sm font-medium">선택된 조달담당자:</div>
+ <div className="flex items-center gap-2 mt-1">
+ <span className="font-mono text-sm">[{currentSelectedManager.PROCUREMENT_MANAGER_CODE}]</span>
+ <span>{currentSelectedManager.DISPLAY_NAME}</span>
+ {currentSelectedManager.DEPARTMENT_NAME && (
+ <span className="text-muted-foreground">({currentSelectedManager.DEPARTMENT_NAME})</span>
+ )}
+ {currentSelectedManager.user && (
+ <span className="text-muted-foreground text-xs">- {currentSelectedManager.user.name}</span>
+ )}
+ </div>
+ </div>
+ )}
+
+ <div className="flex items-center space-x-2">
+ <Search className="h-4 w-4" />
+ <Input
+ placeholder="조달담당자코드, 담당자명, 부서명으로 검색..."
+ value={globalFilter}
+ onChange={(e) => handleSearchChange(e.target.value)}
+ className="flex-1"
+ />
+ </div>
+
+ {isPending ? (
+ <div className="flex justify-center py-8">
+ <div className="text-sm text-muted-foreground">조달담당자를 불러오는 중...</div>
+ </div>
+ ) : (
+ <div className="border rounded-md">
+ <Table>
+ <TableHeader>
+ {table.getHeaderGroups().map((headerGroup) => (
+ <TableRow key={headerGroup.id}>
+ {headerGroup.headers.map((header) => (
+ <TableHead key={header.id}>
+ {header.isPlaceholder
+ ? null
+ : flexRender(
+ header.column.columnDef.header,
+ header.getContext()
+ )}
+ </TableHead>
+ ))}
+ </TableRow>
+ ))}
+ </TableHeader>
+ <TableBody>
+ {table.getRowModel().rows?.length ? (
+ table.getRowModel().rows.map((row) => {
+ const isRowSelected = currentSelectedManager?.PROCUREMENT_MANAGER_CODE === row.original.PROCUREMENT_MANAGER_CODE
+ return (
+ <TableRow
+ key={row.id}
+ data-state={isRowSelected && "selected"}
+ className={`cursor-pointer hover:bg-muted/50 ${
+ isRowSelected ? 'bg-muted' : ''
+ }`}
+ onClick={() => handleManagerSelect(row.original)}
+ >
+ {row.getVisibleCells().map((cell) => (
+ <TableCell key={cell.id}>
+ {flexRender(
+ cell.column.columnDef.cell,
+ cell.getContext()
+ )}
+ </TableCell>
+ ))}
+ </TableRow>
+ )
+ })
+ ) : (
+ <TableRow>
+ <TableCell
+ colSpan={columns.length}
+ className="h-24 text-center"
+ >
+ 검색 결과가 없습니다.
+ </TableCell>
+ </TableRow>
+ )}
+ </TableBody>
+ </Table>
+ </div>
+ )}
+
+ <div className="flex items-center justify-between">
+ <div className="text-sm text-muted-foreground">
+ 총 {table.getFilteredRowModel().rows.length}개 조달담당자
+ </div>
+ <div className="flex items-center space-x-2">
+ <Button
+ variant="outline"
+ size="sm"
+ onClick={() => table.previousPage()}
+ disabled={!table.getCanPreviousPage()}
+ >
+ 이전
+ </Button>
+ <div className="text-sm">
+ {table.getState().pagination.pageIndex + 1} / {table.getPageCount()}
+ </div>
+ <Button
+ variant="outline"
+ size="sm"
+ onClick={() => table.nextPage()}
+ disabled={!table.getCanNextPage()}
+ >
+ 다음
+ </Button>
+ </div>
+ </div>
+ </div>
+
+ {showConfirmButtons && (
+ <DialogFooter>
+ <Button variant="outline" onClick={handleCancel}>
+ <X className="h-4 w-4 mr-2" />
+ 취소
+ </Button>
+ <Button onClick={handleConfirm} disabled={!tempSelectedManager}>
+ <Check className="h-4 w-4 mr-2" />
+ 확인
+ </Button>
+ </DialogFooter>
+ )}
+ </DialogContent>
+ </Dialog>
+ )
+}
+
+