d7c645d24c
Build and Push Images / build-and-push (push) Has been cancelled
신규 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건이므로 스키마만 정합.
136 lines
5.8 KiB
TypeScript
136 lines
5.8 KiB
TypeScript
"use client";
|
|
|
|
/**
|
|
* 공정검사 관리 (IPQC) — wace_plm processInspectionList.jsp 이식.
|
|
*
|
|
* 마스터별 검사 N건을 SUM 으로 집계해 1행 표시 (검사수량/불량수량 합계).
|
|
*/
|
|
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, CompactDateRange } from "@/components/common/CompactFilterBar";
|
|
import { qualityApi, ProcessInspectionRow } from "@/lib/api/quality";
|
|
import { exportToExcel } from "@/lib/utils/excelExport";
|
|
|
|
const GRID_COLUMNS: DataGridColumn[] = [
|
|
{ key: "inspection_date", label: "검사일", width: "w-[120px]", align: "center", frozen: true },
|
|
{ key: "inspector_name", label: "검사자", width: "w-[110px]", align: "center" },
|
|
{ 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: "inspection_qty", label: "검사수량 합계", width: "w-[140px]", align: "right", formatNumber: true },
|
|
{ key: "defect_qty", label: "불량수량 합계", width: "w-[140px]", align: "right", formatNumber: true },
|
|
{ key: "work_env_status", label: "작업환경상태", width: "w-[120px]", align: "center" },
|
|
{ key: "measuring_device", label: "측정기", width: "w-[100px]", align: "center" },
|
|
{ key: "inspection_result", label: "검사결과", width: "w-[100px]", align: "center" },
|
|
{ key: "file_count", label: "첨부파일", width: "w-[100px]", align: "center", renderType: "clip" },
|
|
];
|
|
|
|
export default function ProcessInspectionPage() {
|
|
const { user } = useAuth();
|
|
const [rows, setRows] = useState<ProcessInspectionRow[]>([]);
|
|
const [loading, setLoading] = useState(false);
|
|
const [selectedId, setSelectedId] = useState<string | null>(null);
|
|
|
|
const [search, setSearch] = useState({
|
|
project_no: "",
|
|
productType: "",
|
|
part_name: "",
|
|
inspector_id: "",
|
|
from: "",
|
|
to: "",
|
|
});
|
|
|
|
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.processInspection(params);
|
|
setRows(res.list.map((r) => ({ ...r, id: r.objid } as any)));
|
|
setSelectedId(null);
|
|
} catch {
|
|
toast.error("공정검사 목록 조회 실패");
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
}, [user, search]);
|
|
|
|
useEffect(() => { fetchList(); }, [fetchList]);
|
|
|
|
const handleReset = () =>
|
|
setSearch({ project_no: "", productType: "", part_name: "", inspector_id: "", from: "", to: "" });
|
|
|
|
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="제품구분" width={140}>
|
|
<Input value={search.productType}
|
|
onChange={(e) => setSearch({ ...search, productType: e.target.value })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="품명" width={160}>
|
|
<Input value={search.part_name}
|
|
onChange={(e) => setSearch({ ...search, part_name: e.target.value })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="검사자 ID" width={130}>
|
|
<Input value={search.inspector_id}
|
|
onChange={(e) => setSearch({ ...search, inspector_id: e.target.value })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="검사일" width={280}>
|
|
<CompactDateRange
|
|
from={search.from} setFrom={(v) => setSearch({ ...search, from: v })}
|
|
to={search.to} setTo={(v) => setSearch({ ...search, to: v })}
|
|
/>
|
|
</CompactFilterField>
|
|
</CompactFilterBar>
|
|
<DataGrid
|
|
gridId="quality-process-inspection"
|
|
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>
|
|
);
|
|
}
|