개발관리>E-BOM 화면 운영판 1:1 정정 다수 — 검색폼·상태변경·체크박스·STATUS 표시

사용자 검증으로 발견된 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>
This commit is contained in:
hjjeong
2026-05-13 11:43:29 +09:00
parent 429b1d1e8a
commit 20a429eecb
10 changed files with 295 additions and 70 deletions
@@ -110,6 +110,9 @@ export default function EbomRegistPage() {
[openTree],
);
// DataGrid 는 row.id 를 키로 사용 — backend 응답은 row.objid (lowercase) 이므로 매핑
const gridRows = useMemo(() => rows.map((r) => ({ ...r, id: r.objid })), [rows]);
return (
<div className="flex h-full flex-col">
<div className="border-b bg-card px-4 py-3">
@@ -177,7 +180,7 @@ export default function EbomRegistPage() {
<div className="min-h-0 flex-1 p-2">
<DataGrid
columns={columns}
data={rows}
data={gridRows}
loading={loading}
showRowNumber
showCheckbox
@@ -14,12 +14,12 @@ import {
import { toast } from "sonner";
import { DataGrid, DataGridColumn } from "@/components/common/DataGrid";
import { devBomApi, BomTreeFilter, BomTreeRow } from "@/lib/api/devBom";
import { DevPartSelect } from "@/components/development/DevPartSelect";
type Direction = "ascending" | "descending";
const EMPTY_FILTER: BomTreeFilter = {
project_name: "", unit_code: "",
search_part_no: "", search_part_name: "",
search_part_no: "", search_part_name: "", search_level: "",
};
export default function EbomSearchPage() {
@@ -106,26 +106,42 @@ export default function EbomSearchPage() {
return (
<div className="flex h-full flex-col">
<div className="border-b bg-card px-4 py-3">
<div className="grid grid-cols-4 gap-3 text-sm">
<Field label="프로젝트 OBJID">
<Input value={filter.project_name ?? ""}
onChange={(e) => setFilter({ ...filter, project_name: e.target.value })}
placeholder="project_mgmt.objid" />
</Field>
<Field label="UNIT_CODE">
<Input value={filter.unit_code ?? ""}
onChange={(e) => setFilter({ ...filter, unit_code: e.target.value })}
placeholder="pms_wbs_task.objid" />
</Field>
{/* 운영판 wace structureAscendingList.jsp 1:1 — 노출 검색 필드 3개
(고객사/프로젝트번호/유닛명 은 운영판에서도 주석 처리되어 노출 안 됨) */}
<div className="grid grid-cols-3 gap-3 text-sm">
<Field label="품번">
<Input value={filter.search_part_no ?? ""}
onChange={(e) => setFilter({ ...filter, search_part_no: e.target.value })}
placeholder="part_no LIKE" />
<DevPartSelect mode="partNo"
value={filter.search_part_no ?? ""}
onValueChange={(v, row) => setFilter((prev) => ({
...prev,
search_part_no: v,
// 품번 선택 시 품명 자동 채움 (wace select2-part 1:1)
search_part_name: row?.part_name ?? prev.search_part_name,
}))} />
</Field>
<Field label="품명">
<Input value={filter.search_part_name ?? ""}
onChange={(e) => setFilter({ ...filter, search_part_name: e.target.value })}
placeholder="part_name LIKE" />
<DevPartSelect mode="partName"
value={filter.search_part_name ?? ""}
onValueChange={(v, row) => setFilter((prev) => ({
...prev,
search_part_name: v,
// 품명 선택 시 품번 자동 채움
search_part_no: row?.part_no ?? prev.search_part_no,
}))} />
</Field>
<Field label="표시 레벨">
<select
className="h-9 w-full rounded-md border bg-background px-2 text-sm"
value={String(filter.search_level ?? "")}
onChange={(e) => setFilter({ ...filter, search_level: e.target.value })}
>
<option value=""></option>
<option value="1">1</option>
<option value="2">2</option>
<option value="3">3</option>
<option value="4">4</option>
<option value="5">5</option>
</select>
</Field>
</div>
<div className="mt-3 flex items-center justify-between">
@@ -163,7 +179,7 @@ export default function EbomSearchPage() {
</div>
{direction === "descending" && (
<div className="mt-2 text-xs text-amber-600">
PART (/) BOM/ .
.
</div>
)}
</div>
@@ -100,6 +100,9 @@ export default function PartRegistPage() {
[],
);
// DataGrid 는 row.id 를 키로 사용 — backend 응답은 row.objid 이므로 매핑
const gridRows = useMemo(() => rows.map((r) => ({ ...r, id: r.objid })), [rows]);
// 등록
const handleCreate = () => {
setFormMode("create");
@@ -208,7 +211,7 @@ export default function PartRegistPage() {
<div className="min-h-0 flex-1 p-2">
<DataGrid
columns={columns}
data={rows}
data={gridRows}
loading={loading}
showRowNumber
showCheckbox
@@ -95,6 +95,9 @@ export default function PartSearchPage() {
[],
);
// DataGrid 는 row.id 를 키로 사용 — backend 응답은 row.objid 이므로 매핑
const gridRows = useMemo(() => rows.map((r) => ({ ...r, id: r.objid })), [rows]);
const handleCreate = () => {
setFormMode("create"); setFormObjid(null); setFormOpen(true);
};
@@ -171,7 +174,7 @@ export default function PartSearchPage() {
<div className="min-h-0 flex-1 p-2">
<DataGrid
columns={columns}
data={rows}
data={gridRows}
loading={loading}
showRowNumber
showCheckbox