diff options
| author | joonhoekim <26rote@gmail.com> | 2025-11-13 18:16:05 +0900 |
|---|---|---|
| committer | joonhoekim <26rote@gmail.com> | 2025-11-13 18:16:05 +0900 |
| commit | 1b84707d2f7abba3349fc306b539ef661a22cd45 (patch) | |
| tree | a49d0f89acd99b09bec376708e1161b4aa16318f /components/common/selectors/currency/currency-selector.tsx | |
| parent | 517a3e91a899c54a70de76d380eda5f73e8133e9 (diff) | |
(김준회) 공통컴포넌트: 통화(Currency) 선택기 추가
Diffstat (limited to 'components/common/selectors/currency/currency-selector.tsx')
| -rw-r--r-- | components/common/selectors/currency/currency-selector.tsx | 299 |
1 files changed, 299 insertions, 0 deletions
diff --git a/components/common/selectors/currency/currency-selector.tsx b/components/common/selectors/currency/currency-selector.tsx new file mode 100644 index 00000000..1e56c972 --- /dev/null +++ b/components/common/selectors/currency/currency-selector.tsx @@ -0,0 +1,299 @@ +'use client' + +/** + * 통화 선택기 + * + * @description + * - 오라클에서 CMCTB_CD, CMCTB_CDNM 테이블에서 CD_CLF = 'SPB032' 인 건들을 조회 + * - CURRENCY_CODE: 통화 코드 (예: KRW, USD, JPY) + * - CURRENCY_NAME: 통화 이름/설명 + * - DECIMAL_PLACES: 입력가능 소수점 자리수 (원화/엔화: 0, 기타: 2) + */ + +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 { + getCurrencies, + Currency +} from './currency-service' +import { toast } from 'sonner' + +export interface CurrencySelectorProps { + selectedCurrency?: Currency + onCurrencySelect: (currency: Currency) => void + disabled?: boolean + placeholder?: string + className?: string +} + +export function CurrencySelector({ + selectedCurrency, + onCurrencySelect, + disabled, + placeholder = "통화를 선택하세요", + className +}: CurrencySelectorProps) { + const [open, setOpen] = useState(false) + const [currencies, setCurrencies] = useState<Currency[]>([]) + 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 handleCurrencySelect = useCallback((currency: Currency) => { + onCurrencySelect(currency) + setOpen(false) + }, [onCurrencySelect]) + + // 테이블 컬럼 정의 + const columns: ColumnDef<Currency>[] = useMemo(() => [ + { + accessorKey: 'CURRENCY_CODE', + header: '통화코드', + cell: ({ row }) => ( + <div className="font-mono text-sm font-semibold">{row.getValue('CURRENCY_CODE')}</div> + ), + }, + { + accessorKey: 'CURRENCY_NAME', + header: '통화명', + cell: ({ row }) => ( + <div>{row.getValue('CURRENCY_NAME')}</div> + ), + }, + { + accessorKey: 'DECIMAL_PLACES', + header: '소수점', + cell: ({ row }) => ( + <div className="text-center text-sm text-muted-foreground"> + {row.getValue('DECIMAL_PLACES')}자리 + </div> + ), + }, + { + id: 'actions', + header: '선택', + cell: ({ row }) => ( + <Button + variant="ghost" + size="sm" + onClick={(e) => { + e.stopPropagation() + handleCurrencySelect(row.original) + }} + > + <Check className="h-4 w-4" /> + </Button> + ), + }, + ], [handleCurrencySelect]) + + // 통화 테이블 설정 + const table = useReactTable({ + data: currencies, + 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 loadCurrencies = useCallback(async () => { + startTransition(async () => { + try { + const result = await getCurrencies() + + if (result.success) { + setCurrencies(result.data) + } else { + toast.error(result.error || '통화 목록을 불러오는데 실패했습니다.') + setCurrencies([]) + } + } catch (error) { + console.error('통화 목록 로드 실패:', error) + toast.error('통화 목록을 불러오는 중 오류가 발생했습니다.') + setCurrencies([]) + } + }) + }, []) + + // 다이얼로그 열기 핸들러 + const handleDialogOpenChange = useCallback((newOpen: boolean) => { + setOpen(newOpen) + if (newOpen && currencies.length === 0) { + loadCurrencies() + } + }, [loadCurrencies, currencies.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 || ''}`} + > + {selectedCurrency ? ( + <div className="flex items-center gap-2 w-full"> + <span className="font-mono text-sm font-semibold">[{selectedCurrency.CURRENCY_CODE}]</span> + <span className="truncate flex-1 text-left">{selectedCurrency.CURRENCY_NAME}</span> + <span className="text-xs text-muted-foreground">({selectedCurrency.DECIMAL_PLACES}자리)</span> + </div> + ) : ( + <span className="text-muted-foreground">{placeholder}</span> + )} + </Button> + </DialogTrigger> + <DialogContent className="max-w-4xl max-h-[80vh]"> + <DialogHeader> + <DialogTitle>통화 선택</DialogTitle> + <div className="text-sm text-muted-foreground"> + 통화 코드(CD_CLF=SPB032) 조회 + </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={() => handleCurrencySelect(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> + ) +} + |
