Files
distribution_erp/src/app/(main)/cost/status/page.tsx
T
chpark e8dc97a32f
Deploy momo-erp / deploy (push) Successful in 46s
feat: FITO admin-panel 복원 + menu_info 에 모모유통 메뉴 19개 등록
- src/app/(main), admin, admin-panel, common, api/{admin,common,menu} 복원
- /api/auth/login: FITO 인증 다시 활성화 (plm_admin 등 FITO 사용자 로그인 가능)
- 미들웨어: 옛 경로 강제 리다이렉트 제거
- /m/layout.tsx: FITO 슈퍼관리자(isAdmin)도 ADMIN 으로 받아 모모 페이지 진입 허용
- DB 005: menu_info 에 모모유통 루트(9000000) + 자식 19개(/m/* URL 직접 연결)
  → plm_admin 로그인 후 사이드바 [모모유통] 그룹에서 클릭 시 동작
  → 메뉴 관리 UI 에서 추가/수정/삭제 가능
2026-04-25 23:47:13 +09:00

156 lines
6.4 KiB
TypeScript

"use client";
import { useState, useCallback, useEffect } from "react";
import Swal from "sweetalert2";
import { DataGrid, type GridColumn } from "@/components/grid/data-grid";
import { SearchForm, SearchField } from "@/components/layout/search-form";
import { Input } from "@/components/ui/input";
import { Button } from "@/components/ui/button";
import { CodeSelect } from "@/components/ui/code-select";
// costMgmt/costTotaltList.jsp 대응 - 투입원가관리 현황
export default function CostStatusPage() {
const [year, setYear] = useState(new Date().getFullYear().toString());
const [projectNo, setProjectNo] = useState("");
const [customerObjid, setCustomerObjid] = useState("");
const [product, setProduct] = useState("");
const [pmUserId, setPmUserId] = useState("");
const [data, setData] = useState<Record<string, unknown>[]>([]);
const [selected, setSelected] = useState<Record<string, unknown>[]>([]);
const columns: GridColumn[] = [
{ title: "프로젝트번호", field: "PROJECT_NO", width: 120, hozAlign: "left", frozen: true },
{
title: "프로젝트정보",
columns: [
{ title: "고객사", field: "CUSTOMER_NAME", width: 140, hozAlign: "left" },
{ title: "프로젝트명", field: "PROJECT_NAME", width: 180, hozAlign: "left" },
{ title: "요청납기일", field: "REQ_DEL_DATE", width: 100, hozAlign: "center" },
{ title: "셋업지", field: "SETUP", width: 100, hozAlign: "left" },
{ title: "PM", field: "PM_USER_NAME", width: 80, hozAlign: "center" },
],
},
{
title: "투입원가현황",
columns: [
{ title: "수주가", field: "CONTRACT_PRICE", width: 110, hozAlign: "right", formatter: "money" },
{ title: "목표가", field: "TOTAL_COST_GOAL", width: 110, hozAlign: "right", formatter: "money" },
{ title: "실투입원가", field: "TOTAL_COST_ACTUAL", width: 110, hozAlign: "right", formatter: "money" },
{ title: "투입율(%)", field: "TOTAL_INPUT_RATE", width: 80, hozAlign: "right" },
{ title: "MC율(%)", field: "MC_RATE", width: 80, hozAlign: "right" },
],
},
{
title: "재료비현황",
columns: [
{ title: "목표가", field: "MATERIAL_COST_GOAL", width: 110, hozAlign: "right", formatter: "money" },
{ title: "발생재료비", field: "ACCRUAL_MATERIAL_COST", width: 110, hozAlign: "right", formatter: "money" },
{ title: "투입율(%)", field: "MATERIAL_COST_GOAL_RATE", width: 90, hozAlign: "right" },
],
},
{
title: "노무비현황",
columns: [
{ title: "목표가", field: "LABOR_COST_GOAL", width: 110, hozAlign: "right", formatter: "money" },
{ title: "발생노무비", field: "LABOR_COST_ACTUAL", width: 110, hozAlign: "right", formatter: "money" },
{ title: "투입율(%)", field: "LABOR_INPUT_RATE", width: 90, hozAlign: "right" },
],
},
{
title: "경비현황",
columns: [
{ title: "목표가", field: "EXPENSE_COST_GOAL", width: 110, hozAlign: "right", formatter: "money" },
{ title: "발생경비", field: "ACCRUAL_EXPENSE", width: 110, hozAlign: "right", formatter: "money" },
{ title: "투입율(%)", field: "EXPENSE_RATE", width: 90, hozAlign: "right" },
],
},
];
const fetchData = useCallback(async () => {
const res = await fetch("/api/cost/status", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
year, project_no: projectNo, customer_objid: customerObjid,
product, pm_user_id: pmUserId,
}),
});
if (res.ok) {
const json = await res.json();
setData(json.RESULTLIST || []);
}
}, [year, projectNo, customerObjid, product, pmUserId]);
useEffect(() => { fetchData(); }, [fetchData]);
// 팝업 저장 후 새로고침
useEffect(() => {
(window as unknown as { fn_search?: () => void }).fn_search = fetchData;
return () => { delete (window as unknown as { fn_search?: () => void }).fn_search; };
}, [fetchData]);
const openGoalPopup = () => {
if (selected.length === 0) {
Swal.fire({ icon: "warning", title: "선택된 내용이 없습니다." });
return;
}
if (selected.length > 1) {
Swal.fire({ icon: "warning", title: "한번에 1개의 내용만 등록 가능합니다." });
return;
}
const contractObjid = String(selected[0].OBJID || "");
const w = 500;
const h = 350;
const left = (window.screen.width - w) / 2;
const top = (window.screen.height - h) / 2;
window.open(
`/cost/goal/form?contractObjid=${encodeURIComponent(contractObjid)}`,
"costGoalForm",
`width=${w},height=${h},left=${left},top=${top},scrollbars=yes,resizable=yes`
);
};
return (
<div>
<div className="flex items-center justify-between mb-4">
<h2 className="text-lg font-bold text-gray-800"> </h2>
<div className="flex gap-2">
<Button size="sm" onClick={openGoalPopup}> </Button>
<Button size="sm" variant="secondary" onClick={fetchData}></Button>
</div>
</div>
<SearchForm onSearch={fetchData}>
<SearchField label="년도">
<select value={year} onChange={(e) => setYear(e.target.value)}
className="h-9 w-[100px] rounded border border-gray-300 bg-white px-3 text-sm">
{Array.from({ length: 5 }, (_, i) => new Date().getFullYear() - i).map((y) => (
<option key={y} value={y}>{y}</option>
))}
</select>
</SearchField>
<SearchField label="프로젝트번호">
<Input value={projectNo} onChange={(e) => setProjectNo(e.target.value)} className="w-[140px]" />
</SearchField>
<SearchField label="고객사">
<CodeSelect codeId="CUSTOMER" value={customerObjid} onChange={setCustomerObjid} className="w-[160px]" />
</SearchField>
<SearchField label="제품구분">
<CodeSelect codeId="PRODUCT_TYPE" value={product} onChange={setProduct} className="w-[130px]" />
</SearchField>
<SearchField label="PM">
<CodeSelect codeId="PM_USER" value={pmUserId} onChange={setPmUserId} className="w-[130px]" />
</SearchField>
</SearchForm>
<DataGrid
columns={columns}
data={data}
showCheckbox
onSelectionChange={setSelected}
/>
</div>
);
}