"use client";
/**
* 거래처관리 — Type B 마스터-디테일 레이아웃 (리디자인)
*
* 좌측: 거래처 목록 (customer_mng)
* 우측: 품목별 단가 + 납품처 정보 탭
*
* 모달:
* - 거래처 등록/수정 (customer_mng)
* - 품목 추가 (item_info 검색 → customer_item_mapping + customer_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 CUSTOMER_TABLE = "customer_mng";
const MAPPING_TABLE = "customer_item_mapping";
const PRICE_TABLE = "customer_item_prices";
const DELIVERY_TABLE = "delivery_destination";
const CONTACT_TABLE = "customer_contact";
const CUSTOMER_GRID_COLUMNS = [
{ key: "customer_code", label: "거래처코드" },
{ key: "customer_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 (
);
}
export default function CustomerManagementPage() {
const { user } = useAuth();
const { confirm, ConfirmDialogComponent, isConfirmOpenRef } = useConfirmDialog();
const ts = useTableSettings("c16-customer", CUSTOMER_TABLE, CUSTOMER_GRID_COLUMNS);
const dndSensors = useSensors(useSensor(PointerSensor, { activationConstraint: { distance: 5 } }));
// 검색 필터 (DynamicSearchFilter에서 관리)
const [searchFilters, setSearchFilters] = useState([]);
// 좌측: 거래처 목록
const [customers, setCustomers] = useState([]);
const [rawCustomers, setRawCustomers] = useState([]);
const [customerLoading, setCustomerLoading] = useState(false);
const [showInactive, setShowInactive] = useState(false);
const [mainContactMap, setMainContactMap] = useState>({});
const [customerCount, setCustomerCount] = useState(0);
const [selectedCustomerId, setSelectedCustomerId] = 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 [customerModalOpen, setCustomerModalOpen] = useState(false);
const [customerEditMode, setCustomerEditMode] = useState(false);
const [customerForm, setCustomerForm] = 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 [customerModalTab, setCustomerModalTab] = useState<"basic" | "contacts" | "delivery">("basic");
// 담당자 (customer_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/${CUSTOMER_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/customer_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 fetchCustomers = useCallback(async () => {
setCustomerLoading(true);
try {
const filters = searchFilters.map(f => ({
columnName: f.columnName,
operator: f.operator,
value: f.value,
}));
const res = await apiClient.post(`/table-management/tables/${CUSTOMER_TABLE}/data`, {
page: 1, size: 500,
dataFilter: filters.length > 0 ? { enabled: true, filters } : undefined,
autoFilter: true,
sort: { columnName: "customer_code", order: "desc" },
});
const raw = res.data?.data?.data || res.data?.data?.rows || [];
setRawCustomers(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.customer_code || "").replace(/\D/g, ""), 10) || 0;
const bNum = parseInt((b.customer_code || "").replace(/\D/g, ""), 10) || 0;
return bNum - aNum;
});
setCustomers(data);
setCustomerCount(res.data?.data?.total || raw.length);
} catch (err) {
console.error("거래처 조회 실패:", err);
toast.error("거래처 목록을 불러오는데 실패했습니다.");
} finally {
setCustomerLoading(false);
}
}, [searchFilters, categoryOptions, employeeOptions, mainContactMap]);
useEffect(() => { fetchCustomers(); }, [fetchCustomers]);
// 메인 담당자 조회 (최초 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.customer_id) {
map[c.customer_id] = c;
}
}
setMainContactMap(map);
} catch { /* skip */ }
}, []);
useEffect(() => { fetchMainContacts(); }, [fetchMainContacts]);
const selectedCustomer = customers.find((c) => c.id === selectedCustomerId);
// 선택된 거래처의 품목 단가 조회
useEffect(() => {
if (!selectedCustomer?.customer_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: "customer_id", operator: "equals", value: selectedCustomer.customer_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: "customer_id", operator: "equals", value: selectedCustomer.customer_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 now = new Date();
const today = `${now.getFullYear()}-${String(now.getMonth() + 1).padStart(2, "0")}-${String(now.getDate()).padStart(2, "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();
}, [selectedCustomer?.customer_code]);
// 납품처 조회
useEffect(() => {
if (!selectedCustomer?.customer_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: "customer_code", operator: "equals", value: selectedCustomer.customer_code },
]},
autoFilter: true,
});
setDeliveryItems(res.data?.data?.data || res.data?.data?.rows || []);
} catch { setDeliveryItems([]); } finally { setDeliveryLoading(false); }
};
fetchDelivery();
}, [selectedCustomer?.customer_code]);
const getCategoryLabel = (col: string, code: string) => {
if (!code) return "";
return categoryOptions[col]?.find((o) => o.code === code)?.label || code;
};
// 모달 내 담당자 목록 조회
const fetchModalContacts = useCallback(async (customerId: string) => {
setModalContactLoading(true);
try {
const res = await apiClient.post(`/table-management/tables/${CONTACT_TABLE}/data`, {
page: 1, size: 200,
dataFilter: { enabled: true, filters: [{ columnName: "customer_id", operator: "equals", value: customerId }] },
autoFilter: true,
});
setModalContacts(res.data?.data?.data || res.data?.data?.rows || []);
} catch { setModalContacts([]); } finally { setModalContactLoading(false); }
}, []);
// 모달 내 납품처 목록 조회
const fetchModalDeliveries = useCallback(async (customerCode: string) => {
setModalDeliveryLoading(true);
try {
const res = await apiClient.post(`/table-management/tables/${DELIVERY_TABLE}/data`, {
page: 1, size: 200,
dataFilter: { enabled: true, filters: [{ columnName: "customer_code", operator: "equals", value: customerCode }] },
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 openCustomerRegister = async () => {
setCustomerForm({});
setFormErrors({});
setCustomerEditMode(false);
setCustomerModalTab("basic");
setModalContacts([]);
setModalDeliveries([]);
setModalContactFormOpen(false);
setModalDeliveryFormOpen(false);
setTaxTypeRows([]);
setCustomerModalOpen(true);
// 거래처 코드 자동 채번 — 기존 데이터 max값 기반
try {
const ruleRes = await apiClient.get(`/numbering-rules/by-column/${CUSTOMER_TABLE}/customer_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/${CUSTOMER_TABLE}/data`, {
page: 1, size: 500, autoFilter: true,
sort: { columnName: "customer_code", order: "desc" },
});
const allRows = allRes.data?.data?.data || allRes.data?.data?.rows || [];
let maxSeq = 0;
for (const row of allRows) {
const code = row.customer_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");
setCustomerForm((prev) => ({ ...prev, customer_code: nextCode, _numberingRuleId: ruleId }));
}
}
} catch { /* skip */ }
};
const openCustomerEdit = () => {
if (!selectedCustomer) return;
const rawData = rawCustomers.find((c) => c.id === selectedCustomerId);
setCustomerForm({ ...(rawData || selectedCustomer) });
setFormErrors({});
setCustomerEditMode(true);
setCustomerModalTab("basic");
setModalContactFormOpen(false);
setModalDeliveryFormOpen(false);
setModalContactForm({});
setModalDeliveryForm({});
setModalContactEditId(null);
setModalDeliveryEditId(null);
// 수정 모드에서는 바로 조회
const code = (rawData || selectedCustomer).customer_code;
const id = (rawData || selectedCustomer).id;
if (id) {
fetchModalContacts(id);
// 세금유형 로드
apiClient.post(`/table-management/tables/customer_tax_type/data`, {
page: 1, size: 100,
dataFilter: { enabled: true, filters: [{ columnName: "customer_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);
setCustomerModalOpen(true);
};
// 폼 필드 변경 시 자동 포맷팅 + 실시간 검증
const handleFormChange = (field: string, value: string) => {
const formatted = formatField(field, value);
setCustomerForm((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 (customerId: string, customerCode: string) => {
// 세금유형 — 기존 삭제 후 재생성
try {
const existTax = await apiClient.post(`/table-management/tables/customer_tax_type/data`, {
page: 1, size: 100,
dataFilter: { enabled: true, filters: [{ columnName: "customer_id", operator: "equals", value: customerId }] },
autoFilter: true,
});
const existRows = existTax.data?.data?.data || existTax.data?.data?.rows || [];
if (existRows.length > 0) {
await apiClient.delete(`/table-management/tables/customer_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/customer_tax_type/add`, {
id: crypto.randomUUID(), customer_id: customerId,
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: "customer_id", operator: "equals", value: customerId }] },
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(), customer_id: customerId,
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: "customer_code", operator: "equals", value: customerCode }] },
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(), customer_code: customerCode,
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 handleCustomerSave = async () => {
if (!customerForm.customer_name) { toast.error("거래처명은 필수입니다."); return; }
if (!customerForm.status) { toast.error("상태는 필수입니다."); return; }
const errors = validateForm(customerForm, ["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 } = customerForm;
const cleanFields: Record = {};
for (const [key, value] of Object.entries(fields)) {
cleanFields[key] = value === "" ? null : value;
}
if (customerEditMode && id) {
// 수정
await apiClient.put(`/table-management/tables/${CUSTOMER_TABLE}/edit`, {
originalData: { id }, updatedData: cleanFields,
});
await saveSubTables(id, cleanFields.customer_code || customerForm.customer_code);
toast.success("저장되었습니다.");
} else {
// 신규 등록
await apiClient.post(`/table-management/tables/${CUSTOMER_TABLE}/add`, cleanFields);
// id 획득
const res = await apiClient.post(`/table-management/tables/${CUSTOMER_TABLE}/data`, {
page: 1, size: 1,
dataFilter: { enabled: true, filters: [{ columnName: "customer_code", operator: "equals", value: cleanFields.customer_code }] },
autoFilter: true,
});
const newRow = (res.data?.data?.data || res.data?.data?.rows || [])[0];
if (newRow?.id) {
await saveSubTables(newRow.id, cleanFields.customer_code);
}
toast.success("거래처가 등록되었습니다.");
}
fetchCustomers();
fetchMainContacts();
if (!customerEditMode && continuousInput) {
// 연속입력 — 폼 초기화하고 모달 유지
setCustomerForm({});
setModalContacts([]);
setModalDeliveries([]);
setTaxTypeRows([]);
setCustomerModalTab("basic");
// 새 코드 채번
try {
const ruleRes = await apiClient.get(`/numbering-rules/by-column/${CUSTOMER_TABLE}/customer_code`);
const ruleData = ruleRes.data;
if (ruleData?.success && ruleData?.data?.ruleId) {
const ruleId = ruleData.data.ruleId;
const allRes = await apiClient.post(`/table-management/tables/${CUSTOMER_TABLE}/data`, { page: 1, size: 500, autoFilter: true, sort: { columnName: "customer_code", order: "desc" } });
const allRows = allRes.data?.data?.data || allRes.data?.data?.rows || [];
let maxSeq = 0;
for (const row of allRows) { const match = (row.customer_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;
setCustomerForm({ customer_code: prefix + String(maxSeq + 1).padStart(seqLen, "0") });
}
}
} catch { /* skip */ }
toast.success("등록 완료. 다음 거래처를 입력하세요.");
} else {
setCustomerModalOpen(false);
// 우측 패널 갱신
if (selectedCustomerId) {
const cid = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
}
}
} catch (err: any) {
toast.error(err.response?.data?.message || "저장에 실패했습니다.");
} finally {
setSaving(false);
}
};
// 거래처 삭제
const handleCustomerDelete = async () => {
if (!selectedCustomerId) return;
const ok = await confirm("거래처를 삭제하시겠습니까?", {
variant: "destructive", confirmText: "삭제",
});
if (!ok) return;
try {
await apiClient.delete(`/table-management/tables/${CUSTOMER_TABLE}/delete`, {
data: [{ id: selectedCustomerId }],
});
toast.success("삭제되었습니다.");
setSelectedCustomerId(null);
fetchCustomers();
} catch { toast.error("삭제에 실패했습니다."); }
};
// 품목 검색
const searchItems = useCallback(async () => {
setItemSearchLoading(true);
try {
const salesCode = categoryOptions["item_division"]?.find((o) => o.label === "영업관리")?.code;
const filters: any[] = salesCode
? [{ columnName: "division", operator: "contains", value: salesCode }]
: [];
const res = await apiClient.post(`/table-management/tables/item_info/data`, {
page: 1, size: 5000,
dataFilter: { enabled: true, filters },
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 kw = itemSearchKeyword.toLowerCase();
const seenNumbers = new Set();
const deduped = allItems.filter((item: any) => {
if (existingItemIds.has(item.item_number) || existingItemIds.has(item.id)) return false;
if (item.item_number && seenNumbers.has(item.item_number)) return false;
if (item.item_number) seenNumbers.add(item.item_number);
if (kw) {
const name = (item.item_name || "").toLowerCase();
const code = (item.item_number || "").toLowerCase();
if (!name.includes(kw) && !code.includes(kw)) return false;
}
return true;
});
setItemSearchResults(deduped);
} catch { /* skip */ } finally { setItemSearchLoading(false); }
}, [itemSearchKeyword, priceItems]);
// 실시간 검색 (2글자 이상)
useEffect(() => {
if (!itemSelectOpen) return;
if (itemSearchKeyword.length > 0 && itemSearchKeyword.length < 2) return;
searchItems();
}, [itemSearchKeyword, itemSelectOpen]);
// 품목 선택 완료 → 상세 입력 모달로 전환
const goToItemDetail = () => {
const raw = itemSearchResults.filter((i) => itemCheckedIds.has(i.id));
if (raw.length === 0) { toast.error("품목을 선택해주세요."); return; }
const seenKeys = new Set();
const selected = raw.filter((i) => {
const k = i.item_number || i.id;
if (seenKeys.has(k)) return false;
seenKeys.add(k);
return true;
});
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()}`, customer_item_code: "", customer_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: "customer_id", operator: "equals", value: selectedCustomer!.customer_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.customer_item_code || m.customer_item_name)
.map((m: any) => ({
_id: `m_existing_${m.id}`,
customer_item_code: m.customer_item_code || "",
customer_item_name: m.customer_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: "customer_id", operator: "equals", value: selectedCustomer!.customer_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 (!selectedCustomer) return;
if (savingRef.current) return;
savingRef.current = true;
const isEditingExisting = !!editItemData;
setSaving(true);
try {
const processedKeys = new Set();
for (const item of selectedItemsForDetail) {
const itemKey = item.item_number || item.id;
if (processedKeys.has(itemKey)) continue;
processedKeys.add(itemKey);
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: "customer_id", operator: "equals", value: selectedCustomer.customer_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: {
customer_item_code: mappingRows[mi].customer_item_code || "",
customer_item_name: mappingRows[mi].customer_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(),
customer_id: selectedCustomer.customer_code, item_id: itemKey,
customer_item_code: mappingRows[mi].customer_item_code || "",
customer_item_name: mappingRows[mi].customer_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: "customer_id", operator: "equals", value: selectedCustomer.customer_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,
customer_id: selectedCustomer.customer_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]?.customer_item_code) {
const existingCheck = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, {
page: 1, size: 1,
dataFilter: { enabled: true, filters: [
{ columnName: "customer_id", operator: "equals", value: selectedCustomer.customer_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(),
customer_id: selectedCustomer.customer_code, item_id: itemKey,
customer_item_code: mappingRows[0]?.customer_item_code || "",
customer_item_name: mappingRows[0]?.customer_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(),
customer_id: selectedCustomer.customer_code, item_id: itemKey,
customer_item_code: mappingRows[mi].customer_item_code || "",
customer_item_name: mappingRows[mi].customer_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 || "", customer_id: selectedCustomer.customer_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 = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
} catch (err: any) {
toast.error(err.response?.data?.message || "저장에 실패했습니다.");
} finally {
setSaving(false);
savingRef.current = false;
}
};
// 품목 매핑 해제 — 선택한 품목의 모든 매핑 + 단가에서 customer_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) {
// 해당 품목의 모든 매핑 조회 → customer_id null 처리
const mapRes = await apiClient.post(`/table-management/tables/${MAPPING_TABLE}/data`, {
page: 1, size: 500,
dataFilter: { enabled: true, filters: [
{ columnName: "customer_id", operator: "equals", value: selectedCustomer!.customer_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: { customer_id: null },
});
}
// 해당 품목의 모든 단가 조회 → customer_id null 처리
try {
const priceRes = await apiClient.post(`/table-management/tables/${PRICE_TABLE}/data`, {
page: 1, size: 500,
dataFilter: { enabled: true, filters: [
{ columnName: "customer_id", operator: "equals", value: selectedCustomer!.customer_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: { customer_id: null },
});
}
} catch { /* skip */ }
}
toast.success(`${priceCheckedIds.length}개 품목의 연결이 해제되었습니다.`);
setPriceCheckedIds([]);
const cid = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
} catch {
toast.error("연결 해제에 실패했습니다.");
}
};
// EDataTable 컬럼 정의 (거래처 목록) — ts.visibleColumns 순서를 따름
const customerColumns: EDataTableColumn[] = useMemo(() => {
const colProps: Record> = {
customer_code: { width: "w-[120px]" },
customer_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 (customers.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 customers) {
const custMappings = allMappings.filter((m) => m.customer_id === c.customer_code);
if (custMappings.length === 0) {
rows.push({
거래처코드: c.customer_code, 거래처명: c.customer_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 custMappings) {
rows.push({
거래처코드: c.customer_code, 거래처명: c.customer_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.customer_item_code || "", 거래처품명: m.customer_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) */}
{/* 액션 버튼 영역 */}
{/* 마스터-디테일 분할 패널 */}
{/* 좌측: 거래처 목록 */}
{/* 패널 헤더 */}
{/* 거래처 테이블 */}
c.status !== "거래정지"))}
rowKey={(row) => row.id}
loading={customerLoading}
emptyMessage="등록된 거래처가 없어요"
selectedId={selectedCustomerId}
onSelect={(id) => setSelectedCustomerId(id)}
onRowDoubleClick={(row) => { setSelectedCustomerId(row.id); openCustomerEdit(); }}
showRowNumber
showPagination
defaultPageSize={20}
draggableColumns={false}
columnOrderKey="c16-customer"
/>
{/* 우측: 디테일 패널 */}
{!selectedCustomerId ? (
/* 빈 상태 */
거래처를 선택해주세요
좌측에서 거래처를 선택하면 상세 정보가 표시돼요
) : (
<>
{/* 탭 + 버튼 통합 헤더 */}
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.customer_item_code}
{m.customer_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탭) ── */}
{/* ── 모달: 품목 선택 (1단계) ── */}
{/* ── 모달: 품목 상세 입력 (2단계) ── */}
{/* 엑셀 업로드 (멀티테이블) */}
{excelChainConfig && (
{
setExcelUploadOpen(open);
if (!open) setExcelChainConfig(null);
}}
config={excelChainConfig}
onSuccess={() => {
fetchCustomers();
const cid = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
}}
/>
)}
{/* 테이블 설정 모달 */}
{ConfirmDialogComponent}
);
}