Files
hjjeong 6368258797 생산관리 M-BOM PR-B5+ — BOM 복사 다이얼로그 (운영 structureBomCopyFormPopup 1:1)
5개 메뉴 1:1 검증(Agent 5병렬) 후 발견된 mbom 메인 그리드 [BOM 복사] 누락 보완.
저장 매퍼는 PR-B5 saveBomAssignment 재사용 — 백엔드는 EBOM/MBOM 분기 이미 지원,
이번에 MBOM UI 진입점도 첫 노출.

신규 컴포넌트:
 - BomCopyDialog.tsx — 품번/품명 readonly + E-BOM/M-BOM 셀렉트 상호배타 +
   트리 미리보기 + AttachFileDropZone (docType=MBOM_DRAWING, accept=.stp,.step,.dwg,.dxf,.pdf)
 - 메인 page.tsx [BOM 복사] 버튼 추가 + 체크 단건 검증 +
   Machine(0000928) 외 동일 partNo 최신 M-BOM 자동 추천

신규 백엔드 (mbomService/Controller/Routes):
 - searchAssignableMboms — 매퍼 productionplanning.getMbomListForSelect2 (4007~4014) 1:1
 - previewMbomTree     — getStructureOnly + finalize("ASSIGNED_MBOM")
 - getLatestMbomByPartNo — 매퍼 getLatestMbomByPartNo (3426~3445) 1:1, Machine 외 자동 검색
 - 라우트: GET /assignable-mboms, GET /mbom-preview/:objid, GET /latest-mbom-by-partno/:partNo

도면 업로드 차이: 운영 fn_uploadDrawingFiles 는 placeholder("구현 예정"). RPS 는
공용 AttachFileDropZone 재사용해 실구현 (target_objid=projectObjid).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-15 10:53:35 +09:00

354 lines
15 KiB
TypeScript

"use client";
// 생산관리 > M-BOM 관리 — BOM 복사 다이얼로그 (PR-B5+).
//
// 운영판 partMng/structureBomCopyFormPopup.jsp (774 lines) 1:1 재구성.
// 진입: 메인 그리드 [BOM 복사] 버튼 (체크박스 1개 선택).
// 저장: /production/mbom/assign (saveBomAssignment) — MbomAssignDialog 와 동일 매퍼.
// 차이점: E-BOM/M-BOM 셀렉트 두 개 (상호배타) + 트리 미리보기 + 도면 다중 업로드.
//
// 운영판 fn_uploadDrawingFiles 는 placeholder ("구현 예정") — RPS 는 공용 AttachFileDropZone
// 재사용으로 실구현 (target_objid=projectObjid, docType="MBOM_DRAWING").
import React, { useEffect, useMemo, useState } from "react";
import {
Dialog, DialogContent, DialogHeader, DialogTitle,
} from "@/components/ui/dialog";
import { Button } from "@/components/ui/button";
import { Input } from "@/components/ui/input";
import { Loader2, Copy, Folder, X as XIcon } from "lucide-react";
import { toast } from "sonner";
import { cn } from "@/lib/utils";
import {
mbomApi,
AssignableEbomRow,
AssignableMbomRow,
MbomTreeResponse,
MbomTreeRow,
LatestMbomByPartNoRow,
} from "@/lib/api/mbom";
import { AttachFileDropZone } from "@/components/common/AttachFileDropZone";
type SourceType = "EBOM" | "MBOM";
const MACHINE_PRODUCT_CD = "0000928";
interface Props {
open: boolean;
onOpenChange: (open: boolean) => void;
projectObjid: string | null;
partNo: string | null;
partName: string | null;
productCode?: string | null; // contract_mgmt.product — Machine 외 자동 추천 분기용
hasMbom?: boolean; // 이미 저장된 M-BOM 이 있으면 "초기화" 경고
onSaved: () => void;
}
export function BomCopyDialog({
open, onOpenChange, projectObjid, partNo, partName, productCode, hasMbom, onSaved,
}: Props) {
// 셀렉트 옵션 + 선택 상태
const [ebomOpts, setEbomOpts] = useState<AssignableEbomRow[]>([]);
const [mbomOpts, setMbomOpts] = useState<AssignableMbomRow[]>([]);
const [ebomLoading, setEbomLoading] = useState(false);
const [mbomLoading, setMbomLoading] = useState(false);
const [selectedType, setSelectedType] = useState<SourceType | null>(null);
const [selectedObjid, setSelectedObjid] = useState<string>("");
// 미리보기 트리
const [preview, setPreview] = useState<MbomTreeResponse | null>(null);
const [previewLoading, setPreviewLoading] = useState(false);
// 저장 진행 + 자동 추천 안내 1회 가드
const [saving, setSaving] = useState(false);
const autoCheckedRef = React.useRef(false);
// ── 오픈 시 초기화 + 옵션 로드 + Machine 외 자동 추천 ──
useEffect(() => {
if (!open) {
// 닫힐 때 상태 초기화
setSelectedType(null);
setSelectedObjid("");
setPreview(null);
autoCheckedRef.current = false;
return;
}
// E-BOM 옵션
setEbomLoading(true);
mbomApi.searchAssignableEboms({ limit: 500 })
.then(setEbomOpts)
.catch((e: any) => toast.error(e?.response?.data?.message ?? e?.message ?? "E-BOM 옵션 조회 실패"))
.finally(() => setEbomLoading(false));
// M-BOM 옵션
setMbomLoading(true);
mbomApi.searchAssignableMboms({ limit: 500 })
.then(setMbomOpts)
.catch((e: any) => toast.error(e?.response?.data?.message ?? e?.message ?? "M-BOM 옵션 조회 실패"))
.finally(() => setMbomLoading(false));
}, [open]);
// Machine 이외 제품 + partNo 있으면 자동 추천 (운영 fn_checkExistingMbom)
useEffect(() => {
if (!open) return;
if (autoCheckedRef.current) return;
if (!partNo || !partNo.trim()) return;
const isMachine = (productCode ?? "").trim() === MACHINE_PRODUCT_CD;
if (isMachine) return;
autoCheckedRef.current = true;
mbomApi.getLatestMbomByPartNo(partNo)
.then((row) => {
if (!row) return;
const ok = window.confirm(
`동일 품번(${partNo})의 M-BOM이 이미 존재합니다.\n` +
`M-BOM 품번: ${row.mbom_no ?? ""}\n저장일: ${row.save_date ?? ""}\n\n` +
`기존 M-BOM 을 자동으로 불러오시겠습니까?`
);
if (ok) {
setSelectedType("MBOM");
setSelectedObjid(String(row.template_header_objid));
}
})
.catch(() => { /* 실패 시 조용히 무시 — 자동 추천은 부가기능 */ });
}, [open, partNo, productCode]);
// ── 선택 변경 시 트리 미리보기 ────────────────────────────
useEffect(() => {
if (!selectedType || !selectedObjid) {
setPreview(null);
return;
}
setPreviewLoading(true);
const fetcher = selectedType === "EBOM"
? mbomApi.previewEbomTree(selectedObjid)
: mbomApi.previewMbomTree(selectedObjid);
fetcher
.then(setPreview)
.catch((e: any) => toast.error(e?.response?.data?.message ?? e?.message ?? "트리 미리보기 실패"))
.finally(() => setPreviewLoading(false));
}, [selectedType, selectedObjid]);
const previewTitle = useMemo(() => {
if (!selectedType || !selectedObjid) return "BOM 데이터를 조회하려면 E-BOM 또는 M-BOM 을 선택하세요.";
if (selectedType === "EBOM") {
const r = ebomOpts.find(o => String(o.objid) === String(selectedObjid));
return r ? `${r.part_no ?? ""} - ${r.part_name ?? ""}${r.revision ? ` (Rev.${r.revision})` : ""}` : "";
}
const r = mbomOpts.find(o => String(o.objid) === String(selectedObjid));
return r ? `${r.mbom_no ?? ""}${r.part_name ? ` - ${r.part_name}` : ""}` : "";
}, [selectedType, selectedObjid, ebomOpts, mbomOpts]);
// ── 저장 (운영 fn_saveBomCopy → saveBomAssignment.do) ─────
const handleSave = async () => {
if (!projectObjid) { toast.error("프로젝트가 선택되지 않았습니다."); return; }
if (!partNo || !partName) { toast.error("품번과 품명이 비어 있습니다."); return; }
if (!selectedType || !selectedObjid) { toast.error("복사할 BOM 을 선택하세요."); return; }
if (!preview || preview.rows.length === 0) { toast.error("복사할 BOM 데이터가 없습니다."); return; }
if (hasMbom) {
const ok = window.confirm("저장된 M-BOM 이 초기화 됩니다.\n계속하시겠습니까?");
if (!ok) return;
}
const ok = window.confirm(
`선택한 ${selectedType} 을(를) M-BOM 기준으로 할당하시겠습니까?\n` +
`실제 M-BOM 생성은 M-BOM 상세 팝업에서 저장 시 이루어집니다.`
);
if (!ok) return;
try {
setSaving(true);
await mbomApi.assignBom({
project_obj_id: projectObjid,
source_bom_type: selectedType,
source_bom_obj_id: selectedObjid,
});
toast.success("BOM 할당 정보가 저장되었습니다.");
onSaved();
onOpenChange(false);
} catch (e: any) {
toast.error(e?.response?.data?.message ?? e?.message ?? "BOM 할당 실패");
} finally {
setSaving(false);
}
};
return (
<Dialog open={open} onOpenChange={onOpenChange}>
<DialogContent className="max-w-[1400px] w-[95vw] max-h-[92vh] overflow-hidden flex flex-col p-0">
<DialogHeader className="px-6 pt-5 pb-3 border-b">
<DialogTitle className="flex items-center gap-2 text-base">
<Copy className="h-4 w-4 text-blue-600" />
BOM
</DialogTitle>
</DialogHeader>
<div className="flex-1 overflow-auto px-6 py-4 space-y-4">
{/* 상단: 품번/품명 readonly + [저장] [닫기] (운영 1:1) */}
<div className="grid grid-cols-12 gap-3 items-end">
<div className="col-span-3">
<label className="text-xs text-muted-foreground"></label>
<Input value={partNo ?? ""} readOnly className="bg-muted h-9" />
</div>
<div className="col-span-5">
<label className="text-xs text-muted-foreground"></label>
<Input value={partName ?? ""} readOnly className="bg-muted h-9" />
</div>
<div className="col-span-4 flex gap-2 justify-end">
<Button variant="outline" onClick={() => onOpenChange(false)} disabled={saving}>
<XIcon className="h-4 w-4 mr-1" />
</Button>
<Button onClick={handleSave} disabled={saving || !selectedType || !selectedObjid}>
{saving ? <Loader2 className="h-4 w-4 mr-1 animate-spin" /> : <Copy className="h-4 w-4 mr-1" />}
</Button>
</div>
</div>
{/* 중단: E-BOM / M-BOM 셀렉트 (상호배타) */}
<div className="grid grid-cols-12 gap-3">
<div className="col-span-2 flex items-center text-sm font-medium">E-BOM </div>
<div className="col-span-10">
<select
className="h-9 w-full border rounded px-2 text-sm bg-background disabled:bg-muted disabled:opacity-60"
disabled={ebomLoading || selectedType === "MBOM"}
value={selectedType === "EBOM" ? selectedObjid : ""}
onChange={(e) => {
const v = e.target.value;
if (!v) { setSelectedType(null); setSelectedObjid(""); return; }
setSelectedType("EBOM");
setSelectedObjid(v);
}}
>
<option value="">{ebomLoading ? "로딩…" : "선택"}</option>
{ebomOpts.map(o => (
<option key={o.objid} value={o.objid}>
{(o.part_no ?? "")} - {(o.part_name ?? "")}{o.revision ? ` (Rev.${o.revision})` : ""}
</option>
))}
</select>
</div>
<div className="col-span-2 flex items-center text-sm font-medium">M-BOM </div>
<div className="col-span-10">
<select
className="h-9 w-full border rounded px-2 text-sm bg-background disabled:bg-muted disabled:opacity-60"
disabled={mbomLoading || selectedType === "EBOM"}
value={selectedType === "MBOM" ? selectedObjid : ""}
onChange={(e) => {
const v = e.target.value;
if (!v) { setSelectedType(null); setSelectedObjid(""); return; }
setSelectedType("MBOM");
setSelectedObjid(v);
}}
>
<option value="">{mbomLoading ? "로딩…" : "선택"}</option>
{mbomOpts.map(o => (
<option key={o.objid} value={o.objid}>
{(o.mbom_no ?? "")}{o.part_name ? ` - ${o.part_name}` : ""}
</option>
))}
</select>
</div>
</div>
{/* 하단: 미리보기 트리 + 도면 업로드 */}
<div className="border rounded">
<div className="flex items-center justify-between px-3 py-2 bg-muted/50 border-b text-sm">
<span className="font-medium truncate">{previewTitle}</span>
<span className="text-xs text-muted-foreground">
{previewLoading ? "로딩…" : preview ? `${preview.rows.length} 행 · 최대 ${preview.max_level} 레벨` : ""}
</span>
</div>
<PreviewTree preview={preview} loading={previewLoading} />
</div>
{/* 도면 다중 업로드 (운영판 fn_uploadDrawingFiles placeholder → RPS 실구현) */}
<div className="border rounded p-3">
<div className="flex items-center justify-between mb-2">
<span className="text-sm font-medium flex items-center gap-1">
<Folder className="h-4 w-4 text-amber-600" />
</span>
<span className="text-xs text-muted-foreground">.stp .step .dwg .dxf .pdf</span>
</div>
{projectObjid ? (
<AttachFileDropZone
targetObjid={projectObjid}
docType="MBOM_DRAWING"
docTypeName="M-BOM 도면"
accept=".stp,.step,.dwg,.dxf,.pdf"
/>
) : (
<div className="text-xs text-muted-foreground"> .</div>
)}
</div>
</div>
</DialogContent>
</Dialog>
);
}
// ─── 트리 미리보기 (간단 그리드) ───────────────────────────────
function PreviewTree({
preview, loading,
}: { preview: MbomTreeResponse | null; loading: boolean }) {
if (loading) {
return (
<div className="p-8 text-center text-sm text-muted-foreground">
<Loader2 className="h-5 w-5 animate-spin inline mr-1" />
</div>
);
}
if (!preview || preview.rows.length === 0) {
return (
<div className="p-8 text-center text-sm text-muted-foreground"> BOM .</div>
);
}
const maxLevel = preview.max_level || 1;
return (
<div className="overflow-auto max-h-[40vh]">
<table className="min-w-full text-xs">
<thead className="sticky top-0 bg-muted/80 border-b">
<tr>
<th className="px-2 py-1 text-center" style={{ width: 50 }}>#</th>
{Array.from({ length: maxLevel }).map((_, i) => (
<th key={i} className="px-1 py-1 text-center" style={{ width: 26 }}>{i + 1}</th>
))}
<th className="px-2 py-1 text-left"></th>
<th className="px-2 py-1 text-left"></th>
<th className="px-2 py-1 text-right" style={{ width: 80 }}></th>
<th className="px-2 py-1 text-left"></th>
<th className="px-2 py-1 text-left"></th>
</tr>
</thead>
<tbody>
{preview.rows.map((r: MbomTreeRow, idx: number) => (
<tr key={`${r.objid ?? ""}-${idx}`} className={cn("border-b", levelBg(r.level))}>
<td className="px-2 py-1 text-center text-muted-foreground">{idx + 1}</td>
{Array.from({ length: maxLevel }).map((_, i) => (
<td key={i} className="px-1 py-1 text-center">
{Number(r.level) === i + 1 ? "*" : ""}
</td>
))}
<td className="px-2 py-1 font-mono">{r.part_no ?? ""}</td>
<td className="px-2 py-1">{r.part_name ?? ""}</td>
<td className="px-2 py-1 text-right">{r.item_qty ?? r.qty ?? ""}</td>
<td className="px-2 py-1">{r.spec ?? ""}</td>
<td className="px-2 py-1">{r.material ?? ""}</td>
</tr>
))}
</tbody>
</table>
</div>
);
}
function levelBg(level: number | string | null | undefined): string {
const n = Number(level) || 0;
switch (n) {
case 1: return "bg-white";
case 2: return "bg-blue-50";
case 3: return "bg-amber-50";
case 4: return "bg-emerald-50";
case 5: return "bg-rose-50";
default: return "";
}
}