20a429eecb
사용자 검증으로 발견된 5가지 함정 일괄 정정.
(1) ebom-search 검색폼 운영판 1:1 — wace structureAscendingList.jsp 노출 필드만:
- 제거: 프로젝트 OBJID (raw input), UNIT_CODE (raw input)
운영판도 고객사/프로젝트번호/유닛명 모두 주석 처리되어 노출 안 됨
- 유지: 품번 / 품명 / 표시 레벨 (1~5 select)
- BomTreeFilter.search_level 추가 + ascending/descending CTE 에 T.lev <= $search_level::int
(2) 품번/품명 자동완성 (wace select2-part 1:1):
- 영업관리 PartSelect 는 item_info 마스터 기반 → 개발관리(part_mng)용 별도 컴포넌트 신설
- backend GET /api/development/part/options : IS_LAST='1' part_mng 전체 (영업관리 sales/parts 패턴)
- frontend DevPartSelect.tsx : SmartSelect 캐시 + mode partNo/partName 분리
- ebom-search 페이지 단순 Input → DevPartSelect 교체
- 품번 선택 시 품명 자동 채움 / 품명 선택 시 품번 자동 채움 (운영판 select2-part 1:1)
(3) BomReportStatusDialog 운영판 1:1 재작성 — wace structureStatusChangePopup.jsp:
- 잘못된 점: read-only 박스 + 상태 select(create/changeDesign/deploy 3옵션)
- 정정: 5필드 모두 편집 가능 (CommCodeSelect 제품구분 / 품번 input / 품명 input /
Version input / 상태 Y/N 라디오) — 운영 매퍼 updateStructureStatus 5컬럼 UPDATE 1:1
- 헤더 파란 바 + 4컬럼 테이블(25%/75%) + 저장/닫기 중앙 배치 (운영판 스타일 1:1)
(4) DataGrid id 매핑 — 체크박스 ID 키 불일치 함정:
- DataGrid 는 row.id 로 체크박스 ID 관리, 백엔드 응답은 row.objid (postgres lowercase)
- 결과: checkedIds[0] 가 undefined → 상태변경/수정/삭제 다이얼로그가 objid=undefined 로 열려
detail 호출 안 됨 → 빈 폼 표시 (사용자 지적 "기본 정보 표시 안됨")
- 일괄 수정 (3 페이지) : ebom-regist / part-regist / part-search
gridRows = useMemo(() => rows.map(r => ({ ...r, id: r.objid })), [rows])
영업관리 페이지 동일 패턴 1:1
(5) STATUS_TITLE 매핑 운영판 1:1 — 운영 그리드는 'Y'/'N' 글자 그대로 표시:
CASE UPPER(T.STATUS)
WHEN 'CREATE' THEN '등록중'
WHEN 'CHANGEDESIGN' THEN '설계변경미배포'
WHEN 'DEPLOY' THEN '배포완료'
ELSE COALESCE(T.STATUS, '') END AS STATUS_TITLE
- 운영 매퍼는 ELSE '' 이지만 RPS 는 raw fallback (사용자 화면에서 식별 가능)
- 'Y'/'N' 매핑 라벨 추가 → 운영 스크린샷 확인 후 제거 (운영판은 raw)
미해결 (별 작업):
- 확정일 (DEPLOY_DATE) 표시 — 운영판은 별도 "배포" 액션 (deployBomReport 매퍼) 으로 채움.
RPS ebom-regist 에 배포 버튼 미구현 → 신규 BOM 확정일 빈값. 별 PR.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
190 lines
6.7 KiB
TypeScript
190 lines
6.7 KiB
TypeScript
"use client";
|
|
|
|
// 개발관리 > E-BOM 상태 변경 다이얼로그 — wace structureStatusChangePopup.jsp 1:1
|
|
//
|
|
// 운영 매퍼 updateStructureStatus 5필드 UPDATE :
|
|
// PRODUCT_CD / PART_NO / PART_NAME / REVISION / STATUS (모두 편집 가능)
|
|
// 상태는 Y / N 라디오 (운영판 그대로).
|
|
//
|
|
// 이전 구현은 read-only 요약 박스 + 상태 select(create/changeDesign/deploy) 였으나
|
|
// 운영판과 완전히 달라 정정.
|
|
|
|
import React, { useEffect, useState } from "react";
|
|
import {
|
|
Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter,
|
|
} from "@/components/ui/dialog";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Label } from "@/components/ui/label";
|
|
import { Loader2, Save } from "lucide-react";
|
|
import { toast } from "sonner";
|
|
import { CommCodeSelect } from "@/components/common/CommCodeSelect";
|
|
import { devBomApi, BomReportRow } from "@/lib/api/devBom";
|
|
|
|
const PRODUCT_GROUP = "0000001"; // 제품구분 comm_code
|
|
|
|
interface Props {
|
|
open: boolean;
|
|
onOpenChange: (open: boolean) => void;
|
|
objid: string | null;
|
|
onSaved: () => void;
|
|
}
|
|
|
|
export function BomReportStatusDialog({ open, onOpenChange, objid, onSaved }: Props) {
|
|
const [productCd, setProductCd] = useState<string>("");
|
|
const [partNo, setPartNo] = useState<string>("");
|
|
const [partName, setPartName] = useState<string>("");
|
|
const [version, setVersion] = useState<string>("");
|
|
const [status, setStatus] = useState<string>(""); // 운영판 Y / N
|
|
const [loading, setLoading] = useState(false);
|
|
const [saving, setSaving] = useState(false);
|
|
|
|
useEffect(() => {
|
|
if (!open || !objid) return;
|
|
let alive = true;
|
|
setLoading(true);
|
|
devBomApi.detail(objid)
|
|
.then((data: BomReportRow | null) => {
|
|
if (!alive) return;
|
|
if (!data) {
|
|
toast.error("E-BOM 보고서를 찾을 수 없습니다.");
|
|
onOpenChange(false);
|
|
return;
|
|
}
|
|
setProductCd(data.product_cd ?? "");
|
|
setPartNo(data.part_no ?? "");
|
|
setPartName(data.part_name ?? "");
|
|
setVersion(data.revision ?? "");
|
|
setStatus(data.status ?? "");
|
|
})
|
|
.catch((e: any) => {
|
|
toast.error(e?.response?.data?.message ?? e?.message ?? "조회 실패");
|
|
onOpenChange(false);
|
|
})
|
|
.finally(() => { if (alive) setLoading(false); });
|
|
return () => { alive = false; };
|
|
}, [open, objid, onOpenChange]);
|
|
|
|
// wace fn_check 1:1 — 제품구분 / 품번 필수, 상태 필수
|
|
const validate = (): string | null => {
|
|
if (!productCd) return "제품구분은 필수입니다.";
|
|
if (!partNo.trim()) return "품번은 필수입니다.";
|
|
if (!status) return "상태를 선택하세요.";
|
|
return null;
|
|
};
|
|
|
|
const handleSave = async () => {
|
|
if (!objid) return;
|
|
const err = validate();
|
|
if (err) return toast.error(err);
|
|
setSaving(true);
|
|
try {
|
|
await devBomApi.updateStatus(objid, {
|
|
product_cd: productCd,
|
|
part_no: partNo,
|
|
part_name: partName,
|
|
version: version || undefined,
|
|
status,
|
|
});
|
|
toast.success("상태가 변경되었습니다.");
|
|
onSaved();
|
|
onOpenChange(false);
|
|
} catch (e: any) {
|
|
toast.error(e?.response?.data?.message ?? e?.message ?? "저장 실패");
|
|
} finally {
|
|
setSaving(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<Dialog open={open} onOpenChange={onOpenChange}>
|
|
<DialogContent className="max-w-md p-0 overflow-hidden">
|
|
<DialogHeader className="bg-blue-600 px-4 py-3">
|
|
<DialogTitle className="text-white">E-BOM 상태 변경</DialogTitle>
|
|
</DialogHeader>
|
|
|
|
{loading ? (
|
|
<div className="flex h-48 items-center justify-center">
|
|
<Loader2 className="h-6 w-6 animate-spin" />
|
|
</div>
|
|
) : (
|
|
<div className="px-4 py-3">
|
|
{/* 운영판 colgroup 1:1 (25% / 75%) */}
|
|
<table className="w-full border-collapse text-sm table-fixed">
|
|
<colgroup>
|
|
<col style={{ width: "25%" }} />
|
|
<col />
|
|
</colgroup>
|
|
<tbody>
|
|
<tr>
|
|
<Th>제품구분<Req /></Th>
|
|
<Td>
|
|
<CommCodeSelect groupId={PRODUCT_GROUP} withAll={false}
|
|
value={productCd}
|
|
onValueChange={setProductCd} />
|
|
</Td>
|
|
</tr>
|
|
<tr>
|
|
<Th>품번<Req /></Th>
|
|
<Td><Input value={partNo} onChange={(e) => setPartNo(e.target.value)} /></Td>
|
|
</tr>
|
|
<tr>
|
|
<Th>품명</Th>
|
|
<Td><Input value={partName} onChange={(e) => setPartName(e.target.value)} /></Td>
|
|
</tr>
|
|
<tr>
|
|
<Th>Version</Th>
|
|
<Td><Input value={version} onChange={(e) => setVersion(e.target.value)} /></Td>
|
|
</tr>
|
|
<tr>
|
|
<Th>상태</Th>
|
|
<Td>
|
|
<div className="flex items-center gap-6">
|
|
<label className="flex items-center gap-1.5 cursor-pointer">
|
|
<input type="radio" name="status" value="Y"
|
|
checked={status === "Y"}
|
|
onChange={() => setStatus("Y")} />
|
|
<span>Y</span>
|
|
</label>
|
|
<label className="flex items-center gap-1.5 cursor-pointer">
|
|
<input type="radio" name="status" value="N"
|
|
checked={status === "N"}
|
|
onChange={() => setStatus("N")} />
|
|
<span>N</span>
|
|
</label>
|
|
</div>
|
|
</Td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
)}
|
|
|
|
<DialogFooter className="border-t bg-muted/20 px-4 py-3 sm:justify-center">
|
|
<Button onClick={handleSave} disabled={saving || loading}>
|
|
{saving ? <Loader2 className="h-4 w-4 animate-spin" /> : <Save className="h-4 w-4" />}
|
|
<span className="ml-1">저장</span>
|
|
</Button>
|
|
<Button variant="outline" onClick={() => onOpenChange(false)} disabled={saving}>
|
|
닫기
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
);
|
|
}
|
|
|
|
function Th({ children }: { children: React.ReactNode }) {
|
|
return (
|
|
<th className="border bg-muted/30 px-3 py-2 text-left align-middle font-medium">
|
|
{children}
|
|
</th>
|
|
);
|
|
}
|
|
function Td({ children }: { children: React.ReactNode }) {
|
|
return <td className="border px-3 py-1.5">{children}</td>;
|
|
}
|
|
function Req() {
|
|
return <span className="ml-1 text-red-500">*</span>;
|
|
}
|