Files
pipeline/frontend/app/(main)/COMPANY_16/sales/customer/page.tsx
T

1753 lines
86 KiB
TypeScript

"use client";
/**
* 거래처관리 — Type B 마스터-디테일 레이아웃 (리디자인)
*
* 좌측: 거래처 목록 (customer_mng)
* 우측: 품목별 단가 + 납품처 정보 탭
*
* 모달:
* - 거래처 등록/수정 (customer_mng)
* - 품목 추가 (item_info 검색 → customer_item_mapping + customer_item_prices)
* - 납품처 등록 (delivery_destination)
*/
import React, { useState, useEffect, useCallback, useRef } 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,
} from "lucide-react";
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 { 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 CUSTOMER_GRID_COLUMNS = [
{ key: "customer_code", label: "거래처코드" },
{ key: "customer_name", label: "거래처명" },
{ key: "contact_person", label: "대표자" },
{ key: "contact_phone", label: "연락처" },
{ key: "division", label: "유형" },
{ key: "status", label: "상태" },
];
export default function CustomerManagementPage() {
const { user } = useAuth();
const { confirm, ConfirmDialogComponent } = useConfirmDialog();
const ts = useTableSettings("c16-customer", CUSTOMER_TABLE, CUSTOMER_GRID_COLUMNS);
// 검색 필터 (DynamicSearchFilter에서 관리)
const [searchFilters, setSearchFilters] = useState<FilterValue[]>([]);
// 좌측: 거래처 목록
const [customers, setCustomers] = useState<any[]>([]);
const [rawCustomers, setRawCustomers] = useState<any[]>([]);
const [customerLoading, setCustomerLoading] = useState(false);
const [customerCount, setCustomerCount] = useState(0);
const [selectedCustomerId, setSelectedCustomerId] = useState<string | null>(null);
// 우측: 탭
const [rightTab, setRightTab] = useState<"items" | "delivery">("items");
// 우측: 품목 단가
const [priceItems, setPriceItems] = useState<any[]>([]);
const [priceLoading, setPriceLoading] = useState(false);
const [priceCheckedIds, setPriceCheckedIds] = useState<string[]>([]);
// 우측: 납품처
const [deliveryItems, setDeliveryItems] = useState<any[]>([]);
const [deliveryCheckedIds, setDeliveryCheckedIds] = useState<string[]>([]);
const [deliveryLoading, setDeliveryLoading] = useState(false);
// 품목 편집 데이터 (더블클릭 시 상세 입력 모달 재활용)
const [editItemData, setEditItemData] = useState<any>(null);
const savingRef = useRef(false);
// 거래처 모달
const [customerModalOpen, setCustomerModalOpen] = useState(false);
const [customerEditMode, setCustomerEditMode] = useState(false);
const [customerForm, setCustomerForm] = useState<Record<string, any>>({});
const [formErrors, setFormErrors] = useState<Record<string, string>>({});
const [saving, setSaving] = useState(false);
// 품목 추가 모달 (1단계: 검색/선택)
const [itemSelectOpen, setItemSelectOpen] = useState(false);
const [itemSearchKeyword, setItemSearchKeyword] = useState("");
const [itemSearchResults, setItemSearchResults] = useState<any[]>([]);
const [itemSearchLoading, setItemSearchLoading] = useState(false);
const [itemCheckedIds, setItemCheckedIds] = useState<Set<string>>(new Set());
// 품목 상세 입력 모달 (2단계: 거래처 품번/품명 + 단가)
const [itemDetailOpen, setItemDetailOpen] = useState(false);
const [selectedItemsForDetail, setSelectedItemsForDetail] = useState<any[]>([]);
const [itemMappings, setItemMappings] = useState<Record<string, Array<{ _id: string; customer_item_code: string; customer_item_name: string }>>>({});
const [itemPrices, setItemPrices] = useState<Record<string, Array<{
_id: string; start_date: string; end_date: string; currency_code: string;
base_price_type: string; base_price: string; discount_type: string;
discount_value: string; rounding_type: string; rounding_unit_value: string;
calculated_price: string;
}>>>({});
const [priceCategoryOptions, setPriceCategoryOptions] = useState<Record<string, { code: string; label: string }[]>>({});
// 납품처 모달
const [deliveryModalOpen, setDeliveryModalOpen] = useState(false);
const [deliveryForm, setDeliveryForm] = useState<Record<string, any>>({});
// 엑셀
const [excelUploadOpen, setExcelUploadOpen] = useState(false);
const [excelChainConfig, setExcelChainConfig] = useState<TableChainConfig | null>(null);
const [excelDetecting, setExcelDetecting] = useState(false);
// 카테고리
const [categoryOptions, setCategoryOptions] = useState<Record<string, { code: string; label: string }[]>>({});
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<string, { code: string; label: string }[]> = {};
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 */ }
}
setCategoryOptions(optMap);
const priceOpts: Record<string, { code: string; label: string }[]> = {};
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);
};
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,
});
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) => ({
...r,
division: resolve("division", r.division),
status: resolve("status", r.status),
internal_manager: r.internal_manager
? (employeeOptions.find((e) => e.user_id === r.internal_manager)?.user_name || r.internal_manager)
: "",
}));
setCustomers(data);
setCustomerCount(res.data?.data?.total || raw.length);
} catch (err) {
console.error("거래처 조회 실패:", err);
toast.error("거래처 목록을 불러오는데 실패했습니다.");
} finally {
setCustomerLoading(false);
}
}, [searchFilters, categoryOptions, employeeOptions]);
useEffect(() => { fetchCustomers(); }, [fetchCustomers]);
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<string, any> = {};
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 today = new Date().toISOString().split("T")[0];
const seenItemIds = new Set<string>();
const sortedMappings = [...mappings].sort((a: any, b: any) => (a.item_id || "").localeCompare(b.item_id || ""));
setPriceItems(sortedMappings.map((m: any) => {
const itemKey = m.item_id || "";
const itemInfo = itemMap[itemKey] || {};
const isFirstOfGroup = !seenItemIds.has(itemKey);
if (itemKey) seenItemIds.add(itemKey);
const itemPriceList = allPrices.filter((p: any) => p.item_id === itemKey);
const todayPrice = itemPriceList.find((p: any) =>
(!p.start_date || p.start_date <= today) && (!p.end_date || p.end_date >= today)
) || itemPriceList[0] || {};
return {
...m,
item_number: isFirstOfGroup ? itemKey : "",
item_name: isFirstOfGroup ? (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 || ""),
};
}));
} catch (err) {
console.error("품목 조회 실패:", err);
} finally {
setPriceLoading(false);
}
};
fetchItems();
}, [selectedCustomer?.customer_code]);
// 납품처 조회
useEffect(() => {
if (!selectedCustomer?.customer_code) { setDeliveryItems([]); setDeliveryCheckedIds([]); return; }
setDeliveryCheckedIds([]);
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 openCustomerRegister = () => {
setCustomerForm({});
setFormErrors({});
setCustomerEditMode(false);
setCustomerModalOpen(true);
};
const openCustomerEdit = () => {
if (!selectedCustomer) return;
const rawData = rawCustomers.find((c) => c.id === selectedCustomerId);
setCustomerForm({ ...(rawData || selectedCustomer) });
setFormErrors({});
setCustomerEditMode(true);
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 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, ...fields } = customerForm;
const cleanFields: Record<string, any> = {};
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,
});
toast.success("수정되었습니다.");
} else {
await apiClient.post(`/table-management/tables/${CUSTOMER_TABLE}/add`, cleanFields);
toast.success("등록되었습니다.");
}
setCustomerModalOpen(false);
fetchCustomers();
} catch (err: any) {
toast.error(err.response?.data?.message || "저장에 실패했습니다.");
} finally {
setSaving(false);
}
};
// 거래처 삭제
const handleCustomerDelete = async () => {
if (!selectedCustomerId) return;
const ok = await confirm("거래처를 삭제하시겠습니까?", {
description: "관련된 품목 매핑, 단가, 납품처 정보도 함께 삭제됩니다.",
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 = async () => {
setItemSearchLoading(true);
try {
const filters: any[] = [];
if (itemSearchKeyword) filters.push({ columnName: "item_name", operator: "contains", value: itemSearchKeyword });
filters.push({ columnName: "division", operator: "contains", value: "CAT_ML8ZFVEL_1TOR" });
const res = await apiClient.post(`/table-management/tables/item_info/data`, {
page: 1, size: 50,
dataFilter: filters.length > 0 ? { enabled: true, filters } : undefined,
autoFilter: true,
});
const allItems = res.data?.data?.data || res.data?.data?.rows || [];
const existingItemIds = new Set(priceItems.map((p: any) => p.item_id || p.item_number));
setItemSearchResults(allItems.filter((item: any) => !existingItemIds.has(item.item_number) && !existingItemIds.has(item.id)));
} catch { /* skip */ } finally { setItemSearchLoading(false); }
};
// 품목 선택 완료 → 상세 입력 모달로 전환
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()}`, customer_item_code: "", customer_item_name: "" }],
}));
};
const removeMappingRow = (itemKey: string, rowId: string) => {
setItemMappings((prev) => ({
...prev,
[itemKey]: (prev[itemKey] || []).filter((r) => r._id !== rowId),
}));
};
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"].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;
updated.calculated_price = String(Math.round(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,
});
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 || [];
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 {
for (const item of selectedItemsForDetail) {
const itemKey = item.item_number || item.id;
const mappingRows = itemMappings[itemKey] || [];
if (isEditingExisting && editItemData?.id) {
await apiClient.put(`/table-management/tables/${MAPPING_TABLE}/edit`, {
originalData: { id: editItemData.id },
updatedData: {
customer_item_code: mappingRows[0]?.customer_item_code || "",
customer_item_name: mappingRows[0]?.customer_item_name || "",
base_price: null, discount_type: null, discount_value: null, calculated_price: null,
},
});
try {
const existingPrices = await apiClient.post(`/table-management/tables/${PRICE_TABLE}/data`, {
page: 1, size: 100,
dataFilter: { enabled: true, filters: [
{ columnName: "mapping_id", operator: "equals", value: editItemData.id },
]}, autoFilter: true,
});
const existing = existingPrices.data?.data?.data || existingPrices.data?.data?.rows || [];
if (existing.length > 0) {
await apiClient.delete(`/table-management/tables/${PRICE_TABLE}/delete`, {
data: existing.map((p: any) => ({ id: p.id })),
});
}
} catch { /* skip */ }
const priceRows = (itemPrices[itemKey] || []).filter((p) =>
(p.base_price && Number(p.base_price) > 0) || p.start_date
);
for (const price of priceRows) {
await apiClient.post(`/table-management/tables/${PRICE_TABLE}/add`, {
id: crypto.randomUUID(),
mapping_id: 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,
});
}
} 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 && Number(p.base_price) > 0) || p.start_date
);
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;
}
};
// 납품처 저장
const handleDeliverySave = async () => {
if (!deliveryForm.destination_name || !selectedCustomer) return;
try {
await apiClient.post(`/table-management/tables/${DELIVERY_TABLE}/add`, {
id: crypto.randomUUID(),
...deliveryForm,
customer_code: selectedCustomer.customer_code,
});
toast.success("납품처가 등록되었습니다.");
setDeliveryModalOpen(false);
const cid = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
} catch (err: any) {
toast.error(err.response?.data?.message || "납품처 등록에 실패했습니다.");
}
};
// 품목 매핑 삭제
const handlePriceItemDelete = async () => {
if (priceCheckedIds.length === 0) return;
const ok = await confirm(`선택한 ${priceCheckedIds.length}개 품목 매핑을 삭제하시겠습니까?`, {
description: "관련된 단가 정보도 함께 삭제됩니다.",
variant: "destructive", confirmText: "삭제",
});
if (!ok) return;
try {
for (const mappingId of priceCheckedIds) {
try {
const priceRes = await apiClient.post(`/table-management/tables/${PRICE_TABLE}/data`, {
page: 1, size: 500,
dataFilter: { enabled: true, filters: [{ columnName: "mapping_id", operator: "equals", value: mappingId }] },
autoFilter: true,
});
const prices = priceRes.data?.data?.data || priceRes.data?.data?.rows || [];
if (prices.length > 0) {
await apiClient.delete(`/table-management/tables/${PRICE_TABLE}/delete`, {
data: prices.map((p: any) => ({ id: p.id })),
});
}
} catch { /* skip */ }
}
await apiClient.delete(`/table-management/tables/${MAPPING_TABLE}/delete`, {
data: priceCheckedIds.map((id) => ({ id })),
});
toast.success(`${priceCheckedIds.length}개 품목 매핑이 삭제되었습니다.`);
setPriceCheckedIds([]);
const cid = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
} catch {
toast.error("삭제에 실패했습니다.");
}
};
// 납품처 삭제
const handleDeliveryDelete = async () => {
if (deliveryCheckedIds.length === 0) return;
const ok = await confirm(`선택한 ${deliveryCheckedIds.length}개 납품처를 삭제하시겠습니까?`, {
variant: "destructive", confirmText: "삭제",
});
if (!ok) return;
try {
await apiClient.delete(`/table-management/tables/${DELIVERY_TABLE}/delete`, {
data: deliveryCheckedIds.map((id) => ({ id })),
});
toast.success(`${deliveryCheckedIds.length}개 납품처가 삭제되었습니다.`);
setDeliveryCheckedIds([]);
const cid = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
} catch {
toast.error("삭제에 실패했습니다.");
}
};
// 컬럼 가시성 헬퍼
const isColumnVisible = (key: string) => ts.isVisible(key);
const customerColSpan = 1 + ["customer_code", "customer_name", "contact_person", "contact_phone", "division", "status"]
.filter((k) => isColumnVisible(k)).length;
// EDataTable 컬럼 정의 (거래처 목록)
const customerColumns: EDataTableColumn[] = [
...(isColumnVisible("customer_code") ? [{ key: "customer_code", label: "거래처코드", width: "w-[120px]" }] : []),
...(isColumnVisible("customer_name") ? [{ key: "customer_name", label: "거래처명", minWidth: "min-w-[160px]" }] : []),
...(isColumnVisible("contact_person") ? [{ key: "contact_person", label: "대표자", width: "w-[90px]" }] : []),
...(isColumnVisible("contact_phone") ? [{ key: "contact_phone", label: "연락처", width: "w-[120px]" }] : []),
...(isColumnVisible("division") ? [{
key: "division",
label: "유형",
width: "w-[80px]",
render: (val: any) =>
val ? (
<Badge variant="secondary" className="text-[10px] px-1.5 py-0 h-5 font-normal">
{val}
</Badge>
) : null,
}] : []),
...(isColumnVisible("status") ? [{
key: "status",
label: "상태",
width: "w-[70px]",
render: (val: any) =>
val ? (
<Badge
variant={val === "활성" || val === "거래중" ? "default" as const : "outline" as const}
className="text-[10px] px-1.5 py-0 h-5"
>
{val}
</Badge>
) : null,
}] : []),
];
// 엑셀 다운로드
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<string, any> = {};
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<string, any>[] = [];
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 (
<div className="flex h-full flex-col gap-3 p-4">
{/* 검색 필터 (DynamicSearchFilter) */}
<DynamicSearchFilter
tableName={CUSTOMER_TABLE}
filterId="c16-customer"
onFilterChange={setSearchFilters}
dataCount={customers.length}
externalFilterConfig={ts.filterConfig}
/>
{/* 액션 버튼 영역 */}
<div className="flex items-center gap-2 px-4 shrink-0">
<div className="flex gap-1.5 ml-auto">
<Button
variant="outline" size="sm" className="h-8" disabled={excelDetecting}
onClick={async () => {
setExcelDetecting(true);
try {
const result = await autoDetectMultiTableConfig(CUSTOMER_TABLE);
if (result.success && result.data) {
setExcelChainConfig(result.data);
setExcelUploadOpen(true);
} else {
toast.error(result.message || "테이블 구조 분석 실패");
}
} catch { toast.error("테이블 구조 분석 중 오류"); } finally { setExcelDetecting(false); }
}}
>
{excelDetecting ? <Loader2 className="w-3.5 h-3.5 mr-1 animate-spin" /> : <FileSpreadsheet className="w-3.5 h-3.5 mr-1" />}
</Button>
<Button variant="outline" size="sm" className="h-8" onClick={handleExcelDownload}>
<Download className="w-3.5 h-3.5 mr-1" />
</Button>
</div>
</div>
{/* 마스터-디테일 분할 패널 */}
<div className="flex-1 overflow-hidden border rounded-lg bg-background shadow-sm">
<ResizablePanelGroup direction="horizontal">
{/* 좌측: 거래처 목록 */}
<ResizablePanel defaultSize={55} minSize={30}>
<div className="flex flex-col h-full">
{/* 패널 헤더 */}
<div className="flex items-center justify-between px-4 py-3 border-b bg-muted shrink-0">
<div className="flex items-center gap-2.5">
<span className="text-[13px] font-bold"> </span>
<span className="text-[11px] font-semibold text-primary bg-primary/[0.08] px-2 py-0.5 rounded-full">
{customerCount}
</span>
</div>
<div className="flex gap-1.5">
<Button size="sm" onClick={openCustomerRegister}>
<Plus className="w-3.5 h-3.5 mr-1" />
</Button>
<Button variant="outline" size="sm" disabled={!selectedCustomerId} onClick={openCustomerEdit}>
<Pencil className="w-3.5 h-3.5 mr-1" />
</Button>
<Button variant="destructive" size="sm" disabled={!selectedCustomerId} onClick={handleCustomerDelete}>
<Trash2 className="w-3.5 h-3.5 mr-1" />
</Button>
<Button variant="outline" size="sm" onClick={() => ts.setOpen(true)}>
<Settings2 className="w-3.5 h-3.5" />
</Button>
</div>
</div>
{/* 거래처 테이블 */}
<EDataTable
columns={customerColumns}
data={ts.groupData(customers)}
rowKey={(row) => row.id}
loading={customerLoading}
emptyMessage="등록된 거래처가 없어요"
selectedId={selectedCustomerId}
onSelect={(id) => setSelectedCustomerId(id)}
onRowDoubleClick={(row) => { setSelectedCustomerId(row.id); openCustomerEdit(); }}
showRowNumber
showPagination={false}
draggableColumns={false}
columnOrderKey="c16-customer"
/>
</div>
</ResizablePanel>
<ResizableHandle withHandle />
{/* 우측: 디테일 패널 */}
<ResizablePanel defaultSize={45} minSize={25}>
<div className="flex flex-col h-full">
{!selectedCustomerId ? (
/* 빈 상태 */
<div className="flex-1 flex items-center justify-center p-5">
<div className="flex flex-col items-center justify-center text-center border-2 border-dashed border-border rounded-lg px-10 py-16">
<Users className="w-12 h-12 text-muted-foreground/40 mb-4" />
<div className="text-sm font-semibold text-muted-foreground mb-1.5"> </div>
<div className="text-xs text-muted-foreground"> </div>
</div>
</div>
) : (
<>
{/* 디테일 헤더 */}
<div className="flex items-center gap-3 px-4 py-3 border-b bg-muted shrink-0">
<span className="text-[13px] font-bold">{selectedCustomer?.customer_name || "-"}</span>
<span className="font-mono text-[11px] font-semibold text-primary bg-primary/[0.08] px-2 py-0.5 rounded-full">
{selectedCustomer?.customer_code || "-"}
</span>
</div>
{/* 탭 */}
<Tabs
value={rightTab}
onValueChange={(v) => setRightTab(v as "items" | "delivery")}
className="flex flex-col flex-1 overflow-hidden gap-0"
>
<TabsList className="bg-muted w-full justify-start rounded-none border-b border-border/50 p-0 h-auto shrink-0">
<TabsTrigger
value="items"
className="rounded-none border-b-2 border-transparent data-[state=active]:border-primary data-[state=active]:bg-transparent data-[state=active]:shadow-none px-4 py-2.5 text-xs font-semibold gap-1.5"
>
<Package className="w-3.5 h-3.5" />
</TabsTrigger>
<TabsTrigger
value="delivery"
className="rounded-none border-b-2 border-transparent data-[state=active]:border-primary data-[state=active]:bg-transparent data-[state=active]:shadow-none px-4 py-2.5 text-xs font-semibold gap-1.5"
>
<MapPin className="w-3.5 h-3.5" />
{deliveryItems.length > 0 && (
<Badge variant="secondary" className="ml-1 text-[10px] px-1.5 py-0">{deliveryItems.length}</Badge>
)}
</TabsTrigger>
</TabsList>
{/* 품목정보 탭 */}
<TabsContent value="items" className="flex flex-col flex-1 overflow-hidden mt-0">
<div className="flex items-center justify-between px-4 py-2.5 border-b shrink-0">
<span className="text-xs font-semibold text-muted-foreground">
<span className="text-primary">{priceItems.length}</span>
</span>
<div className="flex gap-1.5">
<Button
size="sm"
onClick={() => { setItemCheckedIds(new Set()); setItemSelectOpen(true); searchItems(); }}
>
<Plus className="w-3.5 h-3.5" />
</Button>
<Button variant="destructive" size="sm" disabled={priceCheckedIds.length === 0} onClick={handlePriceItemDelete}>
<Trash2 className="w-3.5 h-3.5" />
</Button>
</div>
</div>
<div className="flex-1 overflow-auto">
<Table noWrapper>
<TableHeader className="sticky top-0 z-10">
<TableRow className="bg-muted hover:bg-muted">
<TableHead className="w-[40px] text-center px-2">
<input
type="checkbox"
className="rounded"
checked={priceItems.length > 0 && priceCheckedIds.length === priceItems.length}
onChange={(e) => setPriceCheckedIds(e.target.checked ? priceItems.map((p) => p.id) : [])}
/>
</TableHead>
<TableHead className="w-[100px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="min-w-[100px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[100px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[100px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[80px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[80px] text-right text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[70px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[60px] text-right text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[80px] text-right text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[50px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{priceLoading ? (
<TableRow>
<TableCell colSpan={11} className="text-center py-8">
<Loader2 className="w-5 h-5 animate-spin mx-auto text-muted-foreground" />
</TableCell>
</TableRow>
) : priceItems.length === 0 ? (
<TableRow>
<TableCell colSpan={11} className="text-center py-8 text-muted-foreground text-sm">
</TableCell>
</TableRow>
) : priceItems.map((p) => (
<TableRow
key={p.id}
className={cn("cursor-pointer", priceCheckedIds.includes(p.id) && "bg-primary/5")}
onDoubleClick={() => openEditItem(p)}
>
<TableCell className="text-center px-2">
<input
type="checkbox"
className="rounded"
checked={priceCheckedIds.includes(p.id)}
onChange={(e) => {
if (e.target.checked) setPriceCheckedIds((prev) => [...prev, p.id]);
else setPriceCheckedIds((prev) => prev.filter((id) => id !== p.id));
}}
onClick={(e) => e.stopPropagation()}
/>
</TableCell>
<TableCell className="text-[13px] font-mono text-muted-foreground">{p.item_number}</TableCell>
<TableCell className="text-[13px]">{p.item_name}</TableCell>
<TableCell className="text-[13px]">{p.customer_item_code}</TableCell>
<TableCell className="text-[13px]">{p.customer_item_name}</TableCell>
<TableCell className="text-[13px]">{p.base_price_type}</TableCell>
<TableCell className="text-[13px] text-right">
{p.base_price ? Number(p.base_price).toLocaleString() : ""}
</TableCell>
<TableCell className="text-[13px]">{p.discount_type}</TableCell>
<TableCell className="text-[13px] text-right">{p.discount_value}</TableCell>
<TableCell className="text-[13px] text-right font-medium">
{p.calculated_price ? Number(p.calculated_price).toLocaleString() : ""}
</TableCell>
<TableCell className="text-[13px]">{p.currency_code}</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</div>
</TabsContent>
{/* 납품처 탭 */}
<TabsContent value="delivery" className="flex flex-col flex-1 overflow-hidden mt-0">
<div className="flex items-center justify-between px-4 py-2.5 border-b shrink-0">
<span className="text-xs font-semibold text-muted-foreground">
<span className="text-primary">{deliveryItems.length}</span>
</span>
<div className="flex gap-1.5">
<Button size="sm" onClick={() => { setDeliveryForm({}); setDeliveryModalOpen(true); }}>
<Plus className="w-3.5 h-3.5" />
</Button>
<Button variant="destructive" size="sm" disabled={deliveryCheckedIds.length === 0} onClick={handleDeliveryDelete}>
<Trash2 className="w-3.5 h-3.5" />
</Button>
</div>
</div>
<div className="flex-1 overflow-auto">
<Table noWrapper>
<TableHeader className="sticky top-0 z-10">
<TableRow className="bg-muted hover:bg-muted">
<TableHead className="w-[40px] text-center px-2">
<input
type="checkbox"
className="rounded"
checked={deliveryItems.length > 0 && deliveryCheckedIds.length === deliveryItems.length}
onChange={(e) => setDeliveryCheckedIds(e.target.checked ? deliveryItems.map((d) => d.id) : [])}
/>
</TableHead>
<TableHead className="w-[110px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="min-w-[130px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="min-w-[150px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[80px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[110px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[100px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[50px] text-center text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{deliveryLoading ? (
<TableRow>
<TableCell colSpan={8} className="text-center py-8">
<Loader2 className="w-5 h-5 animate-spin mx-auto text-muted-foreground" />
</TableCell>
</TableRow>
) : deliveryItems.length === 0 ? (
<TableRow>
<TableCell colSpan={8} className="text-center py-8 text-muted-foreground text-sm">
</TableCell>
</TableRow>
) : deliveryItems.map((d) => (
<TableRow
key={d.id}
className={cn("cursor-pointer", deliveryCheckedIds.includes(d.id) && "bg-primary/5")}
>
<TableCell className="text-center px-2">
<input
type="checkbox"
className="rounded"
checked={deliveryCheckedIds.includes(d.id)}
onChange={(e) => {
if (e.target.checked) setDeliveryCheckedIds((prev) => [...prev, d.id]);
else setDeliveryCheckedIds((prev) => prev.filter((id) => id !== d.id));
}}
onClick={(e) => e.stopPropagation()}
/>
</TableCell>
<TableCell className="text-[13px] font-mono text-muted-foreground">{d.destination_code}</TableCell>
<TableCell className="text-sm">{d.destination_name}</TableCell>
<TableCell className="text-[13px] text-muted-foreground">{d.address}</TableCell>
<TableCell className="text-[13px]">{d.manager_name}</TableCell>
<TableCell className="text-[13px]">{d.phone}</TableCell>
<TableCell className="text-[13px] text-muted-foreground">{d.memo}</TableCell>
<TableCell className="text-center">
{d.is_default && (
<Badge variant="default" className="text-[10px] px-1.5 py-0 h-5"></Badge>
)}
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</div>
</TabsContent>
</Tabs>
</>
)}
</div>
</ResizablePanel>
</ResizablePanelGroup>
</div>
{/* ── 모달: 거래처 등록/수정 ── */}
<Dialog open={customerModalOpen} onOpenChange={setCustomerModalOpen}>
<DialogContent className="max-w-2xl">
<DialogHeader>
<DialogTitle>{customerEditMode ? "거래처 수정" : "거래처 등록"}</DialogTitle>
<DialogDescription>
{customerEditMode ? "거래처 정보를 수정합니다." : "새로운 거래처를 등록합니다."}
</DialogDescription>
</DialogHeader>
<div className="grid grid-cols-2 gap-4 py-2">
<div className="space-y-1.5">
<Label className="text-sm"> </Label>
<Input
value={customerForm.customer_code || ""}
onChange={(e) => setCustomerForm((p) => ({ ...p, customer_code: e.target.value }))}
placeholder="거래처 코드"
className="h-9"
disabled={customerEditMode}
/>
</div>
<div className="space-y-1.5">
<Label className="text-sm"> <span className="text-destructive">*</span></Label>
<Input
value={customerForm.customer_name || ""}
onChange={(e) => setCustomerForm((p) => ({ ...p, customer_name: e.target.value }))}
placeholder="거래처명"
className="h-9"
/>
</div>
<div className="space-y-1.5">
<Label className="text-sm"> </Label>
<Select
value={customerForm.division || "__none__"}
onValueChange={(v) => setCustomerForm((p) => ({ ...p, division: v === "__none__" ? "" : v }))}
>
<SelectTrigger className="h-9"><SelectValue placeholder="거래 유형" /></SelectTrigger>
<SelectContent>
<SelectItem value="__none__"> </SelectItem>
{(categoryOptions["division"] || []).map((o) => (
<SelectItem key={o.code} value={o.code}>{o.label}</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="space-y-1.5">
<Label className="text-sm"> <span className="text-destructive">*</span></Label>
<Select
value={customerForm.status || "__none__"}
onValueChange={(v) => setCustomerForm((p) => ({ ...p, status: v === "__none__" ? "" : v }))}
>
<SelectTrigger className="h-9"><SelectValue placeholder="상태" /></SelectTrigger>
<SelectContent>
<SelectItem value="__none__"> </SelectItem>
{(categoryOptions["status"] || []).map((o) => (
<SelectItem key={o.code} value={o.code}>{o.label}</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Input
value={customerForm.contact_person || ""}
onChange={(e) => setCustomerForm((p) => ({ ...p, contact_person: e.target.value }))}
placeholder="거래처담당자"
className="h-9"
/>
</div>
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Select
value={customerForm.internal_manager || "__none__"}
onValueChange={(v) => setCustomerForm((p) => ({ ...p, internal_manager: v === "__none__" ? "" : v }))}
>
<SelectTrigger className="h-9"><SelectValue placeholder="사내담당자 선택" /></SelectTrigger>
<SelectContent>
<SelectItem value="__none__"> </SelectItem>
{employeeOptions.map((emp) => (
<SelectItem key={emp.user_id} value={emp.user_id}>
{emp.user_name}{emp.position_name ? ` (${emp.position_name})` : ""}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Input
value={customerForm.contact_phone || ""}
onChange={(e) => handleFormChange("contact_phone", e.target.value)}
placeholder="010-0000-0000"
className={cn("h-9", formErrors.contact_phone && "border-destructive")}
/>
{formErrors.contact_phone && <p className="text-xs text-destructive">{formErrors.contact_phone}</p>}
</div>
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Input
value={customerForm.email || ""}
onChange={(e) => handleFormChange("email", e.target.value)}
placeholder="example@email.com"
className={cn("h-9", formErrors.email && "border-destructive")}
/>
{formErrors.email && <p className="text-xs text-destructive">{formErrors.email}</p>}
</div>
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Input
value={customerForm.business_number || ""}
onChange={(e) => handleFormChange("business_number", e.target.value)}
placeholder="000-00-00000"
className={cn("h-9", formErrors.business_number && "border-destructive")}
/>
{formErrors.business_number && <p className="text-xs text-destructive">{formErrors.business_number}</p>}
</div>
<div className="space-y-1.5 col-span-2">
<Label className="text-sm"></Label>
<Input
value={customerForm.address || ""}
onChange={(e) => setCustomerForm((p) => ({ ...p, address: e.target.value }))}
placeholder="주소"
className="h-9"
/>
</div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setCustomerModalOpen(false)}></Button>
<Button onClick={handleCustomerSave} disabled={saving}>
{saving ? <Loader2 className="w-4 h-4 mr-1.5 animate-spin" /> : <Save className="w-4 h-4 mr-1.5" />}
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
{/* ── 모달: 품목 선택 (1단계) ── */}
<Dialog open={itemSelectOpen} onOpenChange={setItemSelectOpen}>
<DialogContent className="max-w-3xl">
<DialogHeader>
<DialogTitle> </DialogTitle>
<DialogDescription> .</DialogDescription>
</DialogHeader>
<div className="flex gap-2 mb-3">
<Input
placeholder="품명/품목코드 검색"
value={itemSearchKeyword}
onChange={(e) => setItemSearchKeyword(e.target.value)}
onKeyDown={(e) => e.key === "Enter" && searchItems()}
className="h-9 flex-1"
/>
<Button size="sm" onClick={searchItems} disabled={itemSearchLoading} className="h-9">
{itemSearchLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : <><Search className="w-4 h-4 mr-1" /> </>}
</Button>
</div>
<div className="overflow-auto max-h-[350px] border rounded-lg">
<Table noWrapper>
<TableHeader className="sticky top-0 z-10">
<TableRow className="bg-muted hover:bg-muted">
<TableHead className="w-[40px] text-center px-2">
<input
type="checkbox"
className="rounded"
checked={itemSearchResults.length > 0 && itemCheckedIds.size === itemSearchResults.length}
onChange={(e) => {
if (e.target.checked) setItemCheckedIds(new Set(itemSearchResults.map((i) => i.id)));
else setItemCheckedIds(new Set());
}}
/>
</TableHead>
<TableHead className="w-[120px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="min-w-[140px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[100px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[100px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
<TableHead className="w-[60px] text-[11px] font-bold uppercase tracking-wide text-muted-foreground"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{itemSearchResults.length === 0 ? (
<TableRow>
<TableCell colSpan={6} className="text-center text-muted-foreground py-8 text-sm">
</TableCell>
</TableRow>
) : itemSearchResults.map((item) => (
<TableRow
key={item.id}
className={cn("cursor-pointer", itemCheckedIds.has(item.id) && "bg-primary/5")}
onClick={() => setItemCheckedIds((prev) => {
const next = new Set(prev);
if (next.has(item.id)) next.delete(item.id); else next.add(item.id);
return next;
})}
>
<TableCell className="text-center px-2">
<input type="checkbox" className="rounded" checked={itemCheckedIds.has(item.id)} readOnly />
</TableCell>
<TableCell className="text-[13px] font-mono">
<span className="block truncate" title={item.item_number}>{item.item_number}</span>
</TableCell>
<TableCell className="text-sm">{item.item_name}</TableCell>
<TableCell className="text-[13px] text-muted-foreground">{item.size}</TableCell>
<TableCell className="text-[13px] text-muted-foreground">{item.material}</TableCell>
<TableCell className="text-[13px] text-muted-foreground">{item.unit}</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</div>
<DialogFooter className="flex items-center justify-between w-full">
<span className="text-sm text-muted-foreground">{itemCheckedIds.size} </span>
<div className="flex gap-2">
<Button variant="outline" onClick={() => setItemSelectOpen(false)}></Button>
<Button onClick={goToItemDetail} disabled={itemCheckedIds.size === 0}>
<Plus className="w-4 h-4 mr-1" /> {itemCheckedIds.size}
</Button>
</div>
</DialogFooter>
</DialogContent>
</Dialog>
{/* ── 모달: 품목 상세 입력 (2단계) ── */}
<Dialog open={itemDetailOpen} onOpenChange={setItemDetailOpen}>
<DialogContent className="max-w-[1100px] max-h-[90vh] overflow-y-auto">
<DialogHeader>
<DialogTitle>
{editItemData ? "수정" : "입력"} {selectedCustomer?.customer_name || ""}
</DialogTitle>
<DialogDescription>
{editItemData
? "거래처 품번/품명과 기간별 단가를 수정합니다."
: "선택한 품목의 거래처 품번/품명과 기간별 단가를 설정합니다."}
</DialogDescription>
</DialogHeader>
<div className="space-y-6 py-2">
{selectedItemsForDetail.map((item, idx) => {
const itemKey = item.item_number || item.id;
const mappingRows = itemMappings[itemKey] || [];
const prices = itemPrices[itemKey] || [];
return (
<div key={itemKey} className="border rounded-xl overflow-hidden bg-card">
{/* 품목 헤더 */}
<div className="px-5 py-3 bg-muted/30 border-b">
<div className="font-bold">{idx + 1}. {item.item_name || itemKey}</div>
<div className="text-xs text-muted-foreground">{itemKey} | {item.size || ""} | {item.unit || ""}</div>
</div>
<div className="flex gap-4 p-4">
{/* 좌: 거래처 품번/품명 */}
<div className="flex-1 border rounded-lg p-4 bg-muted/30">
<div className="flex items-center justify-between mb-3">
<span className="text-sm font-semibold flex items-center gap-1.5">
<Tag className="w-3.5 h-3.5 text-primary" /> /
</span>
<Button variant="outline" size="sm" className="h-7 text-xs" onClick={() => addMappingRow(itemKey)}>
<Plus className="h-3 w-3 mr-1" />
</Button>
</div>
<div className="space-y-2">
{mappingRows.length === 0 ? (
<div className="text-xs text-muted-foreground py-2"> </div>
) : (
<>
<div className="flex gap-2 items-center text-[10px] text-muted-foreground font-medium">
<span className="w-4 shrink-0"></span>
<span className="flex-1"> </span>
<span className="flex-1"> </span>
<span className="w-7 shrink-0"></span>
</div>
{mappingRows.map((mRow, mIdx) => (
<div key={mRow._id} className="flex gap-2 items-center">
<span className="text-xs text-muted-foreground w-4 shrink-0">{mIdx + 1}</span>
<Input
value={mRow.customer_item_code}
onChange={(e) => updateMappingRow(itemKey, mRow._id, "customer_item_code", e.target.value)}
placeholder="거래처 품번"
className="h-8 text-sm flex-1"
/>
<Input
value={mRow.customer_item_name}
onChange={(e) => updateMappingRow(itemKey, mRow._id, "customer_item_name", e.target.value)}
placeholder="거래처 품명"
className="h-8 text-sm flex-1"
/>
<Button
variant="ghost" size="sm" className="h-7 w-7 p-0 text-destructive shrink-0"
onClick={() => removeMappingRow(itemKey, mRow._id)}
>
<X className="h-3 w-3" />
</Button>
</div>
))}
</>
)}
</div>
</div>
{/* 우: 기간별 단가 */}
<div className="flex-1 border rounded-lg p-4 bg-muted/30">
<div className="flex items-center justify-between mb-3">
<span className="text-sm font-semibold flex items-center gap-1.5">
<Coins className="w-3.5 h-3.5 text-primary" />
</span>
<Button variant="outline" size="sm" className="h-7 text-xs" onClick={() => addPriceRow(itemKey)}>
<Plus className="h-3 w-3 mr-1" />
</Button>
</div>
<div className="space-y-3">
{prices.map((price, pIdx) => (
<div key={price._id} className="border rounded-lg p-3 bg-background space-y-2">
<div className="flex items-center justify-between">
<span className="text-xs font-medium text-muted-foreground"> {pIdx + 1}</span>
{prices.length > 1 && (
<Button
variant="ghost" size="sm" className="h-6 w-6 p-0 text-destructive"
onClick={() => removePriceRow(itemKey, price._id)}
>
<X className="h-3 w-3" />
</Button>
)}
</div>
{/* 기간 + 통화 */}
<div className="flex gap-2 items-center">
<Input
type="date"
value={price.start_date}
onChange={(e) => updatePriceRow(itemKey, price._id, "start_date", e.target.value)}
className="h-8 text-xs flex-1"
/>
<span className="text-xs text-muted-foreground">~</span>
<Input
type="date"
value={price.end_date}
onChange={(e) => updatePriceRow(itemKey, price._id, "end_date", e.target.value)}
className="h-8 text-xs flex-1"
/>
<div className="w-[80px]">
<Select
value={price.currency_code}
onValueChange={(v) => updatePriceRow(itemKey, price._id, "currency_code", v)}
>
<SelectTrigger className="h-8 text-xs"><SelectValue placeholder="통화" /></SelectTrigger>
<SelectContent>
{(priceCategoryOptions["currency_code"] || []).map((o) => (
<SelectItem key={o.code} value={o.code}>{o.label}</SelectItem>
))}
</SelectContent>
</Select>
</div>
</div>
{/* 기준가/할인/반올림 */}
<div className="flex gap-2 items-center">
<div className="w-[90px]">
<Select
value={price.base_price_type}
onValueChange={(v) => updatePriceRow(itemKey, price._id, "base_price_type", v)}
>
<SelectTrigger className="h-8 text-xs"><SelectValue placeholder="기준" /></SelectTrigger>
<SelectContent>
{(priceCategoryOptions["base_price_type"] || []).map((o) => (
<SelectItem key={o.code} value={o.code}>{o.label}</SelectItem>
))}
</SelectContent>
</Select>
</div>
<Input
value={price.base_price}
onChange={(e) => updatePriceRow(itemKey, price._id, "base_price", e.target.value)}
className="h-8 text-xs text-right flex-1"
placeholder="기준가"
/>
<div className="w-[90px]">
<Select
value={price.discount_type}
onValueChange={(v) => updatePriceRow(itemKey, price._id, "discount_type", v)}
>
<SelectTrigger className="h-8 text-xs"><SelectValue placeholder="할인" /></SelectTrigger>
<SelectContent>
<SelectItem value="none"></SelectItem>
{(priceCategoryOptions["discount_type"] || []).map((o) => (
<SelectItem key={o.code} value={o.code}>{o.label}</SelectItem>
))}
</SelectContent>
</Select>
</div>
<Input
value={price.discount_value}
onChange={(e) => updatePriceRow(itemKey, price._id, "discount_value", e.target.value)}
className="h-8 text-xs text-right w-[60px]"
placeholder="0"
/>
<div className="w-[90px]">
<Select
value={price.rounding_unit_value}
onValueChange={(v) => updatePriceRow(itemKey, price._id, "rounding_unit_value", v)}
>
<SelectTrigger className="h-8 text-xs"><SelectValue placeholder="반올림" /></SelectTrigger>
<SelectContent>
{(priceCategoryOptions["rounding_unit_value"] || []).map((o) => (
<SelectItem key={o.code} value={o.code}>{o.label}</SelectItem>
))}
</SelectContent>
</Select>
</div>
</div>
{/* 계산 단가 */}
<div className="flex items-center justify-end gap-2 pt-1 border-t">
<span className="text-xs text-muted-foreground"> :</span>
<span className="font-bold text-sm">
{price.calculated_price ? Number(price.calculated_price).toLocaleString() : "-"}
</span>
</div>
</div>
))}
</div>
</div>
</div>
</div>
);
})}
</div>
<DialogFooter>
<Button
variant="outline"
onClick={() => {
setItemDetailOpen(false);
if (!editItemData) setItemSelectOpen(true);
setEditItemData(null);
}}
>
{editItemData ? "취소" : "← 이전"}
</Button>
<Button onClick={handleItemDetailSave} disabled={saving}>
{saving ? <Loader2 className="w-4 h-4 mr-1.5 animate-spin" /> : <Save className="w-4 h-4 mr-1.5" />}
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
{/* ── 모달: 납품처 등록 ── */}
<Dialog open={deliveryModalOpen} onOpenChange={setDeliveryModalOpen}>
<DialogContent className="max-w-lg">
<DialogHeader>
<DialogTitle> </DialogTitle>
<DialogDescription>{selectedCustomer?.customer_name || ""} .</DialogDescription>
</DialogHeader>
<div className="grid grid-cols-2 gap-4 py-2">
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Input
value={deliveryForm.destination_code || ""}
onChange={(e) => setDeliveryForm((p) => ({ ...p, destination_code: e.target.value }))}
placeholder="납품처코드"
className="h-9"
/>
</div>
<div className="space-y-1.5">
<Label className="text-sm"> <span className="text-destructive">*</span></Label>
<Input
value={deliveryForm.destination_name || ""}
onChange={(e) => setDeliveryForm((p) => ({ ...p, destination_name: e.target.value }))}
placeholder="납품처명"
className="h-9"
/>
</div>
<div className="space-y-1.5 col-span-2">
<Label className="text-sm"></Label>
<Input
value={deliveryForm.address || ""}
onChange={(e) => setDeliveryForm((p) => ({ ...p, address: e.target.value }))}
placeholder="주소"
className="h-9"
/>
</div>
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Input
value={deliveryForm.manager_name || ""}
onChange={(e) => setDeliveryForm((p) => ({ ...p, manager_name: e.target.value }))}
placeholder="담당자"
className="h-9"
/>
</div>
<div className="space-y-1.5">
<Label className="text-sm"></Label>
<Input
value={deliveryForm.phone || ""}
onChange={(e) => {
const formatted = formatField("phone", e.target.value);
setDeliveryForm((p) => ({ ...p, phone: formatted }));
const err = validateField("phone", formatted);
setFormErrors((p) => {
const n = { ...p };
if (err) n.delivery_phone = err; else delete n.delivery_phone;
return n;
});
}}
placeholder="010-0000-0000"
className={cn("h-9", formErrors.delivery_phone && "border-destructive")}
/>
{formErrors.delivery_phone && <p className="text-xs text-destructive">{formErrors.delivery_phone}</p>}
</div>
<div className="space-y-1.5 col-span-2">
<Label className="text-sm"></Label>
<Input
value={deliveryForm.memo || ""}
onChange={(e) => setDeliveryForm((p) => ({ ...p, memo: e.target.value }))}
placeholder="메모"
className="h-9"
/>
</div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setDeliveryModalOpen(false)}></Button>
<Button onClick={handleDeliverySave}><Save className="w-4 h-4 mr-1.5" /> </Button>
</DialogFooter>
</DialogContent>
</Dialog>
{/* 엑셀 업로드 (멀티테이블) */}
{excelChainConfig && (
<MultiTableExcelUploadModal
open={excelUploadOpen}
onOpenChange={(open) => {
setExcelUploadOpen(open);
if (!open) setExcelChainConfig(null);
}}
config={excelChainConfig}
onSuccess={() => {
fetchCustomers();
const cid = selectedCustomerId;
setSelectedCustomerId(null);
setTimeout(() => setSelectedCustomerId(cid), 50);
}}
/>
)}
{/* 테이블 설정 모달 */}
<TableSettingsModal
open={ts.open}
onOpenChange={ts.setOpen}
tableName={ts.tableName}
settingsId={ts.settingsId}
defaultVisibleKeys={ts.defaultVisibleKeys}
onSave={ts.applySettings}
/>
{ConfirmDialogComponent}
</div>
);
}