e51f5f7b69
- backend purchaseInboundService 신설 — getInboundFormInit / saveInboundForm
(arrival_plan UPSERT 트랜잭션) / saveDeadlineInfo (8필드 일괄 UPDATE) /
closeArrival (이미 마감된 건 차단) + listWarehouseOptions / listAcctCodeOptions
- backend routes — GET /inbound-form/:pomObjid / POST /inbound-form/save /
POST /arrival/deadline / POST /arrival/close + 옵션 2개
- InboundFormDialog 신설 — wace deliveryAcceptanceFormPopUp_new.jsp 1:1
(좌 발주품목 read-only + 우 차수별 입고입력 + 미입고 일괄적용)
- DeadlineInfoDialog 신설 — wace swal 모달 1:1 (8필드 일괄, 단건 시 prefill)
- inbound 페이지 입고등록 / inbound-by-date 마감정보입력+매입마감 연결
- 입고등록 master SELECT 함정 수정 — RPS 에 POM.delivery_status 없어 reception_status fallback
- DataGrid 다중 frozen 누적 left 계산 인프라 추가 (frozenLeftPx props 보강)
— shadcn Table 기반이라 진짜 column pinning 불가 (자연 위치 도달 후 sticky),
입고 3페이지의 frozen 부여는 일단 제거. 진짜 pinning 은 별도 작업
199 lines
10 KiB
TypeScript
199 lines
10 KiB
TypeScript
"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<any[]>([]);
|
|
const [total, setTotal] = useState(0);
|
|
const [loading, setLoading] = useState(false);
|
|
const [filter, setFilter] = useState<PurchaseListFilter>(EMPTY_FILTER);
|
|
|
|
const [supplierOpts, setSupplierOpts] = useState<OptionItem[]>([]);
|
|
const [userOpts, setUserOpts] = useState<OptionItem[]>([]);
|
|
const yearOpts = useMemo(() => getYearOptions(), []);
|
|
|
|
const fetchList = useCallback(async (override?: Partial<PurchaseListFilter>) => {
|
|
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 (
|
|
<div className="flex h-full flex-col overflow-hidden p-2 gap-2">
|
|
<PageHeader loading={loading} onSearch={handleSearch} onReset={handleReset} />
|
|
<CompactFilterBar totalText={<>총 {total.toLocaleString()}건</>}>
|
|
<CompactFilterField label="년도" width={100}>
|
|
<SmartSelect options={yearOpts} value={filter.year ?? ""}
|
|
onValueChange={(v) => setFilter({ ...filter, year: v })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="고객사" width={170}>
|
|
<CustomerSelect value={filter.customer_cd ?? ""}
|
|
onValueChange={(v) => setFilter({ ...filter, customer_cd: v })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="프로젝트번호" width={160}>
|
|
<Input value={filter.project_no ?? ""}
|
|
onChange={(e) => setFilter({ ...filter, project_no: e.target.value })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="발주No" width={140}>
|
|
<Input value={filter.purchase_order_no ?? ""}
|
|
onChange={(e) => setFilter({ ...filter, purchase_order_no: e.target.value })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="규격" width={130}>
|
|
<Input value={filter.part_spec ?? ""}
|
|
onChange={(e) => setFilter({ ...filter, part_spec: e.target.value })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="품명" width={150}>
|
|
<Input value={filter.part_name ?? ""}
|
|
onChange={(e) => setFilter({ ...filter, part_name: e.target.value })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="공급업체" width={170}>
|
|
<SmartSelect options={supplierOpts} value={filter.partner_objid ?? ""}
|
|
onValueChange={(v) => setFilter({ ...filter, partner_objid: v })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="구매담당자" width={150}>
|
|
<SmartSelect options={userOpts} value={filter.sales_mng_user_id ?? ""}
|
|
onValueChange={(v) => setFilter({ ...filter, sales_mng_user_id: v })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="입고요청일" width={280}>
|
|
<CompactDateRange
|
|
from={filter.delivery_start_date ?? ""} setFrom={(v) => setFilter({ ...filter, delivery_start_date: v })}
|
|
to={filter.delivery_end_date ?? ""} setTo={(v) => setFilter({ ...filter, delivery_end_date: v })}
|
|
/>
|
|
</CompactFilterField>
|
|
<CompactFilterField label="발주일" width={280}>
|
|
<CompactDateRange
|
|
from={filter.reg_start_date ?? ""} setFrom={(v) => setFilter({ ...filter, reg_start_date: v })}
|
|
to={filter.reg_end_date ?? ""} setTo={(v) => setFilter({ ...filter, reg_end_date: v })}
|
|
/>
|
|
</CompactFilterField>
|
|
<CompactFilterField label="입고결과" width={120}>
|
|
<SmartSelect options={DELIVERY_STATUS_OPTS} value={filter.delivery_status ?? ""}
|
|
onValueChange={(v) => setFilter({ ...filter, delivery_status: v })} />
|
|
</CompactFilterField>
|
|
<CompactFilterField label="품번" width={140}>
|
|
<Input value={filter.part_no ?? ""}
|
|
onChange={(e) => setFilter({ ...filter, part_no: e.target.value })} />
|
|
</CompactFilterField>
|
|
</CompactFilterBar>
|
|
|
|
<DataGrid
|
|
columns={GRID_COLUMNS}
|
|
data={gridRows}
|
|
loading={loading}
|
|
emptyMessage={loading ? "조회 중..." : "데이터가 없습니다"}
|
|
gridId="purchase-inbound-by-item"
|
|
pageSizeOptions={[10, 15, 20, 50, 100]}
|
|
paginationStyle="range"
|
|
serverPaging
|
|
serverPage={filter.page ?? 1}
|
|
serverPageSize={filter.page_size ?? 50}
|
|
serverTotalItems={total}
|
|
onPageChange={(p) => { 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<string, any> = {};
|
|
GRID_COLUMNS.forEach((c) => { out[c.label] = r[c.key] ?? ""; });
|
|
return out;
|
|
});
|
|
exportToExcel(exportRows, "품목별_입고관리.xlsx", "품목별 입고");
|
|
}}
|
|
showChart
|
|
/>
|
|
</div>
|
|
);
|
|
}
|