'use client'; /** * 어음관리 - UniversalListPage 마이그레이션 * * IntegratedListTemplateV2 → UniversalListPage config 기반으로 변환 * - 서버 사이드 필터링/페이지네이션 * - dateRangeSelector + 등록 버튼 (headerActions) * - beforeTableContent: 상태 선택 + 저장 버튼 + 수취/발행 라디오 * - tableHeaderActions: 거래처, 구분, 상태 필터 */ import { useState, useMemo, useCallback } from 'react'; import { useRouter } from 'next/navigation'; import { FileText, Plus, Pencil, Trash2, Save, } from 'lucide-react'; import { Button } from '@/components/ui/button'; import { Checkbox } from '@/components/ui/checkbox'; import { Badge } from '@/components/ui/badge'; import { DeleteConfirmDialog } from '@/components/ui/confirm-dialog'; import { TableRow, TableCell } from '@/components/ui/table'; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue, } from '@/components/ui/select'; import { RadioGroup, RadioGroupItem } from '@/components/ui/radio-group'; import { Label } from '@/components/ui/label'; import { UniversalListPage, type UniversalListConfig, type TableColumn, type SelectionHandlers, type RowClickHandlers, } from '@/components/templates/UniversalListPage'; import { ListMobileCard, InfoField } from '@/components/organisms/MobileCard'; import { toast } from 'sonner'; import type { BillRecord, BillType, BillStatus, SortOption, } from './types'; import { BILL_TYPE_LABELS, BILL_TYPE_FILTER_OPTIONS, BILL_STATUS_COLORS, BILL_STATUS_FILTER_OPTIONS, getBillStatusLabel, } from './types'; import { getBills, deleteBill, updateBillStatus } from './actions'; interface BillManagementClientProps { initialData: BillRecord[]; initialPagination: { currentPage: number; lastPage: number; perPage: number; total: number; }; initialVendorId?: string; initialBillType?: string; } export function BillManagementClient({ initialData, initialPagination, initialVendorId, initialBillType, }: BillManagementClientProps) { const router = useRouter(); // ===== 상태 관리 ===== const [data, setData] = useState(initialData); const [pagination, setPagination] = useState(initialPagination); const [isLoading, setIsLoading] = useState(false); const [searchQuery, setSearchQuery] = useState(''); const [sortOption, setSortOption] = useState('latest'); const [billTypeFilter, setBillTypeFilter] = useState(initialBillType || 'received'); const [vendorFilter, setVendorFilter] = useState(initialVendorId || 'all'); const [statusFilter, setStatusFilter] = useState('all'); const [selectedItems, setSelectedItems] = useState>(new Set()); const [currentPage, setCurrentPage] = useState(initialPagination.currentPage); const itemsPerPage = initialPagination.perPage; // 삭제 다이얼로그 const [showDeleteDialog, setShowDeleteDialog] = useState(false); const [deleteTargetId, setDeleteTargetId] = useState(null); // 날짜 범위 상태 const [startDate, setStartDate] = useState('2025-09-01'); const [endDate, setEndDate] = useState('2025-09-03'); // ===== API 데이터 로드 ===== const loadData = useCallback(async (page: number = 1) => { setIsLoading(true); try { const result = await getBills({ search: searchQuery || undefined, billType: billTypeFilter !== 'all' ? billTypeFilter : undefined, status: statusFilter !== 'all' ? statusFilter : undefined, clientId: vendorFilter !== 'all' ? vendorFilter : undefined, issueStartDate: startDate, issueEndDate: endDate, sortBy: sortOption === 'latest' || sortOption === 'oldest' ? 'issue_date' : sortOption === 'amountHigh' || sortOption === 'amountLow' ? 'amount' : 'maturity_date', sortDir: sortOption === 'oldest' || sortOption === 'amountLow' ? 'asc' : 'desc', perPage: itemsPerPage, page, }); if (result.success) { setData(result.data); setPagination(result.pagination); setCurrentPage(result.pagination.currentPage); } else { toast.error(result.error || '데이터를 불러오는데 실패했습니다.'); } } catch { toast.error('데이터를 불러오는데 실패했습니다.'); } finally { setIsLoading(false); } }, [searchQuery, billTypeFilter, statusFilter, vendorFilter, startDate, endDate, sortOption, itemsPerPage]); // ===== 체크박스 핸들러 ===== const toggleSelection = useCallback((id: string) => { setSelectedItems(prev => { const newSet = new Set(prev); if (newSet.has(id)) newSet.delete(id); else newSet.add(id); return newSet; }); }, []); // ===== 전체 선택 핸들러 ===== const toggleSelectAll = useCallback(() => { if (selectedItems.size === data.length && data.length > 0) { setSelectedItems(new Set()); } else { setSelectedItems(new Set(data.map(item => item.id))); } }, [selectedItems.size, data]); // ===== 액션 핸들러 ===== const handleRowClick = useCallback((item: BillRecord) => { router.push(`/ko/accounting/bills/${item.id}`); }, [router]); const handleDeleteClick = useCallback((id: string) => { setDeleteTargetId(id); setShowDeleteDialog(true); }, []); const handleConfirmDelete = useCallback(async () => { if (deleteTargetId) { setIsLoading(true); const result = await deleteBill(deleteTargetId); if (result.success) { setData(prev => prev.filter(item => item.id !== deleteTargetId)); setSelectedItems(prev => { const newSet = new Set(prev); newSet.delete(deleteTargetId); return newSet; }); toast.success('삭제되었습니다.'); } else { toast.error(result.error || '삭제에 실패했습니다.'); } setIsLoading(false); } setShowDeleteDialog(false); setDeleteTargetId(null); }, [deleteTargetId]); // ===== 페이지 변경 ===== const handlePageChange = useCallback((page: number) => { loadData(page); }, [loadData]); // ===== 테이블 컬럼 ===== const tableColumns: TableColumn[] = useMemo(() => [ { key: 'no', label: '번호', className: 'text-center w-[60px]' }, { key: 'billNumber', label: '어음번호' }, { key: 'billType', label: '구분', className: 'text-center' }, { key: 'vendorName', label: '거래처' }, { key: 'amount', label: '금액', className: 'text-right' }, { key: 'issueDate', label: '발행일' }, { key: 'maturityDate', label: '만기일' }, { key: 'installmentCount', label: '차수', className: 'text-center' }, { key: 'status', label: '상태', className: 'text-center' }, { key: 'actions', label: '작업', className: 'text-center w-[80px]' }, ], []); // ===== 테이블 행 렌더링 ===== const renderTableRow = useCallback(( item: BillRecord, index: number, globalIndex: number, handlers: SelectionHandlers & RowClickHandlers ) => { return ( handleRowClick(item)} > e.stopPropagation()}> {globalIndex} {item.billNumber} {BILL_TYPE_LABELS[item.billType]} {item.vendorName} {item.amount.toLocaleString()} {item.issueDate} {item.maturityDate} {item.installmentCount || '-'} {getBillStatusLabel(item.billType, item.status)} e.stopPropagation()}> {handlers.isSelected && (
)}
); }, [handleRowClick, handleDeleteClick, router]); // ===== 모바일 카드 렌더링 ===== const renderMobileCard = useCallback(( item: BillRecord, index: number, globalIndex: number, handlers: SelectionHandlers & RowClickHandlers ) => { return ( {BILL_TYPE_LABELS[item.billType]} {getBillStatusLabel(item.billType, item.status)} } isSelected={handlers.isSelected} onToggleSelection={handlers.onToggle} infoGrid={
} actions={ handlers.isSelected ? (
) : undefined } onCardClick={() => handleRowClick(item)} /> ); }, [handleRowClick, handleDeleteClick, router]); // ===== 거래처 목록 (필터용) ===== const vendorOptions = useMemo(() => { const uniqueVendors = [...new Set(data.map(d => d.vendorName).filter(v => v))]; return [ { value: 'all', label: '전체' }, ...uniqueVendors.map(v => ({ value: v, label: v })) ]; }, [data]); // ===== 저장 핸들러 ===== const handleSave = useCallback(async () => { if (selectedItems.size === 0) { toast.warning('선택된 항목이 없습니다.'); return; } if (statusFilter === 'all') { toast.warning('상태를 선택해주세요.'); return; } setIsLoading(true); let successCount = 0; for (const id of selectedItems) { const result = await updateBillStatus(id, statusFilter as BillStatus); if (result.success) { successCount++; } } if (successCount > 0) { toast.success(`${successCount}건이 저장되었습니다.`); loadData(currentPage); setSelectedItems(new Set()); } else { toast.error('저장에 실패했습니다.'); } setIsLoading(false); }, [selectedItems, statusFilter, loadData, currentPage]); // ===== UniversalListPage Config ===== const config: UniversalListConfig = useMemo( () => ({ // 페이지 기본 정보 title: '어음관리', description: '어음 및 수취이음 상세 현황을 관리합니다', icon: FileText, basePath: '/accounting/bills', // ID 추출 idField: 'id', // API 액션 actions: { getList: async () => { return { success: true, data: data, totalCount: pagination.total, }; }, }, // 테이블 컬럼 columns: tableColumns, // 서버 사이드 필터링 clientSideFiltering: false, itemsPerPage: pagination.perPage, // 검색 searchPlaceholder: '어음번호, 거래처, 메모 검색...', onSearchChange: setSearchQuery, // 모바일 필터 설정 filterConfig: [ { key: 'vendorFilter', label: '거래처', type: 'single', options: vendorOptions.filter(o => o.value !== 'all'), }, { key: 'billType', label: '구분', type: 'single', options: BILL_TYPE_FILTER_OPTIONS.filter(o => o.value !== 'all'), }, { key: 'status', label: '상태', type: 'single', options: BILL_STATUS_FILTER_OPTIONS.filter(o => o.value !== 'all'), }, ], initialFilters: { vendorFilter: vendorFilter, billType: billTypeFilter, status: statusFilter, }, filterTitle: '어음 필터', // 날짜 선택기 dateRangeSelector: { enabled: true, startDate, endDate, onStartDateChange: setStartDate, onEndDateChange: setEndDate, }, // 등록 버튼 createButton: { label: '어음 등록', onClick: () => router.push('/ko/accounting/bills/new'), icon: Plus, }, // 테이블 헤더 액션 (필터) tableHeaderActions: (
), // beforeTableContent: 상태 선택 + 저장 + 수취/발행 라디오 beforeTableContent: (
{ setBillTypeFilter(value); loadData(1); }} className="flex items-center gap-4" >
), // 렌더링 함수 renderTableRow, renderMobileCard, }), [ data, pagination, tableColumns, startDate, endDate, vendorFilter, vendorOptions, billTypeFilter, statusFilter, isLoading, router, loadData, handleSave, renderTableRow, renderMobileCard, ] ); return ( <> item.id, }} /> ); }