Files
distribution_erp/src/app/(main)/product/spec/page.tsx
T
chpark 6af863199f feat: 모모유통 유통관리 ERP (Next.js 16) — MOMO 브랜딩 + distribution DB + momo.junggomoa.com
- fito-nextjs 기반으로 재구성
- 로그인: MOMO 로고 + 모모유통 + 유통관리 ERP, 하단에 본사/지사 주소 표시
- 사이드바 상단: MOMO 아이콘 + 모모유통 + 유통관리 ERP
- 파비콘: /src/app/icon.svg (MOMO 그린 배지)
- layout.tsx title: 모모유통 | 유통관리 ERP
- DB: 183.99.177.40:5432/distribution (fito 스키마 import 완료)
- Traefik: Host(momo.junggomoa.com), 컨테이너 momo-erp
2026-04-25 02:44:40 +09:00

117 lines
4.9 KiB
TypeScript

"use client";
import { useState, useCallback, useEffect } from "react";
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 Swal from "sweetalert2";
// 사양관리
export default function ProductSpecPage() {
const [projectNo, setProjectNo] = useState("");
const [specName, setSpecName] = useState("");
const [data, setData] = useState<Record<string, unknown>[]>([]);
const columns: GridColumn[] = [
{
title: "프로젝트번호", field: "PROJECT_NO", width: 130, hozAlign: "center",
cellClick: (row) => {
window.open(
`/product/spec/detail?objId=${row.OBJID}`,
"specDetail",
"width=1000,height=700"
);
},
},
{ title: "프로젝트명", field: "PROJECT_NAME", width: 200, hozAlign: "left" },
{ title: "사양명", field: "SPEC_NAME", width: 200, hozAlign: "left" },
{ title: "사양값", field: "SPEC_VALUE", width: 150, hozAlign: "left" },
{ title: "단위", field: "UNIT", width: 80, hozAlign: "center" },
{ title: "비고", field: "REMARK", width: 200, hozAlign: "left" },
{ title: "등록일", field: "REG_DATE", width: 100, hozAlign: "center" },
{ title: "등록자", field: "REG_USER_NAME", width: 100, hozAlign: "center" },
];
const fetchData = useCallback(async () => {
try {
const res = await fetch("/api/product/spec", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ project_no: projectNo, spec_name: specName }),
});
if (res.ok) {
const json = await res.json();
setData(json.RESULTLIST || []);
}
} catch {
Swal.fire("오류", "데이터 조회 중 오류가 발생했습니다.", "error");
}
}, [projectNo, specName]);
const handleAdd = async () => {
const { value: form } = await Swal.fire({
title: "사양 등록",
html: `
<div style="text-align:left;font-size:13px">
<label style="display:block;margin-bottom:4px">프로젝트번호</label>
<input id="swal-pn" class="swal2-input" style="width:100%;margin:0 0 8px" />
<label style="display:block;margin-bottom:4px">사양명 <span style="color:red">*</span></label>
<input id="swal-sn" class="swal2-input" style="width:100%;margin:0 0 8px" />
<label style="display:block;margin-bottom:4px">사양값</label>
<input id="swal-sv" class="swal2-input" style="width:100%;margin:0 0 8px" />
<label style="display:block;margin-bottom:4px">단위</label>
<input id="swal-un" class="swal2-input" style="width:100%;margin:0" />
</div>`,
showCancelButton: true, confirmButtonText: "저장", cancelButtonText: "취소",
preConfirm: () => ({
project_no: (document.getElementById("swal-pn") as HTMLInputElement)?.value,
spec_name: (document.getElementById("swal-sn") as HTMLInputElement)?.value,
spec_value: (document.getElementById("swal-sv") as HTMLInputElement)?.value,
unit: (document.getElementById("swal-un") as HTMLInputElement)?.value,
}),
});
if (!form || !form.spec_name) return;
const res = await fetch("/api/product/spec/save", {
method: "POST", headers: { "Content-Type": "application/json" },
body: JSON.stringify({ actionType: "regist", ...form }),
});
const json = await res.json();
if (json.success) {
Swal.fire({ icon: "success", title: json.message, timer: 1200, showConfirmButton: false });
fetchData();
} else Swal.fire("오류", json.message || "저장 실패", "error");
};
const handleDelete = () => {
Swal.fire("알림", "항목을 선택한 뒤 다시 시도하세요 (리스트 삭제는 별도 구현 예정)", "info");
};
// 페이지 진입 시 자동 로드
useEffect(() => { fetchData(); }, [fetchData]);
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={handleAdd}></Button>
<Button size="sm" variant="danger" onClick={handleDelete}></Button>
<Button size="sm" variant="secondary" onClick={fetchData}></Button>
</div>
</div>
<SearchForm onSearch={fetchData}>
<SearchField label="프로젝트번호">
<Input value={projectNo} onChange={(e) => setProjectNo(e.target.value)} className="w-[130px]" />
</SearchField>
<SearchField label="사양명">
<Input value={specName} onChange={(e) => setSpecName(e.target.value)} className="w-[200px]" />
</SearchField>
</SearchForm>
<DataGrid columns={columns} data={data} showCheckbox />
</div>
);
}