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>
209 lines
9.0 KiB
TypeScript
209 lines
9.0 KiB
TypeScript
"use client";
|
|
|
|
// 개발관리 > PART 조회 (M2) — wace partMngList.jsp 1:1
|
|
// 그리드: status = 'release' 인 PART 23셀 + BOM_QTY
|
|
// 액션: 등록 / 수정 / 삭제 / 조회 (도면연동/ERP/Excel은 별 PR)
|
|
// 참조: docs/migration/development/01-part.md
|
|
|
|
import React, { useCallback, useEffect, useMemo, useState } from "react";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Label } from "@/components/ui/label";
|
|
import {
|
|
Search, Loader2, RotateCcw, Plus, Pencil, Trash2, FileSpreadsheet,
|
|
} from "lucide-react";
|
|
import { toast } from "sonner";
|
|
import { DataGrid, DataGridColumn } from "@/components/common/DataGrid";
|
|
import { devPartApi, PartListFilter, PartRow } from "@/lib/api/devPart";
|
|
import { PartFormDialog } from "@/components/development/PartFormDialog";
|
|
import { PartDetailDialog } from "@/components/development/PartDetailDialog";
|
|
import { PartExcelImportDialog } from "@/components/development/PartExcelImportDialog";
|
|
|
|
const GRID_COLUMNS: DataGridColumn[] = [
|
|
{ key: "part_no", label: "품번", width: "w-[140px]", frozen: true },
|
|
{ key: "part_name", label: "품명", minWidth: "min-w-[220px]" },
|
|
{ key: "cu01_cnt", label: "3D", width: "w-[60px]", align: "right", formatNumber: true },
|
|
{ key: "cu02_cnt", label: "2D", width: "w-[60px]", align: "right", formatNumber: true },
|
|
{ key: "cu03_cnt", label: "PDF", width: "w-[60px]", align: "right", formatNumber: true },
|
|
{ key: "material", label: "재료", width: "w-[100px]" },
|
|
{ key: "heat_treatment_hardness", label: "열처리경도", width: "w-[110px]" },
|
|
{ key: "heat_treatment_method", label: "열처리방법", width: "w-[110px]" },
|
|
{ key: "surface_treatment", label: "표면처리", width: "w-[100px]" },
|
|
{ key: "maker", label: "메이커", width: "w-[100px]" },
|
|
{ key: "part_type_title", label: "범주", width: "w-[100px]" },
|
|
{ key: "spec", label: "규격", width: "w-[140px]" },
|
|
{ key: "acctfg_nm", label: "계정구분", width: "w-[80px]", align: "center" },
|
|
{ key: "odrfg_nm", label: "조달구분", width: "w-[80px]", align: "center" },
|
|
{ key: "unit_dc_nm", label: "재고단위", width: "w-[80px]", align: "center" },
|
|
{ key: "unitmang_dc_nm", label: "관리단위", width: "w-[80px]", align: "center" },
|
|
{ key: "unitchng_nb", label: "환산수량", width: "w-[90px]", align: "right", formatNumber: true },
|
|
{ key: "lot_fg_nm", label: "LOT구분", width: "w-[80px]", align: "center" },
|
|
{ key: "use_yn_nm", label: "사용여부", width: "w-[80px]", align: "center" },
|
|
{ key: "qc_fg_nm", label: "검사여부", width: "w-[80px]", align: "center" },
|
|
{ key: "setitem_fg_nm", label: "SET품여부", width: "w-[90px]", align: "center" },
|
|
{ key: "req_fg_nm", label: "의뢰여부", width: "w-[80px]", align: "center" },
|
|
{ key: "unit_length", label: "개당길이", width: "w-[90px]", align: "right" },
|
|
{ key: "unit_qty", label: "개당수량", width: "w-[90px]", align: "right" },
|
|
// M2 추가
|
|
{ key: "bom_qty", label: "BOM 수량", width: "w-[90px]", align: "right", formatNumber: true },
|
|
{ key: "revision", label: "REV", width: "w-[60px]", align: "center" },
|
|
{ key: "eo_no", label: "EO_NO", width: "w-[120px]" },
|
|
];
|
|
|
|
const EMPTY_FILTER: PartListFilter = {
|
|
search_part_no: "", search_part_name: "",
|
|
page: 1, page_size: 50,
|
|
};
|
|
|
|
export default function PartSearchPage() {
|
|
const [rows, setRows] = useState<PartRow[]>([]);
|
|
const [total, setTotal] = useState(0);
|
|
const [loading, setLoading] = useState(false);
|
|
const [filter, setFilter] = useState<PartListFilter>(EMPTY_FILTER);
|
|
const [checkedIds, setCheckedIds] = useState<string[]>([]);
|
|
|
|
const [formOpen, setFormOpen] = useState(false);
|
|
const [formMode, setFormMode] = useState<"create" | "edit">("create");
|
|
const [formObjid, setFormObjid] = useState<string | null>(null);
|
|
const [detailOpen, setDetailOpen] = useState(false);
|
|
const [detailObjid, setDetailObjid] = useState<string | null>(null);
|
|
const [excelOpen, setExcelOpen] = useState(false);
|
|
|
|
const fetchList = useCallback(async (override?: Partial<PartListFilter>) => {
|
|
setLoading(true);
|
|
try {
|
|
const f = { ...filter, ...override };
|
|
const res = await devPartApi.list(f);
|
|
setRows(res.rows ?? []);
|
|
setTotal(res.total ?? 0);
|
|
setCheckedIds([]);
|
|
} catch (e: any) {
|
|
toast.error(e?.response?.data?.message ?? e?.message ?? "조회 실패");
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
}, [filter]);
|
|
|
|
useEffect(() => { fetchList(); /* eslint-disable-next-line */ }, []);
|
|
|
|
const columns = useMemo(
|
|
() => GRID_COLUMNS.map((c) =>
|
|
c.key === "part_no"
|
|
? { ...c, onClick: (row: any) => { setDetailObjid(row.objid); setDetailOpen(true); } }
|
|
: c,
|
|
),
|
|
[],
|
|
);
|
|
|
|
// 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);
|
|
};
|
|
const handleEdit = () => {
|
|
if (checkedIds.length !== 1) return toast.error("수정할 행 1개를 선택하세요.");
|
|
setFormMode("edit"); setFormObjid(checkedIds[0]); setFormOpen(true);
|
|
};
|
|
const handleDelete = async () => {
|
|
if (checkedIds.length === 0) return toast.error("선택된 행이 없습니다.");
|
|
if (!confirm(`${checkedIds.length}건을 삭제하시겠습니까?`)) return;
|
|
try {
|
|
const res = await devPartApi.remove(checkedIds);
|
|
toast.success(res?.message ?? "삭제되었습니다.");
|
|
fetchList();
|
|
} catch (e: any) {
|
|
toast.error(e?.response?.data?.message ?? e?.message ?? "삭제 실패");
|
|
}
|
|
};
|
|
const handleEditFromDetail = (objid: string) => {
|
|
setDetailOpen(false);
|
|
setFormMode("edit"); setFormObjid(objid); setFormOpen(true);
|
|
};
|
|
|
|
return (
|
|
<div className="flex h-full flex-col">
|
|
<div className="border-b bg-card px-4 py-3">
|
|
<div className="flex flex-wrap items-end gap-4">
|
|
<div className="min-w-[200px]">
|
|
<Label className="mb-1 block text-xs text-muted-foreground">품번</Label>
|
|
<Input
|
|
value={filter.search_part_no ?? ""}
|
|
onChange={(e) => setFilter({ ...filter, search_part_no: e.target.value })}
|
|
placeholder="품번 LIKE"
|
|
/>
|
|
</div>
|
|
<div className="min-w-[200px]">
|
|
<Label className="mb-1 block text-xs text-muted-foreground">품명</Label>
|
|
<Input
|
|
value={filter.search_part_name ?? ""}
|
|
onChange={(e) => setFilter({ ...filter, search_part_name: e.target.value })}
|
|
placeholder="품명 LIKE"
|
|
/>
|
|
</div>
|
|
<div className="ml-auto flex items-end gap-2">
|
|
<Button variant="outline" size="sm"
|
|
onClick={() => { setFilter(EMPTY_FILTER); fetchList(EMPTY_FILTER); }}>
|
|
<RotateCcw className="h-4 w-4" /><span className="ml-1">초기화</span>
|
|
</Button>
|
|
<Button size="sm" onClick={() => fetchList()} disabled={loading}>
|
|
{loading ? <Loader2 className="h-4 w-4 animate-spin" /> : <Search className="h-4 w-4" />}
|
|
<span className="ml-1">조회</span>
|
|
</Button>
|
|
<Button size="sm" variant="default" onClick={handleCreate}>
|
|
<Plus className="h-4 w-4" /><span className="ml-1">등록</span>
|
|
</Button>
|
|
<Button size="sm" variant="secondary" onClick={handleEdit}
|
|
disabled={checkedIds.length !== 1}>
|
|
<Pencil className="h-4 w-4" /><span className="ml-1">수정</span>
|
|
</Button>
|
|
<Button size="sm" variant="destructive" onClick={handleDelete}
|
|
disabled={checkedIds.length === 0}>
|
|
<Trash2 className="h-4 w-4" /><span className="ml-1">삭제</span>
|
|
</Button>
|
|
<Button size="sm" variant="outline" onClick={() => setExcelOpen(true)}>
|
|
<FileSpreadsheet className="h-4 w-4" /><span className="ml-1">Excel Upload</span>
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
<div className="mt-2 text-xs text-muted-foreground">
|
|
총 {total.toLocaleString()}건 (M2: status = 'release')
|
|
</div>
|
|
</div>
|
|
|
|
<div className="min-h-0 flex-1 p-2">
|
|
<DataGrid
|
|
columns={columns}
|
|
data={gridRows}
|
|
loading={loading}
|
|
showRowNumber
|
|
showCheckbox
|
|
checkedIds={checkedIds}
|
|
onCheckedChange={setCheckedIds}
|
|
emptyMessage="조건에 맞는 PART가 없습니다."
|
|
gridId="development-part-search"
|
|
/>
|
|
</div>
|
|
|
|
<PartFormDialog
|
|
open={formOpen}
|
|
onOpenChange={setFormOpen}
|
|
mode={formMode}
|
|
editObjid={formObjid}
|
|
onSaved={fetchList}
|
|
/>
|
|
<PartDetailDialog
|
|
open={detailOpen}
|
|
onOpenChange={setDetailOpen}
|
|
objid={detailObjid}
|
|
onEdit={handleEditFromDetail}
|
|
/>
|
|
<PartExcelImportDialog
|
|
open={excelOpen}
|
|
onOpenChange={setExcelOpen}
|
|
onSaved={fetchList}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|