Files
wace_rps/frontend/app/(main)/COMPANY_16/quality/incoming-request/page.tsx
T
chpark d7c645d24c
Build and Push Images / build-and-push (push) Has been cancelled
품질관리/고객CS/ECR — wace_plm 1:1 이식 + 견적관리 그리드 패턴 통일
신규 4개 메뉴 (PageHeader + CompactFilterBar + DataGrid 통일):
 - 품질관리/수입검사 요청 (/quality/incoming-request)
 - 품질관리/수입검사 관리 (/quality/incoming-mgmt)
 - 품질관리/공정검사 관리 (/quality/process-inspection)
 - 품질관리/반제품검사 관리 (/quality/semi-product-inspection)

DB 마이그레이션 (docs/migration/quality/):
 - 01_quality_tables_from_ilshin.sql — ilshin 운영 5개 테이블 vexplor_rps 정합
   (customer_service_mgmt/part/workingtime, inspection_mgmt, delivery_history_defect)
   + ecr_mng 7개 컬럼 동기화 (project_no, customer_cd, equip_name,
     design_dept, unit_cd, memo, check_result)
 - 02_wace_plm_quality_tables.sql — wace_plm quality.xml 매퍼 호환 신규 5개 테이블
   (incoming_inspection_detail/defect, process_inspection_master/detail,
    pms_quality_semi_product_inspection) + 인덱스 정의

백엔드:
 - qualityRoutes.ts — 4개 메뉴 list 엔드포인트 (실 테이블 조회)
 - ecrMngService SELECT_BASE 에 ilshin 신규 7컬럼 노출
 - app.ts 라우팅 등록 (/api/quality/*)

프론트:
 - DataGrid 4개 신규 페이지 + 그리드 툴바 (차트/엑셀/새로고침/컬럼설정/페이지사이즈)
 - customer-cs/cs, ecr/ecr — 견적관리와 동일한 PageHeader + CompactFilterBar
   + DataGrid 패턴으로 리팩토링 (다이얼로그/기존 API 유지)
 - ECR 그리드에 신규 6개 컬럼 추가 (설비명/프로젝트번호/고객사/설계부서/조치결과 등)
 - AdminPageRenderer 4개 라우트 등록

데이터 복사: ilshin → vexplor_rps (workingtime 5건, inspection_mgmt 1건,
ecr_mng 1건). 나머지 ilshin 운영 테이블은 0건이므로 스키마만 정합.
2026-05-14 19:08:15 +09:00

123 lines
5.2 KiB
TypeScript

"use client";
/**
* 수입검사 요청 — wace_plm incomingInspectionList.jsp + QualityController.getIncomingInspectionList 이식.
*
* 그리드 컬럼은 wace_plm 원본의 12개를 1:1로 따른다.
* 일부 백엔드 테이블 부재로 데이터가 비어 있을 수 있다(qualityRoutes.ts 참조).
*/
import React, { useCallback, useEffect, useState } from "react";
import { Plus } from "lucide-react";
import { Button } from "@/components/ui/button";
import { Input } from "@/components/ui/input";
import { toast } from "sonner";
import { useAuth } from "@/hooks/useAuth";
import { DataGrid, DataGridColumn } from "@/components/common/DataGrid";
import { PageHeader } from "@/components/common/PageHeader";
import { CompactFilterBar, CompactFilterField } from "@/components/common/CompactFilterBar";
import { qualityApi, IncomingRequestRow } from "@/lib/api/quality";
import { exportToExcel } from "@/lib/utils/excelExport";
const GRID_COLUMNS: DataGridColumn[] = [
{ key: "purchase_order_no", label: "발주서 No", width: "w-[150px]", frozen: true },
{ key: "proposal_no", label: "품의서 No", width: "w-[140px]" },
{ key: "project_no", label: "프로젝트번호", width: "w-[150px]" },
{ key: "product_name", label: "제품구분", width: "w-[110px]", align: "center" },
{ key: "part_no", label: "품번", width: "w-[140px]" },
{ key: "part_name", label: "품명", width: "w-[200px]" },
{ key: "partner_name", label: "공급업체", width: "w-[160px]" },
{ key: "delivery_status", label: "입고결과", width: "w-[110px]", align: "center" },
{ key: "request_date", label: "요청일", width: "w-[115px]", align: "center" },
{ key: "request_user_name", label: "요청자", width: "w-[110px]", align: "center" },
{ key: "inspection_yn", label: "검사여부", width: "w-[100px]", align: "center" },
{ key: "request_status", label: "요청현황", width: "w-[110px]", align: "center" },
];
export default function IncomingRequestPage() {
const { user } = useAuth();
const [rows, setRows] = useState<IncomingRequestRow[]>([]);
const [loading, setLoading] = useState(false);
const [selectedId, setSelectedId] = useState<string | null>(null);
const [search, setSearch] = useState({
project_no: "",
partner_objid: "",
request_user_id: "",
});
const fetchList = useCallback(async () => {
if (!user) return;
setLoading(true);
try {
const params: Record<string, string> = {};
Object.entries(search).forEach(([k, v]) => { if (v) params[k] = v; });
const res = await qualityApi.incomingRequest(params);
setRows(res.list.map((r) => ({ ...r, id: r.objid } as any)));
setSelectedId(null);
} catch (e: any) {
toast.error("수입검사 요청 목록 조회 실패");
} finally {
setLoading(false);
}
}, [user, search]);
useEffect(() => { fetchList(); }, [fetchList]);
const handleReset = () => setSearch({ project_no: "", partner_objid: "", request_user_id: "" });
return (
<div className="flex h-full flex-col overflow-hidden p-2 gap-2">
<PageHeader
loading={loading}
onSearch={fetchList}
onReset={handleReset}
actions={
<Button size="sm" className="h-8 gap-1 text-xs" disabled>
<Plus className="h-3.5 w-3.5" />
</Button>
}
/>
<CompactFilterBar totalText={<> {rows.length.toLocaleString()}</>}>
<CompactFilterField label="프로젝트번호" width={160}>
<Input value={search.project_no}
onChange={(e) => setSearch({ ...search, project_no: e.target.value })} />
</CompactFilterField>
<CompactFilterField label="공급업체 ID" width={160}>
<Input value={search.partner_objid}
onChange={(e) => setSearch({ ...search, partner_objid: e.target.value })} />
</CompactFilterField>
<CompactFilterField label="요청자 ID" width={140}>
<Input value={search.request_user_id}
onChange={(e) => setSearch({ ...search, request_user_id: e.target.value })} />
</CompactFilterField>
</CompactFilterBar>
<DataGrid
gridId="quality-incoming-request"
columns={GRID_COLUMNS}
data={rows}
loading={loading}
showCheckbox
checkedIds={selectedId ? [selectedId] : []}
onCheckedChange={(ids) => setSelectedId(ids.length ? ids[ids.length - 1] : null)}
selectedId={selectedId}
onSelect={setSelectedId}
emptyMessage="조회된 수입검사 요청이 없습니다."
showColumnSettings
paginationStyle="range"
pageSizeOptions={[10, 15, 20, 50, 100]}
showChart
onRefresh={fetchList}
onDownload={() => {
if (rows.length === 0) { toast.info("내보낼 데이터가 없습니다."); return; }
const exportRows = rows.map((r) => {
const out: Record<string, any> = {};
GRID_COLUMNS.forEach((c) => { out[c.label] = (r as any)[c.key] ?? ""; });
return out;
});
exportToExcel(exportRows, "수입검사요청.xlsx", "수입검사요청");
}}
/>
</div>
);
}