"use client"; /** * 공급업체관리 — Type B 마스터-디테일 레이아웃 (리디자인) * * 좌측: 공급업체 목록 (supplier_mng) * 우측: 품목별 단가 + 납품처 정보 탭 * * 모달: * - 공급업체 등록/수정 (supplier_mng) * - 품목 추가 (item_info 검색 → supplier_item_mapping + supplier_item_prices) * - 납품처 등록 (delivery_destination) */ import React, { useState, useEffect, useCallback, useRef, useMemo } from "react"; import { Button } from "@/components/ui/button"; import { Input } from "@/components/ui/input"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table"; import { Badge } from "@/components/ui/badge"; import { Tabs, TabsList, TabsTrigger, TabsContent } from "@/components/ui/tabs"; import { Label } from "@/components/ui/label"; import { ResizableHandle, ResizablePanel, ResizablePanelGroup } from "@/components/ui/resizable"; import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter, DialogDescription } from "@/components/ui/dialog"; import { Plus, Trash2, Save, Loader2, FileSpreadsheet, Download, Pencil, Users, Package, MapPin, Search, X, Tag, Coins, Settings2, GripVertical, ChevronRight, ChevronDown, } from "lucide-react"; import { DndContext, closestCenter, PointerSensor, useSensor, useSensors, DragEndEvent, } from "@dnd-kit/core"; import { SortableContext, verticalListSortingStrategy, useSortable, arrayMove } from "@dnd-kit/sortable"; import { CSS } from "@dnd-kit/utilities"; import { cn } from "@/lib/utils"; import { apiClient } from "@/lib/api/client"; import { useAuth } from "@/hooks/useAuth"; import { toast } from "sonner"; import { useConfirmDialog } from "@/components/common/ConfirmDialog"; import { MultiTableExcelUploadModal } from "@/components/common/MultiTableExcelUploadModal"; import { autoDetectMultiTableConfig, TableChainConfig } from "@/lib/api/multiTableExcel"; import { exportToExcel } from "@/lib/utils/excelExport"; import { validateField, validateForm, formatField } from "@/lib/utils/validation"; import { getAvailableNumberingRulesForScreen, previewNumberingCode, allocateNumberingCode } from "@/lib/api/numberingRule"; import { TableSettingsModal } from "@/components/common/TableSettingsModal"; import { useTableSettings } from "@/hooks/useTableSettings"; import { DynamicSearchFilter, FilterValue } from "@/components/common/DynamicSearchFilter"; import { EDataTable, EDataTableColumn } from "@/components/common/EDataTable"; const SUPPLIER_TABLE = "supplier_mng"; const MAPPING_TABLE = "supplier_item_mapping"; const PRICE_TABLE = "supplier_item_prices"; const DELIVERY_TABLE = "delivery_destination"; const CONTACT_TABLE = "supplier_contact"; const SUPPLIER_GRID_COLUMNS = [ { key: "supplier_code", label: "공급업체코드" }, { key: "supplier_name", label: "공급업체명" }, { key: "division", label: "공급업체유형" }, { key: "contact_person", label: "담당자" }, { key: "contact_phone", label: "전화번호" }, { key: "email", label: "이메일" }, { key: "business_number", label: "사업자번호" }, { key: "address", label: "주소" }, { key: "status", label: "상태" }, ]; function SortableMappingRow({ id, children }: { id: string; children: React.ReactNode }) { const { attributes, listeners, setNodeRef, transform, transition, isDragging } = useSortable({ id }); const style: React.CSSProperties = { transform: CSS.Transform.toString(transform), transition, opacity: isDragging ? 0.5 : 1, }; return (
{children}
); } export default function SupplierManagementPage() { const { user } = useAuth(); const { confirm, ConfirmDialogComponent, isConfirmOpenRef } = useConfirmDialog(); const ts = useTableSettings("c16-supplier", SUPPLIER_TABLE, SUPPLIER_GRID_COLUMNS); const dndSensors = useSensors(useSensor(PointerSensor, { activationConstraint: { distance: 5 } })); // 검색 필터 (DynamicSearchFilter에서 관리) const [searchFilters, setSearchFilters] = useState([]); // 좌측: 공급업체 목록 const [suppliers, setSuppliers] = useState([]); const [rawSuppliers, setRawSuppliers] = useState([]); const [supplierLoading, setSupplierLoading] = useState(false); const [showInactive, setShowInactive] = useState(false); const [mainContactMap, setMainContactMap] = useState>({}); const [supplierCount, setSupplierCount] = useState(0); const [selectedSupplierId, setSelectedSupplierId] = useState(null); // 우측: 탭 const [rightTab, setRightTab] = useState<"items" | "delivery">("items"); // 우측: 품목 단가 const [priceItems, setPriceItems] = useState([]); const [priceGroups, setPriceGroups] = useState>({}); const [priceLoading, setPriceLoading] = useState(false); const [priceCheckedIds, setPriceCheckedIds] = useState([]); const [expandedItems, setExpandedItems] = useState>(new Set()); const [collapsedPriceCards, setCollapsedPriceCards] = useState>(new Set()); // 우측: 납품처 const [deliveryItems, setDeliveryItems] = useState([]); const [deliveryLoading, setDeliveryLoading] = useState(false); // 품목 편집 데이터 (더블클릭 시 상세 입력 모달 재활용) const [editItemData, setEditItemData] = useState(null); const savingRef = useRef(false); // 거래처 모달 const [supplierModalOpen, setSupplierModalOpen] = useState(false); const [supplierEditMode, setSupplierEditMode] = useState(false); const [supplierForm, setSupplierForm] = useState>({}); const [formErrors, setFormErrors] = useState>({}); const [saving, setSaving] = useState(false); // 품목 추가 모달 (1단계: 검색/선택) const [itemSelectOpen, setItemSelectOpen] = useState(false); const [itemSearchKeyword, setItemSearchKeyword] = useState(""); const [itemSearchResults, setItemSearchResults] = useState([]); const [itemTotalCount, setItemTotalCount] = useState(0); const [itemSearchLoading, setItemSearchLoading] = useState(false); const [itemCheckedIds, setItemCheckedIds] = useState>(new Set()); // 품목 상세 입력 모달 (2단계: 거래처 품번/품명 + 단가) const [itemDetailOpen, setItemDetailOpen] = useState(false); const [selectedItemsForDetail, setSelectedItemsForDetail] = useState([]); const [itemMappings, setItemMappings] = useState>>({}); const [itemPrices, setItemPrices] = useState>>({}); const [priceCategoryOptions, setPriceCategoryOptions] = useState>({}); // 거래처 모달 탭 const [supplierModalTab, setSupplierModalTab] = useState<"basic" | "contacts" | "delivery">("basic"); // 담당자 (supplier_contact) - 모달 내 const [modalContacts, setModalContacts] = useState([]); const [modalContactLoading, setModalContactLoading] = useState(false); const [modalContactForm, setModalContactForm] = useState>({}); const [modalContactEditId, setModalContactEditId] = useState(null); const [modalContactFormOpen, setModalContactFormOpen] = useState(false); const [modalContactSaving, setModalContactSaving] = useState(false); // 납품처 (delivery_destination) - 모달 내 const [modalDeliveries, setModalDeliveries] = useState([]); const [modalDeliveryLoading, setModalDeliveryLoading] = useState(false); const [modalDeliveryForm, setModalDeliveryForm] = useState>({}); const [modalDeliveryEditId, setModalDeliveryEditId] = useState(null); const [modalDeliveryFormOpen, setModalDeliveryFormOpen] = useState(false); const [modalDeliverySaving, setModalDeliverySaving] = useState(false); const [modalDeliveryFormErrors, setModalDeliveryFormErrors] = useState>({}); const [continuousInput, setContinuousInput] = useState(false); // 세금유형 (기본정보 탭 내) const [taxTypeRows, setTaxTypeRows] = useState<{ _id: string; tax_type_name: string; rate: string }[]>([]); const [taxTypeOptions, setTaxTypeOptions] = useState<{ code: string; label: string }[]>([]); // 엑셀 const [excelUploadOpen, setExcelUploadOpen] = useState(false); const [excelChainConfig, setExcelChainConfig] = useState(null); const [excelDetecting, setExcelDetecting] = useState(false); // 카테고리 const [categoryOptions, setCategoryOptions] = useState>({}); const [employeeOptions, setEmployeeOptions] = useState<{ user_id: string; user_name: string; position_name?: string }[]>([]); // 카테고리 로드 useEffect(() => { const flatten = (vals: any[]): { code: string; label: string }[] => { const result: { code: string; label: string }[] = []; for (const v of vals) { result.push({ code: v.valueCode, label: v.valueLabel }); if (v.children?.length) result.push(...flatten(v.children)); } return result; }; const load = async () => { const optMap: Record = {}; for (const col of ["division", "status"]) { try { const res = await apiClient.get(`/table-categories/${SUPPLIER_TABLE}/${col}/values`); if (res.data?.success) optMap[col] = flatten(res.data.data || []); } catch { /* skip */ } } for (const col of ["division", "unit", "material"]) { try { const res = await apiClient.get(`/table-categories/item_info/${col}/values`); if (res.data?.success) optMap[`item_${col}`] = flatten(res.data.data || []); } catch { /* skip */ } } setCategoryOptions(optMap); const priceOpts: Record = {}; for (const col of ["base_price_type", "currency_code", "discount_type", "rounding_type", "rounding_unit_value"]) { try { const res = await apiClient.get(`/table-categories/${PRICE_TABLE}/${col}/values`); if (res.data?.success) priceOpts[col] = flatten(res.data.data || []); } catch { /* skip */ } } setPriceCategoryOptions(priceOpts); // 세금유형 카테고리 try { const taxRes = await apiClient.get(`/table-categories/supplier_tax_type/tax_type_name/values`); if (taxRes.data?.success) setTaxTypeOptions(flatten(taxRes.data.data || [])); } catch { /* skip */ } }; load(); apiClient.post(`/table-management/tables/user_info/data`, { page: 1, size: 500, autoFilter: true }) .then((res) => { const users = res.data?.data?.data || res.data?.data?.rows || []; setEmployeeOptions(users.map((u: any) => ({ user_id: u.user_id, user_name: u.user_name || u.user_id, position_name: u.position_name, }))); }).catch(() => {}); }, []); // 공급업체 목록 조회 const fetchSuppliers = useCallback(async () => { setSupplierLoading(true); try { const filters = searchFilters.map(f => ({ columnName: f.columnName, operator: f.operator, value: f.value, })); const res = await apiClient.post(`/table-management/tables/${SUPPLIER_TABLE}/data`, { page: 1, size: 500, dataFilter: filters.length > 0 ? { enabled: true, filters } : undefined, autoFilter: true, sort: { columnName: "supplier_code", order: "desc" }, }); const raw = res.data?.data?.data || res.data?.data?.rows || []; setRawSuppliers(raw); const resolve = (col: string, code: string) => { if (!code) return ""; return categoryOptions[col]?.find((o) => o.code === code)?.label || code; }; const data = raw.map((r: any) => { const mainContact = mainContactMap[r.id]; return { ...r, division: resolve("division", r.division), status: resolve("status", r.status), contact_person: mainContact?.contact_name || "", contact_phone: mainContact?.contact_phone || "", email: mainContact?.contact_email || "", }; }); // 공급업체코드 숫자 기준 내림차순 정렬 data.sort((a: any, b: any) => { const aNum = parseInt((a.supplier_code || "").replace(/\D/g, ""), 10) || 0; const bNum = parseInt((b.supplier_code || "").replace(/\D/g, ""), 10) || 0; return bNum - aNum; }); setSuppliers(data); setSupplierCount(res.data?.data?.total || raw.length); } catch (err) { console.error("거래처 조회 실패:", err); toast.error("공급업체 목록을 불러오는데 실패했습니다."); } finally { setSupplierLoading(false); } }, [searchFilters, categoryOptions, employeeOptions, mainContactMap]); useEffect(() => { fetchSuppliers(); }, [fetchSuppliers]); // 메인 담당자 조회 (최초 1번 + 저장 후 갱신) const fetchMainContacts = useCallback(async () => { try { const contactRes = await apiClient.post(`/table-management/tables/${CONTACT_TABLE}/data`, { page: 1, size: 500, autoFilter: true, dataFilter: { enabled: true, filters: [{ columnName: "is_main", operator: "equals", value: "Y" }] }, }); const allContacts = contactRes.data?.data?.data || contactRes.data?.data?.rows || []; const map: Record = {}; for (const c of allContacts) { if ((c.is_main === "Y" || c.is_main === true) && c.supplier_id) { map[c.supplier_id] = c; } } setMainContactMap(map); } catch { /* skip */ } }, []); useEffect(() => { fetchMainContacts(); }, [fetchMainContacts]); const selectedSupplier = suppliers.find((c) => c.id === selectedSupplierId); // 선택된 공급업체의 품목 단가 조회 useEffect(() => { if (!selectedSupplier?.supplier_code) { setPriceItems([]); setPriceCheckedIds([]); return; } setPriceCheckedIds([]); const fetchItems = async () => { setPriceLoading(true); try { const mappingRes = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, { page: 1, size: 500, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier.supplier_code }, ]}, autoFilter: true, }); const mappings = mappingRes.data?.data?.data || mappingRes.data?.data?.rows || []; const itemIds = [...new Set(mappings.map((r: any) => r.item_id).filter(Boolean))]; let itemMap: Record = {}; if (itemIds.length > 0) { try { const itemRes = await apiClient.post(`/table-management/tables/item_info/data`, { page: 1, size: itemIds.length + 10, dataFilter: { enabled: true, filters: [{ columnName: "item_number", operator: "in", value: itemIds }] }, autoFilter: true, }); for (const item of (itemRes.data?.data?.data || itemRes.data?.data?.rows || [])) { itemMap[item.item_number] = item; } } catch { /* skip */ } } let allPrices: any[] = []; if (mappings.length > 0) { try { const priceRes = await apiClient.post(`/table-management/tables/${PRICE_TABLE}/data`, { page: 1, size: 500, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier.supplier_code }, ]}, autoFilter: true, }); allPrices = priceRes.data?.data?.data || priceRes.data?.data?.rows || []; } catch { /* skip */ } } const priceResolve = (col: string, code: string) => { if (!code) return ""; return priceCategoryOptions[col]?.find((o) => o.code === code)?.label || code; }; const today = new Date().toISOString().split("T")[0]; // 품목 기준 그룹핑 — master: 첫 매핑 + 현재 단가, details: 전체 단가 리스트 const grouped: Record = {}; const flatItems: any[] = []; const seenItemIds = new Set(); for (const m of mappings) { const itemKey = m.item_id || ""; if (seenItemIds.has(itemKey)) continue; // 품목당 첫 매핑만 마스터 seenItemIds.add(itemKey); const itemInfo = itemMap[itemKey] || {}; const itemPriceList = allPrices .filter((p: any) => p.item_id === itemKey) .sort((a: any, b: any) => (a.start_date || "").localeCompare(b.start_date || "")); const todayPrice = itemPriceList.find((p: any) => (!p.start_date || p.start_date <= today) && (!p.end_date || p.end_date >= today) ) || itemPriceList[0] || {}; const masterRow = { ...m, item_number: itemKey, item_name: itemInfo.item_name || "", base_price_type: priceResolve("base_price_type", todayPrice.base_price_type || ""), base_price: todayPrice.base_price || "", discount_type: priceResolve("discount_type", todayPrice.discount_type || ""), discount_value: todayPrice.discount_value || "", calculated_price: todayPrice.calculated_price || todayPrice.unit_price || "", currency_code: priceResolve("currency_code", todayPrice.currency_code || ""), }; // 단가 리스트 (라벨 변환) const priceDetails = itemPriceList.map((p: any) => ({ ...p, base_price_type_label: priceResolve("base_price_type", p.base_price_type || ""), discount_type_label: priceResolve("discount_type", p.discount_type || ""), currency_label: priceResolve("currency_code", p.currency_code || ""), is_current: (!p.start_date || p.start_date <= today) && (!p.end_date || p.end_date >= today), })); grouped[itemKey] = { master: masterRow, details: priceDetails }; flatItems.push(masterRow); } setPriceGroups(grouped); setPriceItems(flatItems); } catch (err) { console.error("품목 조회 실패:", err); } finally { setPriceLoading(false); } }; fetchItems(); }, [selectedSupplier?.supplier_code]); // 납품처 조회 useEffect(() => { if (!selectedSupplier?.supplier_code) { setDeliveryItems([]); return; } const fetchDelivery = async () => { setDeliveryLoading(true); try { const res = await apiClient.post(`/table-management/tables/${DELIVERY_TABLE}/data`, { page: 1, size: 500, dataFilter: { enabled: true, filters: [ { columnName: "supplier_code", operator: "equals", value: selectedSupplier.supplier_code }, ]}, autoFilter: true, }); setDeliveryItems(res.data?.data?.data || res.data?.data?.rows || []); } catch { setDeliveryItems([]); } finally { setDeliveryLoading(false); } }; fetchDelivery(); }, [selectedSupplier?.supplier_code]); const getCategoryLabel = (col: string, code: string) => { if (!code) return ""; return categoryOptions[col]?.find((o) => o.code === code)?.label || code; }; // 모달 내 담당자 목록 조회 const fetchModalContacts = useCallback(async (supplierId: string) => { setModalContactLoading(true); try { const res = await apiClient.post(`/table-management/tables/${CONTACT_TABLE}/data`, { page: 1, size: 200, dataFilter: { enabled: true, filters: [{ columnName: "supplier_id", operator: "equals", value: supplierId }] }, autoFilter: true, }); setModalContacts(res.data?.data?.data || res.data?.data?.rows || []); } catch { setModalContacts([]); } finally { setModalContactLoading(false); } }, []); // 모달 내 납품처 목록 조회 const fetchModalDeliveries = useCallback(async (supplierCode: string) => { setModalDeliveryLoading(true); try { const res = await apiClient.post(`/table-management/tables/${DELIVERY_TABLE}/data`, { page: 1, size: 200, dataFilter: { enabled: true, filters: [{ columnName: "supplier_code", operator: "equals", value: supplierCode }] }, autoFilter: true, }); setModalDeliveries(res.data?.data?.data || res.data?.data?.rows || []); } catch { setModalDeliveries([]); } finally { setModalDeliveryLoading(false); } }, []); // 담당자 저장 (등록/수정) const handleModalContactSave = async () => { if (!modalContactForm.contact_name) { toast.error("담당자명은 필수입니다."); return; } if (modalContactEditId) { // 수정 — 로컬 리스트에서 교체 + 메인 설정 시 다른 메인 해제 const isSettingMain = modalContactForm.is_main === "Y" || modalContactForm.is_main === true; setModalContacts((prev) => prev.map((c) => (c._localId || c.id) === modalContactEditId ? { ...c, ...modalContactForm } : isSettingMain ? { ...c, is_main: "N" } : c )); } else { // 추가 — 로컬 리스트에 카드 추가 setModalContacts((prev) => [...prev, { ...modalContactForm, _localId: `local_${Date.now()}_${Math.random()}`, _isNew: true, }]); } setModalContactFormOpen(false); setModalContactForm({}); setModalContactEditId(null); }; // 담당자 삭제 const handleModalContactDelete = (contactId: string) => { setModalContacts((prev) => prev.filter((c) => (c._localId || c.id) !== contactId)); }; // 납품처 자동채번 유틸 const generateDeliveryCode = async () => { try { const ruleRes = await apiClient.get(`/numbering-rules/by-column/${DELIVERY_TABLE}/destination_code`); const ruleData = ruleRes.data; if (ruleData?.success && ruleData?.data?.ruleId) { const ruleId = ruleData.data.ruleId; const allRes = await apiClient.post(`/table-management/tables/${DELIVERY_TABLE}/data`, { page: 1, size: 500, autoFilter: true, sort: { columnName: "destination_code", order: "desc" }, }); const allRows = allRes.data?.data?.data || allRes.data?.data?.rows || []; let maxSeq = 0; for (const row of allRows) { const match = (row.destination_code || "").match(/(\d+)$/); if (match) { const seq = parseInt(match[1], 10); if (seq > maxSeq) maxSeq = seq; } } // 로컬에 추가된 것도 포함 for (const d of modalDeliveries) { const match = (d.destination_code || "").match(/(\d+)$/); if (match) { const seq = parseInt(match[1], 10); if (seq > maxSeq) maxSeq = seq; } } const previewRes = await previewNumberingCode(ruleId); if (previewRes.success && previewRes.data?.generatedCode) { const previewCode = previewRes.data.generatedCode; const prefix = previewCode.replace(/\d+$/, ""); const seqLen = (previewCode.match(/(\d+)$/) || ["", "001"])[1].length; return prefix + String(maxSeq + 1).padStart(seqLen, "0"); } } } catch { /* skip */ } return ""; }; // 납품처 저장 (모달 내) const handleModalDeliverySave = async () => { if (!modalDeliveryForm.destination_name) { toast.error("납품처명은 필수입니다."); return; } if (modalDeliveryEditId) { const isSettingMain = modalDeliveryForm.is_default === "Y" || modalDeliveryForm.is_default === true; setModalDeliveries((prev) => prev.map((d) => (d._localId || d.id) === modalDeliveryEditId ? { ...d, ...modalDeliveryForm } : isSettingMain ? { ...d, is_default: "N" } : d )); } else { setModalDeliveries((prev) => [...prev, { ...modalDeliveryForm, _localId: `local_${Date.now()}_${Math.random()}`, _isNew: true, }]); } setModalDeliveryFormOpen(false); setModalDeliveryForm({}); setModalDeliveryEditId(null); setModalDeliveryFormErrors({}); }; const handleModalDeliveryDelete = (deliveryId: string) => { setModalDeliveries((prev) => prev.filter((d) => (d._localId || d.id) !== deliveryId)); }; // 공급업체 등록 모달 열기 const openSupplierRegister = async () => { setSupplierForm({}); setFormErrors({}); setSupplierEditMode(false); setSupplierModalTab("basic"); setModalContacts([]); setModalDeliveries([]); setModalContactFormOpen(false); setModalDeliveryFormOpen(false); setTaxTypeRows([]); setSupplierModalOpen(true); // 공급업체 코드 자동 채번 — 기존 데이터 max값 기반 try { const ruleRes = await apiClient.get(`/numbering-rules/by-column/${SUPPLIER_TABLE}/supplier_code`); const ruleData = ruleRes.data; if (ruleData?.success && ruleData?.data?.ruleId) { const ruleId = ruleData.data.ruleId; // 기존 데이터에서 CUST-XXX 패턴의 최대 순번 조회 const allRes = await apiClient.post(`/table-management/tables/${SUPPLIER_TABLE}/data`, { page: 1, size: 500, autoFilter: true, sort: { columnName: "supplier_code", order: "desc" }, }); const allRows = allRes.data?.data?.data || allRes.data?.data?.rows || []; let maxSeq = 0; for (const row of allRows) { const code = row.supplier_code || ""; const match = code.match(/(\d+)$/); if (match) { const seq = parseInt(match[1], 10); if (seq > maxSeq) maxSeq = seq; } } // preview로 접두어 패턴 가져오기 const previewRes = await previewNumberingCode(ruleId); if (previewRes.success && previewRes.data?.generatedCode) { const previewCode = previewRes.data.generatedCode; const prefix = previewCode.replace(/\d+$/, ""); const seqLength = (previewCode.match(/(\d+)$/) || ["", "001"])[1].length; const nextSeq = maxSeq + 1; const nextCode = prefix + String(nextSeq).padStart(seqLength, "0"); setSupplierForm((prev) => ({ ...prev, supplier_code: nextCode, _numberingRuleId: ruleId })); } } } catch { /* skip */ } }; const openSupplierEdit = () => { if (!selectedSupplier) return; const rawData = rawSuppliers.find((c) => c.id === selectedSupplierId); setSupplierForm({ ...(rawData || selectedSupplier) }); setFormErrors({}); setSupplierEditMode(true); setSupplierModalTab("basic"); setModalContactFormOpen(false); setModalDeliveryFormOpen(false); setModalContactForm({}); setModalDeliveryForm({}); setModalContactEditId(null); setModalDeliveryEditId(null); // 수정 모드에서는 바로 조회 const code = (rawData || selectedSupplier).supplier_code; const id = (rawData || selectedSupplier).id; if (id) { fetchModalContacts(id); // 세금유형 로드 apiClient.post(`/table-management/tables/supplier_tax_type/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [{ columnName: "supplier_id", operator: "equals", value: id }] }, autoFilter: true, }).then((res: any) => { const rows = res.data?.data?.data || res.data?.data?.rows || []; setTaxTypeRows(rows.map((r: any) => ({ _id: r.id, tax_type_name: r.tax_type_name || "", rate: String(r.rate || "") }))); }).catch(() => setTaxTypeRows([])); } if (code) fetchModalDeliveries(code); setSupplierModalOpen(true); }; // 폼 필드 변경 시 자동 포맷팅 + 실시간 검증 const handleFormChange = (field: string, value: string) => { const formatted = formatField(field, value); setSupplierForm((prev) => ({ ...prev, [field]: formatted })); const error = validateField(field, formatted); setFormErrors((prev) => { const next = { ...prev }; if (error) next[field] = error; else delete next[field]; return next; }); }; // 세금유형/담당자/납품처를 한번에 저장하는 헬퍼 const saveSubTables = async (supplierId: string, supplierCode: string) => { // 세금유형 — 기존 삭제 후 재생성 try { const existTax = await apiClient.post(`/table-management/tables/supplier_tax_type/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [{ columnName: "supplier_id", operator: "equals", value: supplierId }] }, autoFilter: true, }); const existRows = existTax.data?.data?.data || existTax.data?.data?.rows || []; if (existRows.length > 0) { await apiClient.delete(`/table-management/tables/supplier_tax_type/delete`, { data: existRows.map((r: any) => ({ id: r.id })), }); } for (const t of taxTypeRows.filter((r) => r.tax_type_name)) { await apiClient.post(`/table-management/tables/supplier_tax_type/add`, { id: crypto.randomUUID(), supplier_id: supplierId, tax_type_name: t.tax_type_name, tax_type_id: t.tax_type_name, rate: t.rate ? Number(t.rate) : 0, }); } } catch { /* skip */ } // 담당자 — 기존 삭제 후 전체 재생성 try { const existContacts = await apiClient.post(`/table-management/tables/${CONTACT_TABLE}/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [{ columnName: "supplier_id", operator: "equals", value: supplierId }] }, autoFilter: true, }); const existCRows = existContacts.data?.data?.data || existContacts.data?.data?.rows || []; if (existCRows.length > 0) { await apiClient.delete(`/table-management/tables/${CONTACT_TABLE}/delete`, { data: existCRows.map((r: any) => ({ id: r.id })), }); } } catch { /* skip */ } for (const c of modalContacts) { try { await apiClient.post(`/table-management/tables/${CONTACT_TABLE}/add`, { id: crypto.randomUUID(), supplier_id: supplierId, contact_name: c.contact_name || "", contact_phone: c.contact_phone || "", contact_email: c.contact_email || "", department: c.department || "", is_main: c.is_main || "N", memo: c.memo || "", }); } catch { /* skip */ } } // 납품처 — 기존 삭제 후 전체 재생성 try { const existDeliveries = await apiClient.post(`/table-management/tables/${DELIVERY_TABLE}/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [{ columnName: "supplier_code", operator: "equals", value: supplierCode }] }, autoFilter: true, }); const existDRows = existDeliveries.data?.data?.data || existDeliveries.data?.data?.rows || []; if (existDRows.length > 0) { await apiClient.delete(`/table-management/tables/${DELIVERY_TABLE}/delete`, { data: existDRows.map((r: any) => ({ id: r.id })), }); } } catch { /* skip */ } for (const d of modalDeliveries) { try { await apiClient.post(`/table-management/tables/${DELIVERY_TABLE}/add`, { id: crypto.randomUUID(), supplier_code: supplierCode, destination_code: d.destination_code || "", destination_name: d.destination_name || "", address: d.address || "", manager_name: d.manager_name || "", phone: d.phone || "", memo: d.memo || "", is_default: d.is_default || "N", }); } catch { /* skip */ } } }; const handleSupplierSave = async () => { if (!supplierForm.supplier_name) { toast.error("공급업체명은 필수입니다."); return; } if (!supplierForm.status) { toast.error("상태는 필수입니다."); return; } const errors = validateForm(supplierForm, ["contact_phone", "email", "business_number"]); setFormErrors(errors); if (Object.keys(errors).length > 0) { toast.error("입력 형식을 확인해주세요."); return; } setSaving(true); try { const { id, created_date, updated_date, writer, company_code, _numberingRuleId, ...fields } = supplierForm; const cleanFields: Record = {}; for (const [key, value] of Object.entries(fields)) { cleanFields[key] = value === "" ? null : value; } if (supplierEditMode && id) { // 수정 await apiClient.put(`/table-management/tables/${SUPPLIER_TABLE}/edit`, { originalData: { id }, updatedData: cleanFields, }); await saveSubTables(id, cleanFields.supplier_code || supplierForm.supplier_code); toast.success("저장되었습니다."); } else { // 신규 등록 await apiClient.post(`/table-management/tables/${SUPPLIER_TABLE}/add`, cleanFields); // id 획득 const res = await apiClient.post(`/table-management/tables/${SUPPLIER_TABLE}/data`, { page: 1, size: 1, dataFilter: { enabled: true, filters: [{ columnName: "supplier_code", operator: "equals", value: cleanFields.supplier_code }] }, autoFilter: true, }); const newRow = (res.data?.data?.data || res.data?.data?.rows || [])[0]; if (newRow?.id) { await saveSubTables(newRow.id, cleanFields.supplier_code); } toast.success("공급업체가 등록되었습니다."); } fetchSuppliers(); fetchMainContacts(); if (!supplierEditMode && continuousInput) { // 연속입력 — 폼 초기화하고 모달 유지 setSupplierForm({}); setModalContacts([]); setModalDeliveries([]); setTaxTypeRows([]); setSupplierModalTab("basic"); // 새 코드 채번 try { const ruleRes = await apiClient.get(`/numbering-rules/by-column/${SUPPLIER_TABLE}/supplier_code`); const ruleData = ruleRes.data; if (ruleData?.success && ruleData?.data?.ruleId) { const ruleId = ruleData.data.ruleId; const allRes = await apiClient.post(`/table-management/tables/${SUPPLIER_TABLE}/data`, { page: 1, size: 500, autoFilter: true, sort: { columnName: "supplier_code", order: "desc" } }); const allRows = allRes.data?.data?.data || allRes.data?.data?.rows || []; let maxSeq = 0; for (const row of allRows) { const match = (row.supplier_code || "").match(/(\d+)$/); if (match) { const seq = parseInt(match[1], 10); if (seq > maxSeq) maxSeq = seq; } } const previewRes = await previewNumberingCode(ruleId); if (previewRes.success && previewRes.data?.generatedCode) { const prefix = previewRes.data.generatedCode.replace(/\d+$/, ""); const seqLen = (previewRes.data.generatedCode.match(/(\d+)$/) || ["", "001"])[1].length; setSupplierForm({ supplier_code: prefix + String(maxSeq + 1).padStart(seqLen, "0") }); } } } catch { /* skip */ } toast.success("등록 완료. 다음 공급업체를 입력하세요."); } else { setSupplierModalOpen(false); // 우측 패널 갱신 if (selectedSupplierId) { const cid = selectedSupplierId; setSelectedSupplierId(null); setTimeout(() => setSelectedSupplierId(cid), 50); } } } catch (err: any) { toast.error(err.response?.data?.message || "저장에 실패했습니다."); } finally { setSaving(false); } }; // 공급업체 삭제 const handleSupplierDelete = async () => { if (!selectedSupplierId) return; const ok = await confirm("공급업체를 삭제하시겠습니까?", { variant: "destructive", confirmText: "삭제", }); if (!ok) return; try { await apiClient.delete(`/table-management/tables/${SUPPLIER_TABLE}/delete`, { data: [{ id: selectedSupplierId }], }); toast.success("삭제되었습니다."); setSelectedSupplierId(null); fetchSuppliers(); } catch { toast.error("삭제에 실패했습니다."); } }; // 품목 검색 const searchItems = useCallback(async () => { setItemSearchLoading(true); try { const filters: any[] = []; if (itemSearchKeyword) filters.push({ columnName: "item_name", operator: "contains", value: itemSearchKeyword }); const res = await apiClient.post(`/table-management/tables/item_info/data`, { page: 1, size: 5000, dataFilter: filters.length > 0 ? { enabled: true, filters } : undefined, autoFilter: true, }); const allItems = res.data?.data?.data || res.data?.data?.rows || []; setItemTotalCount(allItems.length); const existingItemIds = new Set(priceItems.map((p: any) => p.item_id || p.item_number)); const purchaseCode = categoryOptions["item_division"]?.find((o) => o.label === "구매관리")?.code; setItemSearchResults(allItems.filter((item: any) => { if (existingItemIds.has(item.item_number) || existingItemIds.has(item.id)) return false; if (!purchaseCode) return true; const div = item.division || ""; return div.includes(purchaseCode); })); } catch { /* skip */ } finally { setItemSearchLoading(false); } }, [itemSearchKeyword, priceItems]); // 실��간 검색 (2글자 이상) useEffect(() => { if (!itemSelectOpen) return; if (itemSearchKeyword.length > 0 && itemSearchKeyword.length < 2) return; searchItems(); }, [itemSearchKeyword, itemSelectOpen]); // eslint-disable-line react-hooks/exhaustive-deps // 품목 선택 완료 → 상세 입력 모달로 전환 const goToItemDetail = () => { const selected = itemSearchResults.filter((i) => itemCheckedIds.has(i.id)); if (selected.length === 0) { toast.error("품목을 선택해주세요."); return; } setSelectedItemsForDetail(selected); const mappings: typeof itemMappings = {}; const prices: typeof itemPrices = {}; for (const item of selected) { const key = item.item_number || item.id; mappings[key] = []; prices[key] = [{ _id: `p_${Date.now()}_${Math.random()}`, start_date: "", end_date: "", currency_code: "CAT_MLAMDKVN_PZJI", base_price_type: "CAT_MLAMFGFT_4RZW", base_price: item.standard_price || item.selling_price || "", discount_type: "", discount_value: "", rounding_type: "", rounding_unit_value: "", calculated_price: item.standard_price || item.selling_price || "", }]; } setItemMappings(mappings); setItemPrices(prices); setItemSelectOpen(false); setItemDetailOpen(true); }; // 거래처 품번/품명 행 추가 const addMappingRow = (itemKey: string) => { setItemMappings((prev) => ({ ...prev, [itemKey]: [...(prev[itemKey] || []), { _id: `m_${Date.now()}_${Math.random()}`, supplier_item_code: "", supplier_item_name: "" }], })); }; const removeMappingRow = (itemKey: string, rowId: string) => { setItemMappings((prev) => ({ ...prev, [itemKey]: (prev[itemKey] || []).filter((r) => r._id !== rowId), })); }; const handleMappingDragEnd = (itemKey: string, event: DragEndEvent) => { const { active, over } = event; if (!over || active.id === over.id) return; setItemMappings((prev) => { const arr = [...(prev[itemKey] || [])]; const oldIdx = arr.findIndex((r) => r._id === active.id); const newIdx = arr.findIndex((r) => r._id === over.id); return { ...prev, [itemKey]: arrayMove(arr, oldIdx, newIdx) }; }); }; const updateMappingRow = (itemKey: string, rowId: string, field: string, value: string) => { setItemMappings((prev) => ({ ...prev, [itemKey]: (prev[itemKey] || []).map((r) => r._id === rowId ? { ...r, [field]: value } : r), })); }; // 단가 행 추가 const addPriceRow = (itemKey: string) => { setItemPrices((prev) => ({ ...prev, [itemKey]: [...(prev[itemKey] || []), { _id: `p_${Date.now()}_${Math.random()}`, start_date: "", end_date: "", currency_code: "CAT_MLAMDKVN_PZJI", base_price_type: "CAT_MLAMFGFT_4RZW", base_price: "", discount_type: "", discount_value: "", rounding_type: "", rounding_unit_value: "", calculated_price: "", }], })); }; const removePriceRow = (itemKey: string, rowId: string) => { setItemPrices((prev) => ({ ...prev, [itemKey]: (prev[itemKey] || []).filter((r) => r._id !== rowId), })); }; const updatePriceRow = (itemKey: string, rowId: string, field: string, value: string) => { setItemPrices((prev) => ({ ...prev, [itemKey]: (prev[itemKey] || []).map((r) => { if (r._id !== rowId) return r; const updated = { ...r, [field]: value }; if (["base_price", "discount_type", "discount_value", "rounding_unit_value", "rounding_type"].includes(field)) { const bp = Number(updated.base_price) || 0; const dv = Number(updated.discount_value) || 0; const dt = updated.discount_type; let calc = bp; if (dt === "CAT_MLAMBEC8_URQA") calc = bp * (1 - dv / 100); else if (dt === "CAT_MLAMBLFM_JTLO") calc = bp - dv; // 반올림 유형 + 단위 적용 const rv = updated.rounding_unit_value; const rt = updated.rounding_type; const roundOpts = priceCategoryOptions["rounding_unit_value"] || []; const roundLabel = roundOpts.find((o) => o.code === rv)?.label || ""; const unitOpts = priceCategoryOptions["rounding_type"] || []; const unitLabel = unitOpts.find((o) => o.code === rt)?.label || ""; const unit = parseInt(unitLabel) || 1; // "10원" → 10, "100원" → 100 if (roundLabel === "반올림") { calc = Math.round(calc / unit) * unit; } else if (roundLabel === "절삭") { calc = Math.floor(calc / unit) * unit; } else if (roundLabel === "올림") { calc = Math.ceil(calc / unit) * unit; } updated.calculated_price = String(Math.floor(calc)); } return updated; }), })); }; // 품목 편집 열기 const openEditItem = async (row: any) => { const itemKey = row.item_number || row.item_id; let itemInfo: any = { item_number: itemKey, item_name: row.item_name || "", size: "", unit: "" }; try { const res = await apiClient.post(`/table-management/tables/item_info/data`, { page: 1, size: 1, dataFilter: { enabled: true, filters: [{ columnName: "item_number", operator: "equals", value: itemKey }] }, autoFilter: true, }); const found = (res.data?.data?.data || res.data?.data?.rows || [])[0]; if (found) itemInfo = found; } catch { /* skip */ } let mappingRows: any[] = []; try { const mapRes = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier!.supplier_code }, { columnName: "item_id", operator: "equals", value: itemKey }, ]}, autoFilter: true, sort: { columnName: "created_date", order: "asc" }, }); const allMappings = mapRes.data?.data?.data || mapRes.data?.data?.rows || []; mappingRows = allMappings .filter((m: any) => m.supplier_item_code || m.supplier_item_name) .map((m: any) => ({ _id: `m_existing_${m.id}`, supplier_item_code: m.supplier_item_code || "", supplier_item_name: m.supplier_item_name || "", })); } catch { /* skip */ } let priceRows: any[] = []; try { const priceRes = await apiClient.post(`/table-management/tables/${PRICE_TABLE}/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier!.supplier_code }, { columnName: "item_id", operator: "equals", value: itemKey }, ]}, autoFilter: true, }); const allPriceData = (priceRes.data?.data?.data || priceRes.data?.data?.rows || []) .sort((a: any, b: any) => (a.start_date || "").localeCompare(b.start_date || "")); priceRows = allPriceData.map((p: any) => ({ _id: `p_existing_${p.id}`, start_date: p.start_date ? String(p.start_date).split("T")[0] : "", end_date: p.end_date ? String(p.end_date).split("T")[0] : "", currency_code: p.currency_code || "CAT_MLAMDKVN_PZJI", base_price_type: p.base_price_type || "CAT_MLAMFGFT_4RZW", base_price: p.base_price ? String(p.base_price) : "", discount_type: p.discount_type || "", discount_value: p.discount_value ? String(p.discount_value) : "", rounding_type: p.rounding_type || "", rounding_unit_value: p.rounding_unit_value || "", calculated_price: p.calculated_price ? String(p.calculated_price) : "", })); } catch { /* skip */ } if (priceRows.length === 0) { priceRows.push({ _id: `p_${Date.now()}`, start_date: "", end_date: "", currency_code: "CAT_MLAMDKVN_PZJI", base_price_type: "CAT_MLAMFGFT_4RZW", base_price: "", discount_type: "", discount_value: "", rounding_type: "", rounding_unit_value: "", calculated_price: "", }); } setSelectedItemsForDetail([itemInfo]); setItemMappings({ [itemKey]: mappingRows }); setItemPrices({ [itemKey]: priceRows }); setEditItemData(row); setItemDetailOpen(true); }; const handleItemDetailSave = async () => { if (!selectedSupplier) return; if (savingRef.current) return; savingRef.current = true; const isEditingExisting = !!editItemData; setSaving(true); try { for (const item of selectedItemsForDetail) { const itemKey = item.item_number || item.id; const mappingRows = itemMappings[itemKey] || []; if (isEditingExisting && editItemData?.id) { // 기존 매핑 조회 let existingMaps: any[] = []; try { const existingMappings = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier.supplier_code }, { columnName: "item_id", operator: "equals", value: itemKey }, ]}, autoFilter: true, sort: { columnName: "created_date", order: "asc" }, }); existingMaps = existingMappings.data?.data?.data || existingMappings.data?.data?.rows || []; } catch { /* skip */ } // 매핑 upsert: 기존 것은 update, 새 것은 insert, 남은 것은 delete const usedExistingIds = new Set(); let firstMappingId: string | null = editItemData.id; for (let mi = 0; mi < mappingRows.length; mi++) { const existMap = existingMaps[mi]; if (existMap) { // update await apiClient.put(`/table-management/tables/${MAPPING_TABLE}/edit`, { originalData: { id: existMap.id }, updatedData: { supplier_item_code: mappingRows[mi].supplier_item_code || "", supplier_item_name: mappingRows[mi].supplier_item_name || "", }, }); usedExistingIds.add(existMap.id); if (mi === 0) firstMappingId = existMap.id; } else { // insert const mRes = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/add`, { id: crypto.randomUUID(), supplier_id: selectedSupplier.supplier_code, item_id: itemKey, supplier_item_code: mappingRows[mi].supplier_item_code || "", supplier_item_name: mappingRows[mi].supplier_item_name || "", }); if (mi === 0 && !firstMappingId) firstMappingId = mRes.data?.data?.id || null; } } // 초과분 delete const toDeleteMaps = existingMaps.filter((m) => !usedExistingIds.has(m.id)); if (toDeleteMaps.length > 0) { await apiClient.delete(`/table-management/tables/${MAPPING_TABLE}/delete`, { data: toDeleteMaps.map((m: any) => ({ id: m.id })), }); } // 기존 단가 조회 let existingPriceRows: any[] = []; try { const existingPrices = await apiClient.post(`/table-management/tables/${PRICE_TABLE}/data`, { page: 1, size: 100, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier.supplier_code }, { columnName: "item_id", operator: "equals", value: itemKey }, ]}, autoFilter: true, }); existingPriceRows = (existingPrices.data?.data?.data || existingPrices.data?.data?.rows || []) .sort((a: any, b: any) => (a.start_date || "").localeCompare(b.start_date || "")); } catch { /* skip */ } // 단가 upsert const priceRows = (itemPrices[itemKey] || []).filter((p) => p.base_price || p.start_date || p.currency_code || p.base_price_type ); const usedPriceIds = new Set(); for (let pi = 0; pi < priceRows.length; pi++) { const price = priceRows[pi]; const priceData = { mapping_id: firstMappingId || editItemData.id, supplier_id: selectedSupplier.supplier_code, item_id: itemKey, start_date: price.start_date || null, end_date: price.end_date || null, currency_code: price.currency_code || null, base_price_type: price.base_price_type || null, base_price: price.base_price ? Number(price.base_price) : null, unit_price: price.calculated_price ? Number(price.calculated_price) : (price.base_price ? Number(price.base_price) : null), discount_type: price.discount_type || null, discount_value: price.discount_value ? Number(price.discount_value) : null, rounding_type: price.rounding_type || null, rounding_unit_value: price.rounding_unit_value || null, calculated_price: price.calculated_price ? Number(price.calculated_price) : null, }; const existPrice = existingPriceRows[pi]; if (existPrice) { await apiClient.put(`/table-management/tables/${PRICE_TABLE}/edit`, { originalData: { id: existPrice.id }, updatedData: priceData, }); usedPriceIds.add(existPrice.id); } else { await apiClient.post(`/table-management/tables/${PRICE_TABLE}/add`, { id: crypto.randomUUID(), ...priceData, }); } } // 초과분 delete const toDeletePrices = existingPriceRows.filter((p) => !usedPriceIds.has(p.id)); if (toDeletePrices.length > 0) { await apiClient.delete(`/table-management/tables/${PRICE_TABLE}/delete`, { data: toDeletePrices.map((p: any) => ({ id: p.id })), }); } } else { if (!mappingRows.length || !mappingRows[0]?.supplier_item_code) { const existingCheck = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, { page: 1, size: 1, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier.supplier_code }, { columnName: "item_id", operator: "equals", value: itemKey }, ]}, autoFilter: true, }); if ((existingCheck.data?.data?.data || existingCheck.data?.data?.rows || []).length > 0) { toast.warning(`${item.item_name || itemKey} 품목은 이미 등록되어 있습니다.`); continue; } } let mappingId: string | null = null; const mappingRes = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/add`, { id: crypto.randomUUID(), supplier_id: selectedSupplier.supplier_code, item_id: itemKey, supplier_item_code: mappingRows[0]?.supplier_item_code || "", supplier_item_name: mappingRows[0]?.supplier_item_name || "", }); mappingId = mappingRes.data?.data?.id || null; for (let mi = 1; mi < mappingRows.length; mi++) { await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/add`, { id: crypto.randomUUID(), supplier_id: selectedSupplier.supplier_code, item_id: itemKey, supplier_item_code: mappingRows[mi].supplier_item_code || "", supplier_item_name: mappingRows[mi].supplier_item_name || "", }); } const priceRows = (itemPrices[itemKey] || []).filter((p) => p.base_price || p.start_date || p.currency_code || p.base_price_type ); for (const price of priceRows) { await apiClient.post(`/table-management/tables/${PRICE_TABLE}/add`, { id: crypto.randomUUID(), mapping_id: mappingId || "", supplier_id: selectedSupplier.supplier_code, item_id: itemKey, start_date: price.start_date || null, end_date: price.end_date || null, currency_code: price.currency_code || null, base_price_type: price.base_price_type || null, base_price: price.base_price ? Number(price.base_price) : null, unit_price: price.calculated_price ? Number(price.calculated_price) : (price.base_price ? Number(price.base_price) : null), discount_type: price.discount_type || null, discount_value: price.discount_value ? Number(price.discount_value) : null, rounding_type: price.rounding_type || null, rounding_unit_value: price.rounding_unit_value || null, calculated_price: price.calculated_price ? Number(price.calculated_price) : null, }); } } } toast.success(isEditingExisting ? "수정되었습니다." : `${selectedItemsForDetail.length}개 품목이 추가되었습니다.`); setItemDetailOpen(false); setEditItemData(null); setItemCheckedIds(new Set()); const cid = selectedSupplierId; setSelectedSupplierId(null); setTimeout(() => setSelectedSupplierId(cid), 50); } catch (err: any) { toast.error(err.response?.data?.message || "저장에 실패했습니다."); } finally { setSaving(false); savingRef.current = false; } }; // 품목 매핑 해제 (소프트 삭제 — supplier_id를 null 처리) const handlePriceItemDelete = async () => { if (priceCheckedIds.length === 0) return; const ok = await confirm(`선택한 ${priceCheckedIds.length}개 품목의 연결을 해제하시겠습니까?`, { description: "해당 품목의 공급업체 연결이 해제됩니다. (데이터는 유지)", variant: "destructive", confirmText: "해제", }); if (!ok) return; try { const itemIds = priceCheckedIds.map((mid) => { const group = Object.values(priceGroups).find((g) => g.master.id === mid); return group?.master.item_id || group?.master.item_number || ""; }).filter(Boolean); for (const itemId of itemIds) { // 해당 품목의 모든 매핑 조회 → supplier_id null 처리 const mapRes = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, { page: 1, size: 500, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier!.supplier_code }, { columnName: "item_id", operator: "equals", value: itemId }, ]}, autoFilter: true, }); const allMappings = mapRes.data?.data?.data || mapRes.data?.data?.rows || []; for (const m of allMappings) { await apiClient.put(`/table-management/tables/${MAPPING_TABLE}/edit`, { originalData: { id: m.id }, updatedData: { supplier_id: null }, }); } // 해당 품목의 모든 단가 조회 → supplier_id null 처리 try { const priceRes = await apiClient.post(`/table-management/tables/${PRICE_TABLE}/data`, { page: 1, size: 500, dataFilter: { enabled: true, filters: [ { columnName: "supplier_id", operator: "equals", value: selectedSupplier!.supplier_code }, { columnName: "item_id", operator: "equals", value: itemId }, ]}, autoFilter: true, }); const prices = priceRes.data?.data?.data || priceRes.data?.data?.rows || []; for (const p of prices) { await apiClient.put(`/table-management/tables/${PRICE_TABLE}/edit`, { originalData: { id: p.id }, updatedData: { supplier_id: null }, }); } } catch { /* skip */ } } toast.success(`${priceCheckedIds.length}개 품목의 연결이 해제되었습니다.`); setPriceCheckedIds([]); const cid = selectedSupplierId; setSelectedSupplierId(null); setTimeout(() => setSelectedSupplierId(cid), 50); } catch { toast.error("연결 해제에 실패했습니다."); } }; // EDataTable 컬럼 정의 (공급업체 목록) — ts.visibleColumns 순서를 따름 const supplierColumns: EDataTableColumn[] = useMemo(() => { const colProps: Record> = { supplier_code: { width: "w-[120px]" }, supplier_name: { minWidth: "min-w-[140px]" }, division: { width: "w-[80px]", render: (val: any) => val ? ( {val} ) : null, }, contact_person: { width: "w-[80px]" }, contact_phone: { width: "w-[120px]" }, email: { width: "w-[160px]" }, business_number: { width: "w-[120px]" }, address: { minWidth: "min-w-[150px]" }, status: { width: "w-[70px]", render: (val: any) => val ? ( {val} ) : null, }, }; return ts.visibleColumns.map((col) => ({ key: col.key, label: col.label, ...colProps[col.key], })); }, [ts.visibleColumns]); // 엑셀 다운로드 const handleExcelDownload = async () => { if (suppliers.length === 0) return; toast.loading("엑셀 데이터 준비 중...", { id: "excel-dl" }); try { const allMappings: any[] = []; const mapRes = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, { page: 1, size: 5000, autoFilter: true, }); const mappings = mapRes.data?.data?.data || mapRes.data?.data?.rows || []; const itemIds = [...new Set(mappings.map((m: any) => m.item_id).filter(Boolean))]; let itemMap: Record = {}; if (itemIds.length > 0) { try { const itemRes = await apiClient.post(`/table-management/tables/item_info/data`, { page: 1, size: itemIds.length + 10, dataFilter: { enabled: true, filters: [{ columnName: "item_number", operator: "in", value: itemIds }] }, autoFilter: true, }); for (const item of (itemRes.data?.data?.data || itemRes.data?.data?.rows || [])) { itemMap[item.item_number] = item; } } catch { /* skip */ } } for (const m of mappings) { const itemInfo = itemMap[m.item_id] || {}; allMappings.push({ ...m, item_name: itemInfo.item_name || "", item_spec: itemInfo.size || "" }); } const rows: Record[] = []; for (const c of suppliers) { const suppMappings = allMappings.filter((m) => m.supplier_id === c.supplier_code); if (suppMappings.length === 0) { rows.push({ 공급업체코드: c.supplier_code, 공급업체명: c.supplier_name, 공급업체유형: getCategoryLabel("division", c.division), 담당자: c.contact_person, 전화번호: c.contact_phone, 사업자번호: c.business_number, 이메일: c.email, 상태: getCategoryLabel("status", c.status), 품목코드: "", 품명: "", 규격: "", 공급업체품번: "", 공급업체품명: "", 기준가: "", 할인유형: "", 할인값: "", 단가: "", 통화: "", }); } else { for (const m of suppMappings) { rows.push({ 공급업체코드: c.supplier_code, 공급업체명: c.supplier_name, 공급업체유형: getCategoryLabel("division", c.division), 담당자: c.contact_person, 전화번호: c.contact_phone, 사업자번호: c.business_number, 이메일: c.email, 상태: getCategoryLabel("status", c.status), 품목코드: m.item_id || "", 품명: m.item_name || "", 규격: m.item_spec || "", 공급업체품번: m.supplier_item_code || "", 공급업체품명: m.supplier_item_name || "", 기준가: m.base_price || "", 할인유형: m.discount_type || "", 할인값: m.discount_value || "", 단가: m.calculated_price || "", 통화: m.currency_code || "", }); } } } await exportToExcel(rows, "공급업체관리.xlsx", "거래처+품목"); toast.dismiss("excel-dl"); toast.success(`${rows.length}행 다운로드 완료`); } catch (err) { toast.dismiss("excel-dl"); toast.error("다운로드에 실패했습니다."); } }; return (
{/* 검색 필터 (DynamicSearchFilter) */} {/* 액션 버튼 영역 */}
{/* 마스터-디테일 분할 패널 */}
{/* 좌측: 공급업체 목록 */}
{/* 패널 헤더 */}
공급업체 목록 {supplierCount}건
{/* 거래처 테이블 */} c.status !== "거래정지"))} rowKey={(row) => row.id} loading={supplierLoading} emptyMessage="등록된 공급업체가 없어요" selectedId={selectedSupplierId} onSelect={(id) => setSelectedSupplierId(id)} onRowDoubleClick={(row) => { setSelectedSupplierId(row.id); openSupplierEdit(); }} showRowNumber showPagination defaultPageSize={20} draggableColumns={false} columnOrderKey="c16-supplier" />
{/* 우측: 디테일 패널 */}
{!selectedSupplierId ? ( /* 빈 상태 */
공급업체를 선택해주세요
좌측에서 공급업체를 선택하면 상세 정보가 표시돼요
) : ( <> {/* 탭 + 버튼 통합 헤더 */} setRightTab(v as "items" | "delivery")} className="flex flex-col flex-1 overflow-hidden gap-0" >
공급업체별 품목정보 {Object.keys(priceGroups).length > 0 && ( {Object.keys(priceGroups).length} )} 납품처 정보 {deliveryItems.length > 0 && ( {deliveryItems.length} )}
{rightTab === "items" ? ( <> ) : ( )}
{/* 품목정보 탭 */}
0 && priceCheckedIds.length === priceItems.length} onChange={(e) => setPriceCheckedIds(e.target.checked ? priceItems.map((p) => p.id) : [])} /> 품목코드 품명 공급업체품번 공급업체품명 기준유형 기준가 할인유형 할인값 단가 통화 {priceLoading ? ( ) : Object.keys(priceGroups).length === 0 ? ( 등록된 품목이 없어요 ) : Object.entries(priceGroups).map(([itemKey, group]) => { const isExpanded = expandedItems.has(itemKey); const m = group.master; const isChecked = priceCheckedIds.includes(m.id); return ( {/* 마스터 행 */} { setExpandedItems((prev) => { const next = new Set(prev); if (next.has(itemKey)) next.delete(itemKey); else next.add(itemKey); return next; }); }} onDoubleClick={() => openEditItem(m)} > { e.stopPropagation(); setPriceCheckedIds((prev) => prev.includes(m.id) ? prev.filter((id) => id !== m.id) : [...prev, m.id] ); }} >
{isExpanded ? : } {m.item_number}
{m.item_name} {m.supplier_item_code} {m.supplier_item_name} {m.base_price_type} {m.base_price ? Number(m.base_price).toLocaleString() : ""} {m.discount_type} {m.discount_value ? Number(m.discount_value).toLocaleString() : ""} {m.calculated_price ? Number(m.calculated_price).toLocaleString() : ""} {m.currency_code}
{/* 현재 단가 카드 (펼쳤을 때) */} {isExpanded && (() => { const cp = group.details.find((p) => p.is_current) || group.details[0]; if (!cp) return ( 등록된 단가가 없어요 ); return (
{/* 카드 헤더 */}
적용 단가 현재
{group.details.length > 1 && ( 전체 {group.details.length}건 중 )}
{/* 카드 내용 */}
기간 {cp.start_date ? String(cp.start_date).split("T")[0] : "—"} ~ {cp.end_date ? String(cp.end_date).split("T")[0] : "—"}
기준유형 {cp.base_price_type_label || "-"}
기준가 {cp.base_price ? Number(cp.base_price).toLocaleString() : "-"}
할인유형 {cp.discount_type_label && cp.discount_type_label !== "할인없음" ? cp.discount_type_label : "-"}
할인값 {cp.discount_value ? Number(cp.discount_value).toLocaleString() : "-"}
단수처리 {cp.rounding_unit_value ? (priceCategoryOptions["rounding_unit_value"]?.find((o) => o.code === cp.rounding_unit_value)?.label || cp.rounding_unit_value) : "-"}
계산단가 {(cp.calculated_price || cp.unit_price) ? Number(cp.calculated_price || cp.unit_price).toLocaleString() : "-"} {cp.currency_label}
); })()}
); })}
{/* 납품처 탭 */}
납품처코드 납품처명 주소 담당자 전화번호 메모 메인 {deliveryLoading ? ( ) : deliveryItems.length === 0 ? ( 등록된 납품처가 없어요 ) : deliveryItems.map((d) => ( {d.destination_code} {d.destination_name} {d.address} {d.manager_name} {d.phone} {d.memo} {d.is_default && ( 메인 )} ))}
)}
{/* ── 모달: 공급업체 등록/수정 (3탭) ── */} { if (!open && isConfirmOpenRef.current) return; setSupplierModalOpen(open); if (!open) { setModalContactFormOpen(false); setModalDeliveryFormOpen(false); setModalContactForm({}); setModalDeliveryForm({}); setModalContactEditId(null); setModalDeliveryEditId(null); fetchSuppliers(); if (supplierForm.supplier_code) { const cid = selectedSupplierId; setSelectedSupplierId(null); setTimeout(() => setSelectedSupplierId(cid), 50); } } }}> {supplierEditMode ? "공급업체 수정" : "공급업체 등록"} {supplierEditMode ? "공급업체 정보를 수정합니다." : "새 공급업체를 등록합니다."} setSupplierModalTab(v)} className="flex flex-col flex-1 overflow-hidden">
기본정보 담당자 관리 {modalContacts.length > 0 && ( {modalContacts.length} )} 납품처 관리 {modalDeliveries.length > 0 && ( {modalDeliveries.length} )}
{/* 기본정보 탭 */}
setSupplierForm((p) => ({ ...p, supplier_code: e.target.value }))} placeholder={supplierEditMode ? "" : "자동 생성"} className={cn("h-9 font-mono", !supplierEditMode && supplierForm.supplier_code && "bg-muted")} readOnly={!supplierEditMode && !!supplierForm.supplier_code} />
setSupplierForm((p) => ({ ...p, supplier_name: e.target.value }))} placeholder="공급업체명" className="h-9" />
setSupplierForm((p) => ({ ...p, contact_person: e.target.value }))} placeholder="공급업체담당자" className="h-9" />
handleFormChange("contact_phone", e.target.value)} placeholder="010-0000-0000" className={cn("h-9", formErrors.contact_phone && "border-destructive")} /> {formErrors.contact_phone &&

{formErrors.contact_phone}

}
handleFormChange("email", e.target.value)} placeholder="example@email.com" className={cn("h-9", formErrors.email && "border-destructive")} /> {formErrors.email &&

{formErrors.email}

}
handleFormChange("business_number", e.target.value)} placeholder="000-00-00000" className={cn("h-9", formErrors.business_number && "border-destructive")} /> {formErrors.business_number &&

{formErrors.business_number}

}
setSupplierForm((p) => ({ ...p, address: e.target.value }))} placeholder="주소" className="h-9" />
{/* 세금유형 */}
{taxTypeRows.map((row, idx) => (
{idx + 1} { const v = e.target.value.replace(/[^\d.]/g, ""); setTaxTypeRows((prev) => prev.map((r) => r._id === row._id ? { ...r, rate: v } : r)); }} placeholder="세율 %" className="h-9 text-[13px] w-[80px] text-right" /> %
))}
{/* 담당자 관리 탭 */}
{/* 담당자 목록 */}
담당자 {modalContacts.length}명
{modalContactLoading ? (
) : modalContacts.length === 0 ? (
등록된 담당자가 없어요
) : ( 담당자명 전화번호 이메일 부서 메인 메모 관리 {[...modalContacts].sort((a, b) => { const aMain = a.is_main === "Y" || a.is_main === true ? 0 : 1; const bMain = b.is_main === "Y" || b.is_main === true ? 0 : 1; return aMain - bMain; }).map((c) => ( {c.contact_name} {c.contact_phone} {c.contact_email} {c.department} {c.memo}
))}
)}
{/* 담당자 폼 (인라인) */} {modalContactFormOpen && (
{modalContactEditId ? "담당자 수정" : "담당자 추가"}
setModalContactForm((p) => ({ ...p, contact_name: e.target.value }))} placeholder="담당자명" className="h-8 text-sm" />
{ const formatted = formatField("phone", e.target.value); setModalContactForm((p) => ({ ...p, contact_phone: formatted })); }} placeholder="010-0000-0000" className="h-8 text-sm" />
setModalContactForm((p) => ({ ...p, contact_email: e.target.value }))} placeholder="example@email.com" className="h-8 text-sm" />
setModalContactForm((p) => ({ ...p, department: e.target.value }))} placeholder="부서명" className="h-8 text-sm" />
setModalContactForm((p) => ({ ...p, memo: e.target.value }))} placeholder="메모" className="h-8 text-sm" />
)}
{/* ── 탭3: 납품처 관리 ── */}
{/* 납품처 목록 헤더 */}
납품처 {modalDeliveries.length}개
{modalDeliveryLoading ? (
) : modalDeliveries.length === 0 ? (
등록된 납품처가 없어요
) : ( 납품처코드 납품처명 주소 담당자 전화번호 메모 메인 관리 {[...modalDeliveries].sort((a, b) => { const aMain = a.is_default === "Y" || a.is_default === true ? 0 : 1; const bMain = b.is_default === "Y" || b.is_default === true ? 0 : 1; return aMain - bMain; }).map((d) => ( {d.destination_code} {d.destination_name} {d.address} {d.manager_name} {d.phone} {d.memo}
))}
)}
{/* 납품처 폼 (인라인) */} {modalDeliveryFormOpen && (
{modalDeliveryEditId ? "납품처 수정" : "납품처 추가"}
setModalDeliveryForm((p) => ({ ...p, destination_code: e.target.value }))} placeholder="자동 생성" className={cn("h-8 text-sm font-mono", modalDeliveryForm.destination_code && "bg-muted")} readOnly={!!modalDeliveryForm.destination_code && !modalDeliveryEditId} />
setModalDeliveryForm((p) => ({ ...p, destination_name: e.target.value }))} placeholder="납품처명" className="h-8 text-sm" />
setModalDeliveryForm((p) => ({ ...p, manager_name: e.target.value }))} placeholder="담당자" className="h-8 text-sm" />
setModalDeliveryForm((p) => ({ ...p, address: e.target.value }))} placeholder="주소" className="h-8 text-sm" />
{ const formatted = formatField("phone", e.target.value); setModalDeliveryForm((p) => ({ ...p, phone: formatted })); const err = validateField("phone", formatted); setModalDeliveryFormErrors((p) => { const n = { ...p }; if (err) n.phone = err; else delete n.phone; return n; }); }} placeholder="010-0000-0000" className={cn("h-8 text-sm", modalDeliveryFormErrors.phone && "border-destructive")} /> {modalDeliveryFormErrors.phone &&

{modalDeliveryFormErrors.phone}

}
setModalDeliveryForm((p) => ({ ...p, memo: e.target.value }))} placeholder="메모" className="h-8 text-sm" />
)}
{!supplierEditMode && ( )}
{/* ── 모달: 품목 선택 (1단계) ── */} 품목 선택 공급업체에 추가할 품목을 선택하세요. (전체: {itemTotalCount}건 / 대상: {itemSearchResults.length}건{itemCheckedIds.size > 0 ? ` / 선택: ${itemCheckedIds.size}건` : ""})
setItemSearchKeyword(e.target.value)} onKeyDown={(e) => e.key === "Enter" && searchItems()} className="h-9 flex-1" />
0 && itemCheckedIds.size === itemSearchResults.length} onChange={(e) => { if (e.target.checked) setItemCheckedIds(new Set(itemSearchResults.map((i) => i.id))); else setItemCheckedIds(new Set()); }} /> 품목코드 품명 규격 재질 단위 {itemSearchResults.length === 0 ? ( 검색 결과가 없어요 ) : itemSearchResults.map((item) => ( setItemCheckedIds((prev) => { const next = new Set(prev); if (next.has(item.id)) next.delete(item.id); else next.add(item.id); return next; })} > {item.item_number} {item.item_name} {item.size} {categoryOptions["item_material"]?.find((o) => o.code === item.material)?.label || item.material} {categoryOptions["item_unit"]?.find((o) => o.code === item.unit)?.label || item.unit} ))}
{itemCheckedIds.size}개 선택됨
{/* ── 모달: 품목 상세 입력 (2단계) ── */} 품목 상세정보 {editItemData ? "수정" : "입력"} — {selectedSupplier?.supplier_name || ""} {editItemData ? "거래처 품번/품명과 기간별 단가를 수정합니다." : "선택한 품목의 거래처 품번/품명과 기간별 단가를 설정합니다."}
{selectedItemsForDetail.map((item, idx) => { const itemKey = item.item_number || item.id; const mappingRows = itemMappings[itemKey] || []; const prices = itemPrices[itemKey] || []; return (
{/* 품목 헤더 */}
{idx + 1}. {item.item_name || itemKey}
{itemKey} | {item.size || ""} | {categoryOptions["item_unit"]?.find((o) => o.code === item.unit)?.label || item.unit || ""}
{/* 좌: 거래처 품번/품명 */}
거래처 품번/품명 관리
{mappingRows.length === 0 ? (
입력된 거래처 품번이 없어요
) : ( handleMappingDragEnd(itemKey, e)} > r._id)} strategy={verticalListSortingStrategy}> {mappingRows.map((mRow, mIdx) => ( {mIdx + 1} updateMappingRow(itemKey, mRow._id, "supplier_item_code", e.target.value)} placeholder="거래처 품번" className="h-9 text-[13px] flex-1" /> updateMappingRow(itemKey, mRow._id, "supplier_item_name", e.target.value)} placeholder="거래처 품명" className="h-9 text-[13px] flex-1" /> ))} )}
{/* 우: 기간별 단가 */}
기간별 단가 설정
{prices.map((price, pIdx) => (
setCollapsedPriceCards((prev) => { const next = new Set(prev); if (next.has(price._id)) next.delete(price._id); else next.add(price._id); return next; })} >
{collapsedPriceCards.has(price._id) ? : } 단가 {pIdx + 1} {collapsedPriceCards.has(price._id) && price.calculated_price && ( {price.start_date || "—"} ~ {price.end_date || "—"} · {Number(price.calculated_price).toLocaleString()} {priceCategoryOptions["currency_code"]?.find((o) => o.code === price.currency_code)?.label || ""} )}
{prices.length > 1 && ( )}
{!collapsedPriceCards.has(price._id) &&
{/* 기간 + 통화 */}
{ const v = e.target.value; updatePriceRow(itemKey, price._id, "start_date", v); if (price.end_date && v > price.end_date) { updatePriceRow(itemKey, price._id, "end_date", v); } }} max={price.end_date || undefined} className="h-9 text-[13px] w-full" />
~
updatePriceRow(itemKey, price._id, "end_date", e.target.value)} min={price.start_date || undefined} className="h-9 text-[13px] w-full" />
{/* 기준유형 + 기준가 */}
{ const raw = e.target.value.replace(/[^\d.-]/g, ""); updatePriceRow(itemKey, price._id, "base_price", raw); }} className="h-9 text-[13px] text-right col-span-3" placeholder="기준가" />
{/* 할인 + 반올림 */}
{ const raw = e.target.value.replace(/[^\d.-]/g, ""); updatePriceRow(itemKey, price._id, "discount_value", raw); }} className="h-9 text-[13px] text-right" placeholder="0" />
{/* 계산 단가 */}
계산 단가: {price.calculated_price ? Number(price.calculated_price).toLocaleString() : "-"} {price.calculated_price && price.currency_code && ( {priceCategoryOptions["currency_code"]?.find((o) => o.code === price.currency_code)?.label || ""} )}
}
))}
); })}
{/* 엑셀 업로드 (멀티테이블) */} {excelChainConfig && ( { setExcelUploadOpen(open); if (!open) setExcelChainConfig(null); }} config={excelChainConfig} onSuccess={() => { fetchSuppliers(); const cid = selectedSupplierId; setSelectedSupplierId(null); setTimeout(() => setSelectedSupplierId(cid), 50); }} /> )} {/* 테이블 설정 모달 */} {ConfirmDialogComponent}
); }