Merge branch 'main' into jskim-node

This commit is contained in:
jskim
2026-04-10 01:42:46 +00:00
16 changed files with 1188 additions and 152 deletions
@@ -359,11 +359,21 @@ export const syncWorkInstructions = async (
}); });
// 미동기화 작업지시 조회: routing이 있지만 work_order_process가 없는 항목 // 미동기화 작업지시 조회: routing이 있지만 work_order_process가 없는 항목
// header에 routing 없으면 detail에서 가져옴 (PC가 detail에만 저장하는 경우 대응)
const unsyncedResult = await pool.query( const unsyncedResult = await pool.query(
`SELECT wi.id, wi.work_instruction_no, wi.routing, wi.qty `SELECT wi.id, wi.work_instruction_no,
COALESCE(wi.routing, wid.routing_version_id) AS routing,
COALESCE(NULLIF(wi.qty, ''), wid.qty) AS qty,
COALESCE(wi.item_id, (SELECT id FROM item_info WHERE item_number = wid.item_number AND company_code = $1 LIMIT 1)) AS item_id
FROM work_instruction wi FROM work_instruction wi
LEFT JOIN LATERAL (
SELECT routing_version_id, qty, item_number
FROM work_instruction_detail
WHERE work_instruction_no = wi.work_instruction_no AND company_code = $1
LIMIT 1
) wid ON true
WHERE wi.company_code = $1 WHERE wi.company_code = $1
AND wi.routing IS NOT NULL AND COALESCE(wi.routing, wid.routing_version_id) IS NOT NULL
AND NOT EXISTS ( AND NOT EXISTS (
SELECT 1 FROM work_order_process wop SELECT 1 FROM work_order_process wop
WHERE wop.wo_id = wi.id AND wop.company_code = $1 WHERE wop.wo_id = wi.id AND wop.company_code = $1
@@ -373,6 +383,20 @@ export const syncWorkInstructions = async (
const unsynced = unsyncedResult.rows; const unsynced = unsyncedResult.rows;
// header에 routing/qty/item_id가 비어있으면 자동 보정 (detail → header 동기화)
for (const wi of unsynced) {
await pool.query(
`UPDATE work_instruction SET
routing = COALESCE(routing, $2),
qty = COALESCE(NULLIF(qty, ''), $3),
item_id = COALESCE(item_id, $4),
updated_date = NOW()
WHERE id = $1 AND company_code = $5
AND (routing IS NULL OR qty IS NULL OR qty = '' OR item_id IS NULL)`,
[wi.id, wi.routing, wi.qty, wi.item_id, companyCode],
);
}
if (unsynced.length === 0) { if (unsynced.length === 0) {
return res.json({ return res.json({
success: true, success: true,
@@ -1050,7 +1050,7 @@ export async function getWarehouses(req: AuthenticatedRequest, res: Response) {
const result = await pool.query( const result = await pool.query(
`SELECT warehouse_code, warehouse_name, warehouse_type `SELECT warehouse_code, warehouse_name, warehouse_type
FROM warehouse_info FROM warehouse_info
WHERE company_code = $1 AND status != '삭제' WHERE company_code = $1 AND COALESCE(status, '') != '삭제'
ORDER BY warehouse_name`, ORDER BY warehouse_name`,
[companyCode], [companyCode],
); );
@@ -0,0 +1,6 @@
"use client";
import { EquipmentInspection } from "@/components/pop/hardcoded/equipment/EquipmentInspection";
export default function EquipmentInspectionPage() {
return <EquipmentInspection />;
}
@@ -0,0 +1,6 @@
"use client";
import { EquipmentList } from "@/components/pop/hardcoded/equipment/EquipmentList";
export default function EquipmentManagementPage() {
return <EquipmentList />;
}
@@ -0,0 +1,6 @@
"use client";
import { EquipmentHome } from "@/components/pop/hardcoded/equipment/EquipmentHome";
export default function EquipmentPage() {
return <EquipmentHome />;
}
@@ -0,0 +1,6 @@
"use client";
import { InventoryMove } from "@/components/pop/hardcoded/inventory/InventoryMove";
export default function InventoryMovePage() {
return <InventoryMove />;
}
@@ -0,0 +1,6 @@
"use client";
import { InventoryTransfer } from "@/components/pop/hardcoded/inventory/InventoryTransfer";
export default function TransferPage() {
return <InventoryTransfer />;
}
@@ -126,7 +126,7 @@ const MENU_ITEMS: MenuIconItem[] = [
/> />
</svg> </svg>
), ),
href: "/pop/screens/equipment", href: "/pop/equipment",
}, },
{ {
id: "inventory", id: "inventory",
@@ -0,0 +1,198 @@
"use client";
import { useRouter } from "next/navigation";
import React, { useEffect, useState } from "react";
import { apiClient } from "@/lib/api/client";
/* ------------------------------------------------------------------ */
/* Types */
/* ------------------------------------------------------------------ */
interface EquipmentItem {
id: string;
equipment_code: string;
equipment_name: string;
equipment_type?: string;
status?: string;
}
interface KpiData {
total: number;
active: number;
idle: number;
inspect: number;
rate: string;
}
/* ------------------------------------------------------------------ */
/* Menu */
/* ------------------------------------------------------------------ */
const MENU_ITEMS = [
{
id: "management",
title: "설비관리",
gradient: "linear-gradient(135deg,#8b5cf6,#6d28d9)",
shadowColor: "rgba(139,92,246,.3)",
icon: (
<svg className="w-7 h-7 text-white" fill="none" stroke="currentColor" strokeWidth={1.5} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M11.42 15.17l-5.65-5.65a8 8 0 1111.31 0l-5.65 5.65z" />
</svg>
),
href: "/pop/equipment/management",
},
{
id: "inspection",
title: "설비점검",
gradient: "linear-gradient(135deg,#f59e0b,#d97706)",
shadowColor: "rgba(245,158,11,.3)",
icon: (
<svg className="w-7 h-7 text-white" fill="none" stroke="currentColor" strokeWidth={1.5} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M21.75 6.75a4.5 4.5 0 01-4.884 4.484c-1.076-.091-2.264.071-2.95.904l-7.152 8.684a2.548 2.548 0 11-3.586-3.586l8.684-7.152c.833-.686.995-1.874.904-2.95a4.5 4.5 0 016.336-4.486l-3.276 3.276a3.004 3.004 0 002.25 2.25l3.276-3.276c.256.565.398 1.192.398 1.852z" />
</svg>
),
href: "/pop/equipment/inspection",
},
];
/* ------------------------------------------------------------------ */
/* Component */
/* ------------------------------------------------------------------ */
export function EquipmentHome() {
const router = useRouter();
const [kpi, setKpi] = useState<KpiData>({
total: 0,
active: 0,
idle: 0,
inspect: 0,
rate: "0%",
});
const [recentItems, setRecentItems] = useState<EquipmentItem[]>([]);
const [loading, setLoading] = useState(true);
useEffect(() => {
const fetchData = async () => {
setLoading(true);
try {
const res = await apiClient.get("/data/equipment_mng", {
params: { pageSize: 500 },
});
const data = res.data?.data?.data ?? res.data?.data ?? [];
const items = Array.isArray(data) ? data : [];
const total = items.length;
const active = items.filter((i: EquipmentItem) => !i.status || i.status === "가동" || i.status === "정상").length;
const idle = items.filter((i: EquipmentItem) => i.status === "대기").length;
const inspect = items.filter((i: EquipmentItem) => i.status === "점검").length;
const rate = total > 0 ? `${Math.round((active / total) * 100)}%` : "0%";
setKpi({ total, active, idle, inspect, rate });
setRecentItems(items.slice(0, 5));
} catch { /* */ }
setLoading(false);
};
fetchData();
}, []);
return (
<div className="min-h-screen bg-gray-50">
{/* Header */}
<div className="bg-white px-5 pt-5 pb-4">
<div className="flex items-center gap-3 mb-3">
<button onClick={() => router.back()} className="w-9 h-9 rounded-xl bg-gray-100 flex items-center justify-center hover:bg-gray-200 transition-colors">
<svg className="w-5 h-5 text-gray-600" fill="none" stroke="currentColor" strokeWidth={2} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M15.75 19.5L8.25 12l7.5-7.5" />
</svg>
</button>
<div>
<h1 className="text-xl font-bold text-gray-900"></h1>
<p className="text-sm text-gray-500"> </p>
</div>
</div>
{/* KPI */}
<div className="bg-white rounded-2xl border border-gray-200 p-4">
<div className="grid grid-cols-5 text-center divide-x divide-gray-100">
{[
{ value: loading ? "-" : kpi.total, label: "전체", color: "text-gray-900" },
{ value: loading ? "-" : kpi.active, label: "가동", color: "text-green-600" },
{ value: loading ? "-" : kpi.idle, label: "대기", color: "text-blue-600" },
{ value: loading ? "-" : kpi.inspect, label: "점검", color: "text-red-600" },
{ value: loading ? "-" : kpi.rate, label: "가동률", color: "text-purple-600" },
].map((item) => (
<div key={item.label} className="px-2">
<p className={`text-2xl font-extrabold ${item.color}`}>{item.value}</p>
<p className="text-xs text-gray-400 mt-0.5">{item.label}</p>
</div>
))}
</div>
</div>
</div>
{/* Menu Icons */}
<div className="px-5 pt-5">
<h2 className="flex items-center gap-2 text-sm font-bold text-gray-700 mb-3">
<span className="w-1 h-4 bg-purple-500 rounded" />
</h2>
<div className="flex gap-4">
{MENU_ITEMS.map((item) => (
<button
key={item.id}
onClick={() => router.push(item.href)}
className="flex flex-col items-center gap-2 active:scale-95 transition-transform"
>
<div
className="w-14 h-14 rounded-2xl flex items-center justify-center"
style={{
background: item.gradient,
boxShadow: `0 4px 12px ${item.shadowColor}`,
}}
>
{item.icon}
</div>
<span className="text-xs font-semibold text-gray-700">{item.title}</span>
</button>
))}
</div>
</div>
{/* Recent Equipment */}
<div className="px-5 pt-6 pb-24">
<div className="bg-white rounded-2xl border border-gray-200 overflow-hidden">
<div className="flex items-center justify-between px-4 py-3 border-b border-gray-100">
<h3 className="font-bold text-gray-900"> </h3>
<span className="text-xs text-gray-400"> 5</span>
</div>
{loading ? (
<div className="flex justify-center py-12">
<div className="w-6 h-6 border-3 border-purple-500 border-t-transparent rounded-full animate-spin" />
</div>
) : recentItems.length === 0 ? (
<div className="text-center py-12 text-gray-400 text-sm"> </div>
) : (
<div className="divide-y divide-gray-50">
{recentItems.map((item) => (
<div key={item.id} className="flex items-center justify-between px-4 py-3">
<div>
<p className="text-sm font-semibold text-gray-900">{item.equipment_name || "-"}</p>
<p className="text-xs text-gray-400">{item.equipment_code} {item.equipment_type ? `· ${item.equipment_type}` : ""}</p>
</div>
<span className={`text-xs px-2 py-0.5 rounded-full font-semibold ${
item.status === "정지" || item.status === "비가동"
? "bg-red-50 text-red-600"
: item.status === "점검"
? "bg-amber-50 text-amber-600"
: "bg-green-50 text-green-600"
}`}>
{item.status || "정상"}
</span>
</div>
))}
</div>
)}
</div>
</div>
</div>
);
}
@@ -0,0 +1,119 @@
"use client";
import React, { useState } from "react";
import { useRouter } from "next/navigation";
import { PopShell } from "../PopShell";
type TabType = "all" | "running" | "idle" | "inspect" | "stopped";
export function EquipmentInspection() {
const router = useRouter();
const [activeTab, setActiveTab] = useState<TabType>("all");
const [searchKeyword, setSearchKeyword] = useState("");
// 시안 기준 KPI (점검 테이블 없으므로 0)
const kpi = { total: 0, running: 0, idle: 0, inspect: 0, rate: "0%" };
const tabs: { key: TabType; label: string; count: number }[] = [
{ key: "all", label: "전체", count: kpi.total },
{ key: "running", label: "가동", count: kpi.running },
{ key: "idle", label: "대기", count: kpi.idle },
{ key: "inspect", label: "점검", count: kpi.inspect },
{ key: "stopped", label: "비가동", count: 0 },
];
return (
<PopShell title="점검관리" showBanner={false}>
<div className="min-h-screen bg-gray-50">
{/* Header */}
<div className="bg-white border-b border-gray-200 px-5 pt-5 pb-4">
<div className="flex items-center gap-3 mb-4">
<button onClick={() => router.back()} className="w-9 h-9 rounded-xl bg-gray-100 flex items-center justify-center">
<svg className="w-5 h-5 text-gray-600" fill="none" stroke="currentColor" strokeWidth={2} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M15.75 19.5L8.25 12l7.5-7.5" />
</svg>
</button>
<div>
<h1 className="text-xl font-bold text-gray-900">🔧 </h1>
</div>
</div>
{/* Search */}
<div className="flex gap-2">
<input
type="text"
placeholder="설비명 / 설비코드 검색"
value={searchKeyword}
onChange={(e) => setSearchKeyword(e.target.value)}
className="flex-1 px-4 py-3 rounded-xl border border-gray-200 text-sm focus:outline-none focus:border-amber-400"
/>
<select className="px-3 py-3 rounded-xl border border-gray-200 text-sm bg-white">
<option></option>
<option></option>
<option></option>
<option></option>
</select>
<button className="px-5 py-3 rounded-xl bg-amber-500 text-white text-sm font-bold active:bg-amber-600">
🔍
</button>
</div>
</div>
{/* KPI */}
<div className="px-5 py-3">
<div className="grid grid-cols-5 gap-2">
{[
{ label: "전체", value: kpi.total, color: "border-t-amber-500", icon: "🎬" },
{ label: "가동", value: kpi.running, color: "border-t-green-500", icon: "🟢" },
{ label: "대기", value: kpi.idle, color: "border-t-blue-500", icon: "🔵" },
{ label: "점검", value: kpi.inspect, color: "border-t-red-500", icon: "🔴" },
{ label: "가동률", value: kpi.rate, color: "border-t-purple-500", icon: "📊" },
].map((item) => (
<div key={item.label} className={`bg-white rounded-xl border border-gray-200 ${item.color} border-t-[3px] p-3 text-center`}>
<p className="text-lg mb-0.5">{item.icon}</p>
<p className="text-xl font-bold text-gray-900">{item.value}</p>
<p className="text-[10px] text-gray-500">{item.label}</p>
</div>
))}
</div>
</div>
{/* Tabs */}
<div className="px-5">
<div className="flex bg-white rounded-xl border border-gray-200 overflow-hidden">
{tabs.map((tab) => (
<button
key={tab.key}
onClick={() => setActiveTab(tab.key)}
className={`flex-1 py-3 text-xs font-medium text-center relative transition-all ${
activeTab === tab.key
? "text-amber-600 font-bold bg-amber-50"
: "text-gray-500"
}`}
>
{tab.label}
<span className={`ml-1 text-[9px] px-1.5 py-0.5 rounded-full font-bold ${
activeTab === tab.key ? "bg-amber-500 text-white" : "bg-gray-200 text-gray-500"
}`}>
{tab.count}
</span>
{activeTab === tab.key && (
<span className="absolute bottom-0 left-[20%] right-[20%] h-[3px] bg-amber-500 rounded-t" />
)}
</button>
))}
</div>
</div>
{/* Card List — 데이터 없음 */}
<div className="px-5 py-6">
<div className="flex flex-col items-center justify-center py-20 text-gray-400">
<span className="text-5xl mb-4">🔧</span>
<p className="text-base font-semibold mb-1"> </p>
<p className="text-sm">PC에서 </p>
</div>
</div>
</div>
</PopShell>
);
}
@@ -0,0 +1,169 @@
"use client";
import React, { useState, useEffect, useCallback } from "react";
import { useRouter } from "next/navigation";
import { apiClient } from "@/lib/api/client";
import { PopShell } from "../PopShell";
interface EquipmentItem {
id: string;
equipment_code: string;
equipment_name: string;
equipment_type?: string;
location?: string;
status?: string;
last_inspection_date?: string;
next_inspection_date?: string;
memo?: string;
}
export function EquipmentList() {
const router = useRouter();
const [items, setItems] = useState<EquipmentItem[]>([]);
const [loading, setLoading] = useState(true);
const [searchKeyword, setSearchKeyword] = useState("");
const fetchEquipments = useCallback(async () => {
setLoading(true);
try {
const res = await apiClient.get("/data/equipment_mng", {
params: { pageSize: 500 },
});
const data = res.data?.data?.data ?? res.data?.data ?? [];
setItems(Array.isArray(data) ? data : []);
} catch {
setItems([]);
} finally {
setLoading(false);
}
}, []);
useEffect(() => {
fetchEquipments();
}, [fetchEquipments]);
const filtered = items.filter((item) => {
if (!searchKeyword) return true;
const kw = searchKeyword.toLowerCase();
return (
(item.equipment_code || "").toLowerCase().includes(kw) ||
(item.equipment_name || "").toLowerCase().includes(kw) ||
(item.equipment_type || "").toLowerCase().includes(kw)
);
});
// KPI
const totalCount = items.length;
const activeCount = items.filter((i) => i.status === "가동" || i.status === "정상" || !i.status).length;
const stopCount = items.filter((i) => i.status === "정지" || i.status === "비가동").length;
return (
<PopShell title="설비관리" showBanner={false}>
<div className="min-h-screen bg-gray-50">
{/* Header */}
<div className="bg-white border-b border-gray-200 px-5 pt-5 pb-4">
<div className="flex items-center gap-3 mb-4">
<button
onClick={() => router.back()}
className="w-9 h-9 rounded-xl bg-gray-100 flex items-center justify-center"
>
<svg className="w-5 h-5 text-gray-600" fill="none" stroke="currentColor" strokeWidth={2} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M15.75 19.5L8.25 12l7.5-7.5" />
</svg>
</button>
<div>
<h1 className="text-xl font-bold text-gray-900"></h1>
<p className="text-sm text-gray-500"> </p>
</div>
</div>
{/* Search */}
<div className="relative">
<svg className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-gray-400" fill="none" stroke="currentColor" strokeWidth={2} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M21 21l-5.197-5.197m0 0A7.5 7.5 0 105.196 5.196a7.5 7.5 0 0010.607 10.607z" />
</svg>
<input
type="text"
placeholder="설비명 또는 코드 검색..."
value={searchKeyword}
onChange={(e) => setSearchKeyword(e.target.value)}
className="w-full pl-10 pr-4 py-3 rounded-xl border border-gray-200 text-sm focus:outline-none focus:border-blue-400"
/>
</div>
</div>
{/* KPI */}
<div className="px-5 py-3">
<div className="grid grid-cols-3 gap-3">
<div className="bg-white rounded-xl border border-gray-200 p-3 text-center">
<p className="text-xs text-gray-400"></p>
<p className="text-2xl font-bold text-gray-900">{totalCount}</p>
</div>
<div className="bg-white rounded-xl border border-green-200 p-3 text-center">
<p className="text-xs text-green-500"></p>
<p className="text-2xl font-bold text-green-600">{activeCount}</p>
</div>
<div className="bg-white rounded-xl border border-red-200 p-3 text-center">
<p className="text-xs text-red-500"></p>
<p className="text-2xl font-bold text-red-600">{stopCount}</p>
</div>
</div>
</div>
{/* List */}
<div className="px-5 pb-24">
<div className="flex items-center justify-between mb-3">
<h2 className="text-sm font-semibold text-gray-500"> </h2>
<span className="text-xs text-gray-400">{filtered.length}</span>
</div>
{loading ? (
<div className="flex items-center justify-center py-20">
<div className="w-8 h-8 border-4 border-blue-500 border-t-transparent rounded-full animate-spin" />
</div>
) : filtered.length === 0 ? (
<div className="flex flex-col items-center justify-center py-20 text-gray-400">
<svg className="w-16 h-16 mb-4 text-gray-200" fill="none" stroke="currentColor" strokeWidth={1} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M11.42 15.17l-5.65-5.65a8 8 0 1111.31 0l-5.65 5.65z" />
</svg>
<p className="text-sm"> </p>
</div>
) : (
<div className="space-y-3">
{filtered.map((item) => (
<div
key={item.id}
className="bg-white rounded-2xl border border-gray-200 p-4 hover:shadow-md transition-shadow"
>
<div className="flex items-start justify-between mb-2">
<div>
<h3 className="font-bold text-gray-900">{item.equipment_name || "-"}</h3>
<p className="text-xs text-gray-400">{item.equipment_code}</p>
</div>
<span
className={`text-xs px-2.5 py-1 rounded-full font-semibold ${
item.status === "정지" || item.status === "비가동"
? "bg-red-100 text-red-600"
: "bg-green-100 text-green-600"
}`}
>
{item.status || "정상"}
</span>
</div>
<div className="grid grid-cols-2 gap-2 text-xs text-gray-500">
{item.equipment_type && (
<div>: <span className="text-gray-700 font-medium">{item.equipment_type}</span></div>
)}
{item.location && (
<div>: <span className="text-gray-700 font-medium">{item.location}</span></div>
)}
</div>
</div>
))}
</div>
)}
</div>
</div>
</PopShell>
);
}
@@ -95,7 +95,29 @@ const MENU_ITEMS = [
/> />
</svg> </svg>
), ),
href: "#", href: "/pop/inventory/transfer",
},
{
id: "move",
title: "재고이동",
gradient: "linear-gradient(135deg,#10b981,#059669)",
shadowColor: "rgba(16,185,129,.3)",
icon: (
<svg
className="w-7 h-7 text-white"
fill="none"
stroke="currentColor"
strokeWidth={1.5}
viewBox="0 0 24 24"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M7.5 21L3 16.5m0 0L7.5 12M3 16.5h13.5m0-13.5L21 7.5m0 0L16.5 12M21 7.5H7.5"
/>
</svg>
),
href: "/pop/inventory/move",
}, },
]; ];
@@ -0,0 +1,289 @@
"use client";
import React, { useState, useEffect, useCallback } from "react";
import { useRouter } from "next/navigation";
import { apiClient } from "@/lib/api/client";
import { PopShell } from "../PopShell";
interface Warehouse {
id: string;
warehouse_code: string;
warehouse_name: string;
}
interface StockItem {
id: string;
item_code: string;
item_name?: string;
warehouse_code: string;
location_code?: string;
current_qty: string;
}
interface PendingItem {
stock: StockItem;
moveQty: number;
toWarehouse: string;
}
export function InventoryMove() {
const router = useRouter();
const [warehouses, setWarehouses] = useState<Warehouse[]>([]);
const [fromWarehouse, setFromWarehouse] = useState("");
const [toWarehouse, setToWarehouse] = useState("");
const [stockItems, setStockItems] = useState<StockItem[]>([]);
const [loading, setLoading] = useState(false);
const [searchKeyword, setSearchKeyword] = useState("");
const [pendingItems, setPendingItems] = useState<PendingItem[]>([]);
const fetchWarehouses = useCallback(async () => {
try {
const res = await apiClient.get("/outbound/warehouses");
setWarehouses(res.data?.data || []);
} catch { /* */ }
}, []);
const fetchStock = useCallback(async () => {
if (!fromWarehouse) { setStockItems([]); return; }
setLoading(true);
try {
const res = await apiClient.get("/data/inventory_stock", {
params: { pageSize: "500", filters: JSON.stringify({ warehouse_code: fromWarehouse }) },
});
const data = res.data?.data?.data ?? res.data?.data ?? [];
setStockItems(Array.isArray(data) ? data.filter((d: StockItem) => parseFloat(d.current_qty || "0") > 0) : []);
} catch {
setStockItems([]);
} finally {
setLoading(false);
}
}, [fromWarehouse]);
useEffect(() => { fetchWarehouses(); }, [fetchWarehouses]);
useEffect(() => { fetchStock(); }, [fetchStock]);
const filtered = stockItems.filter((item) => {
if (!searchKeyword) return true;
const kw = searchKeyword.toLowerCase();
return (item.item_code || "").toLowerCase().includes(kw) || (item.item_name || "").toLowerCase().includes(kw);
});
const addToPending = (stock: StockItem) => {
if (!toWarehouse) { alert("도착 창고를 먼저 선택하세요."); return; }
if (pendingItems.find((p) => p.stock.id === stock.id)) return;
const qty = parseFloat(stock.current_qty || "0");
setPendingItems((prev) => [...prev, { stock, moveQty: qty, toWarehouse }]);
};
const removePending = (id: string) => {
setPendingItems((prev) => prev.filter((p) => p.stock.id !== id));
};
const fromWh = warehouses.find((w) => w.warehouse_code === fromWarehouse);
const toWh = warehouses.find((w) => w.warehouse_code === toWarehouse);
return (
<PopShell title="재고이동" showBanner={false}>
<div className="flex flex-col h-screen bg-gray-100">
{/* Header */}
<div className="bg-white border-b border-gray-200 px-4 py-3 flex items-center gap-3 shrink-0">
<button onClick={() => router.back()} className="w-9 h-9 rounded-xl bg-gray-100 flex items-center justify-center">
<svg className="w-5 h-5 text-gray-600" fill="none" stroke="currentColor" strokeWidth={2} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M15.75 19.5L8.25 12l7.5-7.5" />
</svg>
</button>
<div className="flex-1">
<h1 className="text-lg font-bold text-gray-900">📦 </h1>
<p className="text-xs text-gray-500"> </p>
</div>
</div>
{/* 좌우 분할 */}
<div className="flex-1 flex overflow-hidden">
{/* ===== 왼쪽: 출발 창고 + 품목 선택 ===== */}
<div className="flex-1 flex flex-col bg-white border-r-2 border-gray-200">
{/* 출발 창고 헤더 */}
<div className="px-4 py-3 border-b border-gray-100 bg-blue-50">
<div className="flex items-center justify-between mb-2">
<span className="text-sm font-bold text-blue-800">📤 </span>
<span className="text-xs px-2 py-0.5 rounded-full bg-blue-100 text-blue-600 font-semibold">FROM</span>
</div>
<div className="flex gap-2 overflow-x-auto pb-1">
{warehouses.map((wh) => (
<button
key={wh.warehouse_code}
onClick={() => { setFromWarehouse(wh.warehouse_code); setPendingItems([]); }}
className={`px-4 py-2.5 rounded-xl text-sm font-semibold whitespace-nowrap transition-all ${
fromWarehouse === wh.warehouse_code
? "bg-blue-600 text-white shadow-md"
: "bg-white text-gray-600 border border-gray-200"
}`}
>
{wh.warehouse_name}
</button>
))}
</div>
</div>
{/* 검색 */}
{fromWarehouse && (
<div className="px-4 py-2 border-b border-gray-100">
<div className="flex gap-2">
<input
type="text"
placeholder="품목명 / 코드 검색"
value={searchKeyword}
onChange={(e) => setSearchKeyword(e.target.value)}
className="flex-1 px-4 py-2.5 rounded-xl border border-gray-200 text-sm focus:outline-none focus:border-blue-400"
/>
<button className="px-4 py-2.5 rounded-xl bg-blue-500 text-white text-sm font-bold">🔍</button>
</div>
</div>
)}
{/* 품목 리스트 */}
<div className="flex-1 overflow-y-auto px-3 py-2">
{!fromWarehouse ? (
<div className="flex flex-col items-center justify-center py-20 text-gray-400">
<span className="text-4xl mb-3">📦</span>
<p className="text-base font-semibold"> </p>
</div>
) : loading ? (
<div className="flex justify-center py-16">
<div className="w-8 h-8 border-4 border-blue-500 border-t-transparent rounded-full animate-spin" />
</div>
) : filtered.length === 0 ? (
<div className="flex flex-col items-center justify-center py-16 text-gray-400">
<p className="text-sm"> </p>
</div>
) : (
<div className="space-y-2">
{filtered.map((item) => {
const isPending = pendingItems.some((p) => p.stock.id === item.id);
return (
<button
key={item.id}
onClick={() => addToPending(item)}
disabled={isPending}
className={`w-full text-left p-3.5 rounded-xl border transition-all active:scale-[0.98] ${
isPending
? "bg-green-50 border-green-300 opacity-60"
: "bg-white border-gray-200 hover:border-blue-300 hover:shadow-sm"
}`}
>
<div className="flex items-center justify-between">
<div>
<p className="text-base font-bold text-gray-900">{item.item_name || item.item_code}</p>
<p className="text-xs text-gray-400">{item.item_code} · {item.location_code || item.warehouse_code}</p>
</div>
<div className="text-right">
<p className="text-xl font-bold text-blue-600">{parseFloat(item.current_qty || "0").toLocaleString()}</p>
</div>
</div>
</button>
);
})}
</div>
)}
</div>
</div>
{/* ===== 오른쪽: 도착 창고 + 이동 대기열 ===== */}
<div className="flex-1 flex flex-col bg-gray-50">
{/* 도착 창고 헤더 */}
<div className="px-4 py-3 border-b border-gray-100 bg-green-50">
<div className="flex items-center justify-between mb-2">
<span className="text-sm font-bold text-green-800">📥 </span>
<span className="text-xs px-2 py-0.5 rounded-full bg-green-100 text-green-600 font-semibold">TO</span>
</div>
<div className="flex gap-2 overflow-x-auto pb-1">
{warehouses
.filter((wh) => wh.warehouse_code !== fromWarehouse)
.map((wh) => (
<button
key={wh.warehouse_code}
onClick={() => setToWarehouse(wh.warehouse_code)}
className={`px-4 py-2.5 rounded-xl text-sm font-semibold whitespace-nowrap transition-all ${
toWarehouse === wh.warehouse_code
? "bg-green-600 text-white shadow-md"
: "bg-white text-gray-600 border border-gray-200"
}`}
>
{wh.warehouse_name}
</button>
))}
</div>
</div>
{/* 이동 방향 표시 */}
{fromWh && toWh && (
<div className="px-4 py-2 bg-white border-b border-gray-200 flex items-center justify-center gap-3">
<span className="text-sm font-bold text-blue-600">{fromWh.warehouse_name}</span>
<span className="text-lg"></span>
<span className="text-sm font-bold text-green-600">{toWh.warehouse_name}</span>
</div>
)}
{/* 이동 대기 목록 */}
<div className="flex-1 overflow-y-auto px-3 py-2">
{pendingItems.length === 0 ? (
<div className="flex flex-col items-center justify-center py-20 text-gray-400">
<span className="text-4xl mb-3">📋</span>
<p className="text-base font-semibold"> </p>
<p className="text-sm mt-1"> </p>
</div>
) : (
<div className="space-y-2">
{pendingItems.map((p) => (
<div key={p.stock.id} className="bg-white rounded-xl border-l-4 border-l-blue-500 border border-gray-200 p-3.5">
<div className="flex items-center justify-between mb-1">
<p className="text-base font-bold text-gray-900">{p.stock.item_name || p.stock.item_code}</p>
<button
onClick={() => removePending(p.stock.id)}
className="px-3 py-1.5 rounded-lg border border-red-200 bg-red-50 text-red-600 text-sm font-bold active:bg-red-100"
>
</button>
</div>
<p className="text-xs text-gray-400 mb-2">{p.stock.item_code}</p>
<div className="flex items-center gap-2">
<span className="text-sm font-bold text-blue-600 bg-blue-50 px-2 py-0.5 rounded">
{p.moveQty.toLocaleString()} EA
</span>
<span className="text-xs text-gray-400">
{p.stock.warehouse_code} {p.toWarehouse}
</span>
</div>
</div>
))}
</div>
)}
</div>
{/* 하단 확정 바 */}
<div className="px-4 py-3 border-t border-gray-200 bg-white flex items-center justify-between shrink-0">
<div className="text-sm text-gray-500">
: <strong className="text-blue-600">{pendingItems.length}</strong>
</div>
<button
onClick={() => alert("재고 이동 API 준비 중입니다.")}
disabled={pendingItems.length === 0}
className={`px-6 py-3 rounded-xl text-base font-bold text-white active:scale-[0.98] transition-all ${
pendingItems.length > 0
? "bg-red-500 hover:bg-red-600"
: "bg-gray-300"
}`}
>
{pendingItems.length > 0 && (
<span className="ml-1 bg-white text-red-500 text-xs font-bold px-2 py-0.5 rounded-full">
{pendingItems.length}
</span>
)}
</button>
</div>
</div>
</div>
</div>
</PopShell>
);
}
@@ -0,0 +1,252 @@
"use client";
import React, { useState, useEffect, useCallback } from "react";
import { useRouter } from "next/navigation";
import { apiClient } from "@/lib/api/client";
import { PopShell } from "../PopShell";
interface Warehouse {
id: string;
warehouse_code: string;
warehouse_name: string;
warehouse_type?: string;
}
interface StockItem {
id: string;
item_code: string;
item_name?: string;
warehouse_code: string;
location_code?: string;
current_qty: string;
unit?: string;
}
interface SelectedItem {
stock: StockItem;
adjustQty: string;
type: "confirm" | "adjust";
}
export function InventoryTransfer() {
const router = useRouter();
const [warehouses, setWarehouses] = useState<Warehouse[]>([]);
const [selectedWarehouse, setSelectedWarehouse] = useState("all");
const [stockItems, setStockItems] = useState<StockItem[]>([]);
const [loading, setLoading] = useState(true);
const [searchKeyword, setSearchKeyword] = useState("");
const [selectedItems, setSelectedItems] = useState<SelectedItem[]>([]);
const fetchWarehouses = useCallback(async () => {
try {
const res = await apiClient.get("/outbound/warehouses");
setWarehouses(res.data?.data || []);
} catch { /* */ }
}, []);
const fetchStock = useCallback(async () => {
setLoading(true);
try {
const params: Record<string, string> = { pageSize: "500" };
if (selectedWarehouse !== "all") {
params.filters = JSON.stringify({ warehouse_code: selectedWarehouse });
}
const res = await apiClient.get("/data/inventory_stock", { params });
const data = res.data?.data?.data ?? res.data?.data ?? [];
setStockItems(Array.isArray(data) ? data.filter((d: StockItem) => parseFloat(d.current_qty || "0") > 0) : []);
} catch {
setStockItems([]);
} finally {
setLoading(false);
}
}, [selectedWarehouse]);
useEffect(() => { fetchWarehouses(); }, [fetchWarehouses]);
useEffect(() => { fetchStock(); }, [fetchStock]);
const filtered = stockItems.filter((item) => {
if (!searchKeyword) return true;
const kw = searchKeyword.toLowerCase();
return (item.item_code || "").toLowerCase().includes(kw) || (item.item_name || "").toLowerCase().includes(kw);
});
const addItem = (stock: StockItem) => {
if (selectedItems.find((s) => s.stock.id === stock.id)) return;
setSelectedItems((prev) => [...prev, { stock, adjustQty: "", type: "confirm" }]);
};
const removeItem = (id: string) => {
setSelectedItems((prev) => prev.filter((s) => s.stock.id !== id));
};
const confirmCount = selectedItems.filter((s) => s.type === "confirm").length;
const adjustCount = selectedItems.filter((s) => s.type === "adjust").length;
return (
<PopShell title="재고조정" showBanner={false}>
<div className="min-h-screen bg-gray-50 flex flex-col">
{/* Header */}
<div className="bg-white border-b border-gray-200 px-5 pt-5 pb-4 shrink-0">
<div className="flex items-center gap-3">
<button onClick={() => router.back()} className="w-9 h-9 rounded-xl bg-gray-100 flex items-center justify-center">
<svg className="w-5 h-5 text-gray-600" fill="none" stroke="currentColor" strokeWidth={2} viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" d="M15.75 19.5L8.25 12l7.5-7.5" />
</svg>
</button>
<h1 className="text-xl font-bold text-gray-900">📦 </h1>
</div>
</div>
{/* Main — 2단 레이아웃 */}
<div className="flex-1 flex flex-col lg:flex-row overflow-hidden">
{/* 왼쪽: 제품 선택 */}
<div className="flex-1 flex flex-col border-r border-gray-200 bg-white">
<div className="px-4 pt-4 pb-2 border-b border-gray-100">
<div className="flex items-center justify-between mb-3">
<h2 className="text-sm font-bold text-gray-700">📦 </h2>
<button className="px-3 py-1.5 text-xs rounded-lg border border-gray-200 text-gray-500">📋 </button>
</div>
{/* 창고 탭 */}
<div className="flex gap-2 mb-3 overflow-x-auto pb-1">
<button
onClick={() => setSelectedWarehouse("all")}
className={`px-4 py-2 rounded-full text-xs font-semibold whitespace-nowrap transition-all ${
selectedWarehouse === "all" ? "bg-amber-500 text-white" : "bg-gray-100 text-gray-600"
}`}
>
</button>
{warehouses.map((wh) => (
<button
key={wh.warehouse_code}
onClick={() => setSelectedWarehouse(wh.warehouse_code)}
className={`px-4 py-2 rounded-full text-xs font-semibold whitespace-nowrap transition-all ${
selectedWarehouse === wh.warehouse_code ? "bg-amber-500 text-white" : "bg-gray-100 text-gray-600"
}`}
>
{wh.warehouse_name}
</button>
))}
</div>
{/* 검색 */}
<div className="flex gap-2">
<input
type="text"
placeholder="품목명 / 코드 검색"
value={searchKeyword}
onChange={(e) => setSearchKeyword(e.target.value)}
className="flex-1 px-4 py-2.5 rounded-xl border border-gray-200 text-sm focus:outline-none focus:border-amber-400"
/>
<button className="px-4 py-2.5 rounded-xl bg-amber-500 text-white text-sm font-bold">🔍</button>
</div>
</div>
{/* 품목 리스트 */}
<div className="flex-1 overflow-y-auto px-4 py-2">
{loading ? (
<div className="flex justify-center py-16">
<div className="w-8 h-8 border-4 border-amber-500 border-t-transparent rounded-full animate-spin" />
</div>
) : filtered.length === 0 ? (
<div className="flex flex-col items-center justify-center py-16 text-gray-400">
<span className="text-4xl mb-3">📦</span>
<p className="text-sm"> </p>
</div>
) : (
<div className="divide-y divide-gray-100">
{filtered.map((item) => (
<div key={item.id} className="flex items-center justify-between py-3">
<div className="flex items-center gap-3">
<div className="w-8 h-8 rounded-lg bg-amber-50 flex items-center justify-center text-sm">📦</div>
<div>
<p className="text-sm font-bold text-gray-900">
{item.item_name || item.item_code}
{item.item_name && <span className="text-gray-400 font-normal"> ({item.item_code})</span>}
</p>
<p className="text-xs text-gray-400">{item.warehouse_code}{item.location_code ? ` · ${item.location_code}` : ""}</p>
</div>
</div>
<div className="flex items-center gap-3">
<div className="text-right">
<p className="text-lg font-bold text-gray-900">{parseFloat(item.current_qty || "0").toLocaleString()}</p>
<p className="text-[10px] text-gray-400">{item.location_code || item.warehouse_code}</p>
</div>
<button
onClick={() => addItem(item)}
className={`w-8 h-8 rounded-full flex items-center justify-center text-white text-lg font-bold transition-all ${
selectedItems.find((s) => s.stock.id === item.id)
? "bg-gray-300"
: "bg-amber-500 active:bg-amber-600"
}`}
>
+
</button>
</div>
</div>
))}
</div>
)}
</div>
</div>
{/* 오른쪽: 처리 결과 */}
<div className="w-full lg:w-[400px] bg-gray-50 flex flex-col">
<div className="px-4 pt-4 pb-2 border-b border-gray-200 bg-white flex items-center justify-between">
<h2 className="text-sm font-bold text-gray-700">📋 </h2>
<span className="text-xs font-bold text-amber-600 bg-amber-50 px-2.5 py-1 rounded-full">
{selectedItems.length}
</span>
</div>
<div className="flex-1 overflow-y-auto px-4 py-3">
{selectedItems.length === 0 ? (
<div className="flex flex-col items-center justify-center py-16 text-gray-400">
<span className="text-4xl mb-3">📋</span>
<p className="text-sm"> / </p>
</div>
) : (
<div className="space-y-3">
{selectedItems.map((sel) => (
<div key={sel.stock.id} className="bg-white rounded-xl border border-gray-200 p-3">
<div className="flex items-center justify-between mb-2">
<p className="text-sm font-bold text-gray-900">{sel.stock.item_name || sel.stock.item_code}</p>
<button onClick={() => removeItem(sel.stock.id)} className="text-xs text-red-500 font-semibold"></button>
</div>
<p className="text-xs text-gray-400 mb-2"> : {parseFloat(sel.stock.current_qty || "0").toLocaleString()}</p>
<input
type="number"
placeholder="조정 수량"
className="w-full px-3 py-2 rounded-lg border border-gray-200 text-sm focus:outline-none focus:border-amber-400"
/>
</div>
))}
</div>
)}
</div>
{/* Footer */}
<div className="px-4 py-3 border-t border-gray-200 bg-white flex items-center justify-between">
<div className="flex gap-3 text-xs">
<span className="text-blue-600 font-semibold"> {confirmCount}</span>
<span className="text-amber-600 font-semibold"> {adjustCount}</span>
</div>
<div className="flex gap-2">
<button
onClick={() => setSelectedItems([])}
className="px-4 py-2.5 rounded-xl border border-gray-200 text-xs font-semibold text-gray-600"
>
</button>
<button className="px-4 py-2.5 rounded-xl bg-amber-500 text-white text-xs font-bold active:bg-amber-600">
</button>
</div>
</div>
</div>
</div>
</div>
</PopShell>
);
}
@@ -1208,34 +1208,45 @@ export function ProcessWork({ processId }: ProcessWorkProps) {
behavior: "smooth", behavior: "smooth",
}); });
}} }}
className={`w-full flex items-center gap-2 px-3 py-2 text-left transition-all ${ className={`w-full flex items-center gap-3 mx-2 mb-1.5 px-3 py-3 rounded-xl text-left transition-all ${
isSelected isSelected
? "border-l-[3px] border-l-gray-900 bg-white" ? "bg-blue-50 border-2 border-blue-400 shadow-sm"
: "border-l-[3px] border-l-transparent hover:bg-gray-50" : isDone
? "bg-green-50/50 border border-green-200"
: "bg-white border border-gray-200 hover:border-blue-300 hover:shadow-sm"
}`} }`}
style={{ width: "calc(100% - 16px)" }}
> >
<span <span
className={`w-2 h-2 rounded-full shrink-0 ${ className={`w-3 h-3 rounded-full shrink-0 ${
isDone isDone
? "bg-green-500" ? "bg-green-500"
: g.timerStarted : g.timerStarted
? "bg-blue-500 animate-pulse" ? "bg-blue-500 animate-pulse"
: "bg-gray-300" : isSelected
? "bg-blue-500"
: "bg-gray-300"
}`} }`}
/> />
<span <span
className={`text-sm truncate flex-1 ${ className={`text-sm flex-1 ${
isSelected isSelected
? "font-semibold text-blue-700" ? "font-bold text-blue-800"
: isDone : isDone
? "text-gray-400" ? "text-green-700 font-medium"
: "text-gray-700" : "text-gray-700 font-medium"
}`} }`}
> >
{g.title} {g.title}
</span> </span>
<span <span
className={`text-[13px] shrink-0 ${isDone ? "text-green-500" : "text-gray-300"}`} className={`text-xs font-bold px-2 py-0.5 rounded-full ${
isDone
? "bg-green-100 text-green-600"
: isSelected
? "bg-blue-100 text-blue-600"
: "bg-gray-100 text-gray-400"
}`}
> >
{g.completed}/{g.total} {g.completed}/{g.total}
</span> </span>
@@ -1259,41 +1270,22 @@ export function ProcessWork({ processId }: ProcessWorkProps) {
behavior: "smooth", behavior: "smooth",
}); });
}} }}
className={`w-full flex items-center gap-2 px-3 py-2.5 mb-2 text-left transition-all ${ className={`w-full flex items-center gap-3 mx-2 mb-1.5 px-3 py-3 rounded-xl text-left transition-all ${
activeSection === "material" activeSection === "material"
? "border-l-[3px] border-l-gray-900 bg-white" ? "bg-blue-50 border-2 border-blue-400 shadow-sm"
: "border-l-[3px] border-l-transparent hover:bg-gray-50" : "bg-white border border-gray-200 hover:border-blue-300 hover:shadow-sm"
}`} }`}
style={{ width: "calc(100% - 16px)" }}
> >
<span className="text-sm">📦</span> <span className="text-base">📦</span>
<span <span
className={`text-sm ${activeSection === "material" ? "font-semibold text-blue-700" : "text-gray-600 font-medium"}`} className={`text-sm font-medium ${activeSection === "material" ? "font-bold text-blue-800" : "text-gray-700"}`}
> >
</span> </span>
</button> </button>
)} )}
<div className="flex items-center gap-2 px-3 mb-1.5">
<div className="w-4 h-4 rounded-full bg-amber-500 flex items-center justify-center">
<svg
className="w-2.5 h-2.5 text-white"
fill="none"
stroke="currentColor"
strokeWidth={2.5}
viewBox="0 0 24 24"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M9 12h3.75M9 15h3.75M9 18h3.75m3 .75H18a2.25 2.25 0 002.25-2.25V6.108c0-1.135-.845-2.098-1.976-2.192a48.424 48.424 0 00-1.123-.08m-5.801 0c-.065.21-.1.433-.1.664 0 .414.336.75.75.75h4.5a.75.75 0 00.75-.75 2.25 2.25 0 00-.1-.664m-5.8 0A2.251 2.251 0 0113.5 2.25H15a2.25 2.25 0 012.15 1.586m-5.8 0c-.376.023-.75.05-1.124.08C9.095 4.01 8.25 4.973 8.25 6.108V19.5a2.25 2.25 0 002.25 2.25h.75"
/>
</svg>
</div>
<span className="text-base font-bold text-gray-400 uppercase tracking-wider">
</span>
</div>
<button <button
onClick={() => { onClick={() => {
setActiveSection("result"); setActiveSection("result");
@@ -1302,27 +1294,16 @@ export function ProcessWork({ processId }: ProcessWorkProps) {
behavior: "smooth", behavior: "smooth",
}); });
}} }}
className={`w-full flex items-center gap-2 px-3 py-2 text-left transition-all ${ className={`w-full flex items-center gap-3 mx-2 mb-1.5 px-3 py-3 rounded-xl text-left transition-all ${
activeSection === "result" activeSection === "result"
? "border-l-[3px] border-l-gray-900 bg-white" ? "bg-blue-50 border-2 border-blue-400 shadow-sm"
: "border-l-[3px] border-l-transparent hover:bg-gray-50" : "bg-white border border-gray-200 hover:border-blue-300 hover:shadow-sm"
}`} }`}
style={{ width: "calc(100% - 16px)" }}
> >
<svg <span className="text-base">📋</span>
className={`w-3.5 h-3.5 ${activeSection === "result" ? "text-blue-500" : "text-amber-500"}`}
fill="none"
stroke="currentColor"
strokeWidth={2}
viewBox="0 0 24 24"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M12 9v6m3-3H9m12 0a9 9 0 11-18 0 9 9 0 0118 0z"
/>
</svg>
<span <span
className={`text-sm ${activeSection === "result" ? "font-semibold text-blue-700" : "text-amber-700 font-medium"}`} className={`text-sm font-medium ${activeSection === "result" ? "font-bold text-blue-800" : "text-gray-700"}`}
> >
</span> </span>
@@ -1333,28 +1314,6 @@ export function ProcessWork({ processId }: ProcessWorkProps) {
{/* Inventory section link */} {/* Inventory section link */}
{isLastProcess && ( {isLastProcess && (
<div> <div>
<div className="flex items-center gap-2 px-3 mb-1.5">
<div
className={`w-4 h-4 rounded-full flex items-center justify-center ${inboundDone ? "bg-green-500" : "bg-amber-500"}`}
>
<svg
className="w-2.5 h-2.5 text-white"
fill="none"
stroke="currentColor"
strokeWidth={2.5}
viewBox="0 0 24 24"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M20.25 7.5l-.625 10.632a2.25 2.25 0 01-2.247 2.118H6.622a2.25 2.25 0 01-2.247-2.118L3.75 7.5M10 11.25h4M3.375 7.5h17.25c.621 0 1.125-.504 1.125-1.125v-1.5c0-.621-.504-1.125-1.125-1.125H3.375c-.621 0-1.125.504-1.125 1.125v1.5c0 .621.504 1.125 1.125 1.125z"
/>
</svg>
</div>
<span className="text-base font-bold text-gray-400 uppercase tracking-wider">
</span>
</div>
<button <button
onClick={() => { onClick={() => {
setActiveSection("inventory"); setActiveSection("inventory");
@@ -1363,45 +1322,27 @@ export function ProcessWork({ processId }: ProcessWorkProps) {
behavior: "smooth", behavior: "smooth",
}); });
}} }}
className={`w-full flex items-center gap-2 px-3 py-2 text-left transition-all ${ className={`w-full flex items-center gap-3 mx-2 mb-1.5 px-3 py-3 rounded-xl text-left transition-all ${
activeSection === "inventory" activeSection === "inventory"
? "border-l-[3px] border-l-gray-900 bg-white" ? "bg-blue-50 border-2 border-blue-400 shadow-sm"
: "border-l-[3px] border-l-transparent hover:bg-gray-50" : inboundDone
? "bg-green-50 border border-green-300"
: "bg-white border border-gray-200 hover:border-blue-300 hover:shadow-sm"
}`} }`}
style={{ width: "calc(100% - 16px)" }}
> >
<svg <span className="text-base">{inboundDone ? "✅" : "🏭"}</span>
className={`w-3.5 h-3.5 ${activeSection === "inventory" ? "text-blue-500" : inboundDone ? "text-green-500" : "text-amber-500"}`}
fill="none"
stroke="currentColor"
strokeWidth={2}
viewBox="0 0 24 24"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M20.25 7.5l-.625 10.632a2.25 2.25 0 01-2.247 2.118H6.622a2.25 2.25 0 01-2.247-2.118L3.75 7.5M10 11.25h4M3.375 7.5h17.25c.621 0 1.125-.504 1.125-1.125v-1.5c0-.621-.504-1.125-1.125-1.125H3.375c-.621 0-1.125.504-1.125 1.125v1.5c0 .621.504 1.125 1.125 1.125z"
/>
</svg>
<span <span
className={`text-sm ${activeSection === "inventory" ? "font-semibold text-blue-700" : inboundDone ? "text-green-700 font-medium" : "text-amber-700 font-medium"}`} className={`text-sm font-medium ${
activeSection === "inventory"
? "font-bold text-blue-800"
: inboundDone
? "text-green-700 font-bold"
: "text-gray-700"
}`}
> >
{inboundDone ? " 완료" : ""}
</span> </span>
{inboundDone && (
<svg
className="w-3.5 h-3.5 ml-auto text-green-500"
fill="none"
stroke="currentColor"
strokeWidth={2.5}
viewBox="0 0 24 24"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M4.5 12.75l6 6 9-13.5"
/>
</svg>
)}
</button> </button>
</div> </div>
)} )}
@@ -2767,18 +2708,16 @@ function MaterialQtyInputRow({
}) { }) {
const [open, setOpen] = useState(false); const [open, setOpen] = useState(false);
return ( return (
<div className="flex items-center gap-2"> <div className="flex items-center shrink-0">
<button <button
type="button" type="button"
onClick={() => setOpen(true)} onClick={() => setOpen(true)}
className="flex-1 px-4 py-3 rounded-xl border-2 border-gray-200 text-base font-bold text-left text-gray-900 hover:border-blue-400 active:scale-[0.98] transition-all bg-white" className="px-8 py-4 rounded-xl border-2 border-blue-300 text-xl font-bold text-blue-700 hover:border-blue-500 active:scale-[0.96] transition-all bg-blue-50 min-w-[120px] text-center"
style={{ minHeight: 56 }}
> >
{value || ( {value || (
<span className="text-gray-300 font-normal"> </span> <span className="text-blue-300 font-semibold"></span>
)} )}
</button> </button>
<span className="text-sm text-gray-500 shrink-0">{material.unit}</span>
<MaterialQtyKeypad <MaterialQtyKeypad
open={open} open={open}
onClose={() => setOpen(false)} onClose={() => setOpen(false)}
@@ -2903,50 +2842,44 @@ function MaterialInputSection({ processId }: { processId: string }) {
} }
return ( return (
<div className="space-y-4"> <div className="space-y-3">
{/* BOM 기준 자재 목록 */} {/* BOM 기준 자재 목록 — 컴팩트 */}
<div className="bg-white rounded-2xl border border-gray-100 shadow-sm p-4"> <div className="bg-white rounded-2xl border border-gray-100 shadow-sm p-3">
<h3 className="text-base font-bold text-gray-900 mb-3"> <div className="flex items-center justify-between mb-2">
BOM <h3 className="text-sm font-bold text-gray-900">BOM </h3>
</h3> <span className="text-xs text-gray-400">{bomMaterials.length}</span>
</div>
{bomMaterials.length === 0 ? ( {bomMaterials.length === 0 ? (
<p className="text-sm text-gray-400 py-4 text-center"> <p className="text-sm text-gray-400 py-4 text-center">
BOM BOM
</p> </p>
) : ( ) : (
<div className="space-y-3"> <div>
{bomMaterials.map((m) => ( <div className="divide-y divide-gray-200">
<div {bomMaterials.map((m) => (
key={m.id} <div key={m.id} className="flex items-center gap-2 py-3">
className="p-3 rounded-xl border border-gray-200 bg-gray-50" {/* 자재명(코드) + 소요량 */}
> <div className="flex-1 min-w-0">
<div className="flex items-center justify-between mb-2"> <span className="text-base font-bold text-gray-900">{m.child_item_name}</span>
<div> <span className="text-sm text-gray-400 ml-1">({m.child_item_code})</span>
<p className="text-base font-semibold text-gray-900"> <span className="text-base font-bold text-blue-600 ml-3"> {m.required_qty}</span>
{m.child_item_name}
</p>
<p className="text-sm text-gray-400">{m.child_item_code}</p>
</div>
<div className="text-right">
<p className="text-sm text-gray-500"></p>
<p className="text-lg font-bold text-blue-600">
{m.required_qty} {m.unit}
</p>
</div> </div>
{/* 입력 버튼 + 단위 */}
<MaterialQtyInputRow
material={m}
value={inputValues[m.id] || ""}
onChange={(v) =>
setInputValues((prev) => ({ ...prev, [m.id]: v }))
}
/>
<span className="text-base font-semibold text-gray-500 shrink-0 w-8">{m.unit}</span>
</div> </div>
<MaterialQtyInputRow ))}
material={m} </div>
value={inputValues[m.id] || ""}
onChange={(v) =>
setInputValues((prev) => ({ ...prev, [m.id]: v }))
}
/>
</div>
))}
<button <button
onClick={handleSave} onClick={handleSave}
disabled={saving} disabled={saving}
className="w-full py-4 rounded-xl text-base font-bold text-white active:scale-[0.98] transition-all disabled:opacity-40" className="w-full mt-4 py-4 rounded-xl text-lg font-bold text-white active:scale-[0.98] transition-all disabled:opacity-40"
style={{ style={{
background: "linear-gradient(135deg, #3b82f6, #1d4ed8)", background: "linear-gradient(135deg, #3b82f6, #1d4ed8)",
}} }}
@@ -2077,7 +2077,7 @@ export function WorkOrderList() {
(p) => (p) =>
p.parent_process_id && p.parent_process_id &&
p.accepted_by === currentUserId && p.accepted_by === currentUserId &&
(p.status === "in_progress" || p.status === "completed"), p.status === "in_progress",
)} )}
instructionMap={instructionMap} instructionMap={instructionMap}
onSwitch={(id) => setWorkModalProcessId(id)} onSwitch={(id) => setWorkModalProcessId(id)}