"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[]>([]); const [selected, setSelected] = useState[]>([]); 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 (

투입원가관리 현황

setProjectNo(e.target.value)} className="w-[140px]" />
); }