"use client"; // 구매관리 > 품목별 입고관리 — wace purchaseOrder/deliveryMngAcceptancePartList.jsp 1:1 // 검색: 입고관리와 거의 동일 + 부품품번 추가 // 그리드: 품의서/발주서/프로젝트/부품품번/품번/품명/공급업체/환종/입고요청일/담당자/입고등록자/일/ // 발주·입고·미입고 수량+금액 / 검사현황 / 폐기수량 / 확정입고수량 // 액션: 조회만 (입고등록·매입마감은 비활성) // ⚠️ purchase_order_part / arrival_plan 미존재 → 빈 그리드 import React, { useCallback, useEffect, useMemo, useState } from "react"; import { Input } from "@/components/ui/input"; import { toast } from "sonner"; import { DataGrid, DataGridColumn } from "@/components/common/DataGrid"; import { SmartSelect, SmartSelectOption } from "@/components/common/SmartSelect"; import { CustomerSelect } from "@/components/common/CustomerSelect"; import { CompactFilterBar, CompactFilterField, CompactDateRange } from "@/components/common/CompactFilterBar"; import { PageHeader } from "@/components/common/PageHeader"; import { purchaseApi, PurchaseListFilter, OptionItem, getYearOptions } from "@/lib/api/purchase"; import { exportToExcel } from "@/lib/utils/excelExport"; const DELIVERY_STATUS_OPTS: SmartSelectOption[] = [ { code: "입고중", label: "입고중" }, { code: "입고완료", label: "입고완료" }, { code: "지연", label: "지연" }, ]; const EMPTY_FILTER: PurchaseListFilter = { year: String(new Date().getFullYear()), customer_cd: "", project_no: "", purchase_order_no: "", part_spec: "", part_name: "", partner_objid: "", sales_mng_user_id: "", delivery_start_date: "", delivery_end_date: "", reg_start_date: "", reg_end_date: "", delivery_status: "", part_no: "", page: 1, page_size: 50, }; export default function InboundByItemPage() { const [rows, setRows] = useState([]); const [total, setTotal] = useState(0); const [loading, setLoading] = useState(false); const [filter, setFilter] = useState(EMPTY_FILTER); const [supplierOpts, setSupplierOpts] = useState([]); const [userOpts, setUserOpts] = useState([]); const yearOpts = useMemo(() => getYearOptions(), []); const fetchList = useCallback(async (override?: Partial) => { setLoading(true); try { const f = { ...filter, ...override }; const res = await purchaseApi.listInboundByItem(f); setRows(res.rows ?? []); setTotal(res.totalCount ?? 0); } catch (e: any) { toast.error(e?.response?.data?.message ?? e?.message ?? "조회 실패"); } finally { setLoading(false); } }, [filter]); useEffect(() => { let dead = false; (async () => { try { const [s, u] = await Promise.all([purchaseApi.listSuppliers(), purchaseApi.listUsers()]); if (dead) return; setSupplierOpts(s); setUserOpts(u); } catch { /* skip */ } })(); fetchList(EMPTY_FILTER); return () => { dead = true; }; // eslint-disable-next-line react-hooks/exhaustive-deps }, []); const gridRows = useMemo(() => rows.map((r, i) => ({ ...r, id: r.objid ?? `ii_${i}` })), [rows]); const GRID_COLUMNS: DataGridColumn[] = useMemo(() => ([ { key: "proposal_no", label: "품의서 No", width: "w-[125px]", align: "center" }, { key: "purchase_order_no", label: "발주서 No", width: "w-[125px]", align: "center" }, { key: "project_no", label: "프로젝트번호", width: "w-[135px]", align: "center" }, { key: "component_part_no", label: "부품품번", width: "w-[140px]" }, { key: "part_no", label: "품번", width: "w-[140px]" }, { key: "part_name", label: "품명", width: "w-[280px]" }, { key: "partner_name", label: "공급업체", minWidth: "min-w-[150px]" }, { key: "currency_name", label: "환종", width: "w-[80px]", align: "center" }, { key: "delivery_request_date", label: "입고요청일", width: "w-[115px]", align: "center" }, { key: "writer_name", label: "구매담당자", width: "w-[110px]", align: "center" }, { key: "delivery_writer_name", label: "입고등록자", width: "w-[110px]", align: "center" }, { key: "delivery_regdate", label: "입고등록일", width: "w-[115px]", align: "center" }, { key: "order_qty", label: "발주수량", width: "w-[110px]", align: "right", formatNumber: true }, { key: "delivery_qty", label: "입고수량", width: "w-[110px]", align: "right", formatNumber: true }, { key: "non_delivery_qty", label: "미입고수량", width: "w-[110px]", align: "right", formatNumber: true }, { key: "total_supply_price", label: "발주금액", width: "w-[120px]", align: "right", formatMoney: true }, { key: "total_delivery_price", label: "입고금액", width: "w-[120px]", align: "right", formatMoney: true }, { key: "total_not_delivery_price", label: "미입고금액", width: "w-[120px]", align: "right", formatMoney: true }, { key: "inspection_status", label: "검사현황", width: "w-[110px]", align: "center" }, { key: "defect_qty", label: "폐기수량", width: "w-[110px]", align: "right", formatNumber: true }, { key: "confirmed_qty", label: "확정입고수량", width: "w-[120px]", align: "right", formatNumber: true }, ]), []); const summary = useMemo(() => [ { label: "전체 건수", value: total.toLocaleString(), suffix: "건" }, ], [total]); const handleSearch = () => { setFilter(f => ({ ...f, page: 1 })); fetchList({ page: 1 }); }; const handleReset = () => { setFilter(EMPTY_FILTER); fetchList(EMPTY_FILTER); }; return (
총 {total.toLocaleString()}건}> setFilter({ ...filter, year: v })} /> setFilter({ ...filter, customer_cd: v })} /> setFilter({ ...filter, project_no: e.target.value })} /> setFilter({ ...filter, purchase_order_no: e.target.value })} /> setFilter({ ...filter, part_spec: e.target.value })} /> setFilter({ ...filter, part_name: e.target.value })} /> setFilter({ ...filter, partner_objid: v })} /> setFilter({ ...filter, sales_mng_user_id: v })} /> setFilter({ ...filter, delivery_start_date: v })} to={filter.delivery_end_date ?? ""} setTo={(v) => setFilter({ ...filter, delivery_end_date: v })} /> setFilter({ ...filter, reg_start_date: v })} to={filter.reg_end_date ?? ""} setTo={(v) => setFilter({ ...filter, reg_end_date: v })} /> setFilter({ ...filter, delivery_status: v })} /> setFilter({ ...filter, part_no: e.target.value })} /> { setFilter(f => ({ ...f, page: p })); fetchList({ page: p }); }} onPageSizeChange={(n) => { setFilter(f => ({ ...f, page: 1, page_size: n })); fetchList({ page: 1, page_size: n }); }} showColumnSettings summaryStats={summary} systemColumnKeys={["delivery_writer_name", "delivery_regdate"]} onRefresh={() => fetchList()} onDownload={() => { if (gridRows.length === 0) { toast.info("내보낼 데이터가 없습니다."); return; } const exportRows = gridRows.map((r: any) => { const out: Record = {}; GRID_COLUMNS.forEach((c) => { out[c.label] = r[c.key] ?? ""; }); return out; }); exportToExcel(exportRows, "품목별_입고관리.xlsx", "품목별 입고"); }} showChart />
); }