7a7f4f03b5
PR-B1 본 편집/저장 (운영 saveMbom.do 1:1)
· 매퍼 7종 1:1 (insert/updateMbomHeader, insert/updateMbomDetail,
deleteMbomDetailByObjid, insertMbomHistory, updateProjectMbomStatus)
· 신규 CREATE: createObjId + generateMbomNo(M-{partNo}-YYMMDD-NN) +
child_objid 재매핑 + detail 일괄 insert + history(CREATE) + project_mgmt.mbom_status='Y'
· 수정 UPDATE: 기존 mbom_header.objid UPSERT(insert/update/delete) + history(UPDATE)
· POST /api/production/mbom/save (BEGIN/COMMIT/ROLLBACK 트랜잭션)
· MbomDetailDialog: '본 편집' 토글 + 13개 셀 인라인 편집 + 저장/취소 가드
M-BOM 컬럼 폴더 아이콘
· production/mbom/page.tsx: mbom_status 컬럼 → mbom_has(0/1) renderType=folder
· onClick → MbomDetailDialog 오픈 (행 더블클릭도 그대로 유지)
· 운영판 wace 견적/partMng 폴더 아이콘 패턴 1:1
DataGrid 서버 페이지네이션
· props 신설: serverPaging/serverPage/serverPageSize/serverTotalItems
+ onPageChange/onPageSizeChange
· 5메뉴 적용: production/mbom, development/change-list/ebom-regist/part-search/part-regist
· pageSizeOptions=[10,15,20,50,100,200,500] 통일
· 클라이언트 모드 하위호환 유지
bigint=varchar fix (mbom 트리 SQL 4종)
· ATTACH_FILE_INFO 서브쿼리: P.OBJID(bigint) = F.TARGET_OBJID(varchar) → P.OBJID::varchar 캐스트
· EBOM_WORKING_TREE_SQL INNER JOIN: P.OBJID = COALESCE(V.LAST_PART_OBJID,V.PART_NO) → ::varchar 캐스트
· 사용자 보고: 폴더 클릭 시 'operator does not exist: bigint = character varying' 토스트
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
236 lines
7.4 KiB
TypeScript
236 lines
7.4 KiB
TypeScript
import { apiClient } from "./client";
|
|
|
|
// ============================================================
|
|
// 생산관리 > M-BOM 관리 — wace productionplanning.xml 1:1
|
|
// 라우트:
|
|
// GET /api/production/mbom/list (PR-A1, 그리드)
|
|
// GET /api/production/mbom/detail/:objid (PR-A2, 단건 상세)
|
|
// GET /api/production/mbom/tree/:objid (PR-A2, read-only 트리 4분기)
|
|
// ============================================================
|
|
|
|
export interface MbomListFilter {
|
|
search_category_cd?: string;
|
|
search_product_cd?: string;
|
|
search_area_cd?: string;
|
|
search_customer_objid?: string;
|
|
search_paid_type?: string;
|
|
search_serial_no?: string;
|
|
search_part_no?: string;
|
|
search_part_name?: string;
|
|
search_receipt_date_from?: string;
|
|
search_receipt_date_to?: string;
|
|
search_req_del_date_from?: string;
|
|
search_req_del_date_to?: string;
|
|
page?: number;
|
|
page_size?: number;
|
|
}
|
|
|
|
export interface MbomRow {
|
|
objid: string;
|
|
contract_objid: string | null;
|
|
project_no: string | null;
|
|
category_cd: string | null;
|
|
category_name: string | null;
|
|
product: string | null;
|
|
product_name: string | null;
|
|
area_cd: string | null;
|
|
area_name: string | null;
|
|
receipt_date: string | null;
|
|
writer_name: string | null;
|
|
customer_objid: string | null;
|
|
customer_name: string | null;
|
|
paid_type: string | null;
|
|
paid_type_name: string | null;
|
|
part_no: string | null;
|
|
part_name: string | null;
|
|
part_objid: string | null;
|
|
serial_no: string | null;
|
|
serial_no_list: string | null;
|
|
quantity: string | number | null;
|
|
req_del_date: string | null;
|
|
customer_request: string | null;
|
|
bom_report_objid: string | null;
|
|
ebom_status: string | null;
|
|
ebom_regdate: string | null;
|
|
mbom_header_objid: string | null;
|
|
purchase_list_objid: string | null;
|
|
purchase_list_date: string | null;
|
|
mbom_status: string | null;
|
|
mbom_part_no: string | null;
|
|
mbom_regdate: string | null;
|
|
mbom_editor: string | null;
|
|
mbom_version: number | null;
|
|
}
|
|
|
|
export interface MbomListResponse {
|
|
rows: MbomRow[];
|
|
totalCount: number;
|
|
page: number;
|
|
pageSize: number;
|
|
}
|
|
|
|
// ─── 단건 상세 (PR-A2) ──────────────────────────────────────
|
|
|
|
export interface MbomDetail {
|
|
objid: string;
|
|
contract_objid: string | null;
|
|
project_no: string | null;
|
|
bom_report_objid: string | null;
|
|
part_objid: string | null;
|
|
part_no: string | null;
|
|
part_name: string | null;
|
|
source_bom_type: string | null;
|
|
source_ebom_objid: string | null;
|
|
source_mbom_objid: string | null;
|
|
quantity: string | number | null;
|
|
total_prod_qty: string | number | null;
|
|
mbom_part_no: string | null;
|
|
category_cd: string | null;
|
|
category_name: string | null;
|
|
product: string | null;
|
|
product_code: string | null;
|
|
product_name: string | null;
|
|
area_cd: string | null;
|
|
area_name: string | null;
|
|
customer_objid: string | null;
|
|
customer_name: string | null;
|
|
paid_type: string | null;
|
|
req_del_date: string | null;
|
|
receipt_date: string | null;
|
|
mbom_regdate: string | null;
|
|
}
|
|
|
|
// ─── read-only 트리 (PR-A2) ─────────────────────────────────
|
|
// 운영판 mBomPopupLeft.do 4분기 자동 판별:
|
|
// SAVED — mbom_header.status='Y' 최신
|
|
// ASSIGNED_EBOM — source_bom_type='EBOM' + source_ebom_objid
|
|
// ASSIGNED_MBOM — source_bom_type='MBOM' + source_mbom_objid
|
|
// TEMPLATE — Machine 이외 + 동일 part_no 의 mbom_header
|
|
// NONE — 빈 트리
|
|
|
|
export type MbomBomDataType = "SAVED" | "ASSIGNED_EBOM" | "ASSIGNED_MBOM" | "TEMPLATE" | "NONE";
|
|
|
|
export interface MbomTreeRow {
|
|
objid: string;
|
|
parent_objid: string | null;
|
|
child_objid: string | null;
|
|
part_objid: string | null;
|
|
part_no: string | null;
|
|
part_name: string | null;
|
|
qty: string | number | null;
|
|
item_qty: string | number | null;
|
|
qty_temp: string | number | null;
|
|
level: number;
|
|
sub_part_cnt: number;
|
|
seq: number;
|
|
status: string | null;
|
|
unit: string | null;
|
|
unit_title: string | null;
|
|
supply_type: string | null;
|
|
make_or_buy: string | null;
|
|
raw_material_no: string | null;
|
|
raw_material_spec: string | null;
|
|
raw_material: string | null;
|
|
size: string | null;
|
|
processing_vendor: string | null;
|
|
processing_vendor_name: string | null;
|
|
processing_deadline: string | null;
|
|
grinding_deadline: string | null;
|
|
required_qty: string | number | null;
|
|
order_qty: string | number | null;
|
|
production_qty: string | number | null;
|
|
vendor: string | null;
|
|
vendor_name: string | null;
|
|
unit_price: string | number | null;
|
|
total_price: string | number | null;
|
|
currency: string | null;
|
|
writer: string | null;
|
|
regdate: string | null;
|
|
editer: string | null;
|
|
edit_date: string | null;
|
|
remark: string | null;
|
|
spec: string | null;
|
|
material: string | null;
|
|
weight: string | number | null;
|
|
revision: string | null;
|
|
cu01_cnt: number;
|
|
cu02_cnt: number;
|
|
cu03_cnt: number;
|
|
[key: string]: any;
|
|
}
|
|
|
|
export interface MbomTreeResponse {
|
|
bom_data_type: MbomBomDataType;
|
|
bom_report_objid: string | null;
|
|
max_level: number;
|
|
rows: MbomTreeRow[];
|
|
}
|
|
|
|
// ─── 저장 (PR-B1) ───────────────────────────────────────────
|
|
// 운영판 saveMbom.do 1:1 — 신규/수정 통합 엔드포인트.
|
|
// is_update=false → 새 mbom_header + child_objid 재매핑 후 detail 일괄 insert
|
|
// is_update=true → 기존 mbom_header.objid 조회 → UPSERT + 누락 행 delete
|
|
|
|
export interface MbomSaveRow {
|
|
objid?: string | null;
|
|
parent_objid?: string | null;
|
|
child_objid?: string | null;
|
|
seq?: number | string | null;
|
|
level?: number | string | null;
|
|
part_objid?: string | number | null;
|
|
part_no?: string | null;
|
|
part_name?: string | null;
|
|
qty?: number | string | null;
|
|
item_qty?: number | string | null;
|
|
unit?: string | null;
|
|
supply_type?: string | null;
|
|
make_or_buy?: string | null;
|
|
raw_material_no?: string | null;
|
|
raw_material_spec?: string | null;
|
|
raw_material?: string | null;
|
|
size?: string | null;
|
|
raw_material_size?: string | null;
|
|
processing_vendor?: string | null;
|
|
processing_deadline?: string | null;
|
|
grinding_deadline?: string | null;
|
|
required_qty?: number | string | null;
|
|
order_qty?: number | string | null;
|
|
production_qty?: number | string | null;
|
|
remark?: string | null;
|
|
}
|
|
|
|
export interface MbomSavePayload {
|
|
project_obj_id: string;
|
|
is_update: boolean;
|
|
mbom_part_no?: string | null;
|
|
rows: MbomSaveRow[];
|
|
}
|
|
|
|
export interface MbomSaveResult {
|
|
mode: "CREATE" | "UPDATE";
|
|
mbom_header_objid: string;
|
|
mbom_no: string;
|
|
inserted: number;
|
|
updated: number;
|
|
deleted: number;
|
|
}
|
|
|
|
export const mbomApi = {
|
|
async list(filter: MbomListFilter = {}): Promise<MbomListResponse> {
|
|
const res = await apiClient.get("/production/mbom/list", { params: filter });
|
|
return res.data?.data as MbomListResponse;
|
|
},
|
|
async getDetail(objid: string): Promise<MbomDetail> {
|
|
const res = await apiClient.get(`/production/mbom/detail/${encodeURIComponent(objid)}`);
|
|
return res.data?.data as MbomDetail;
|
|
},
|
|
async getTree(objid: string): Promise<MbomTreeResponse> {
|
|
const res = await apiClient.get(`/production/mbom/tree/${encodeURIComponent(objid)}`);
|
|
return res.data?.data as MbomTreeResponse;
|
|
},
|
|
async save(payload: MbomSavePayload): Promise<MbomSaveResult> {
|
|
const res = await apiClient.post("/production/mbom/save", payload);
|
|
return res.data?.data as MbomSaveResult;
|
|
},
|
|
};
|