Files
star-erp/resources/js/Components/Warehouse/WarehouseCard.tsx
sky121113 ac6a81b3d2
All checks were successful
Koori-ERP-Deploy-System / deploy-demo (push) Successful in 58s
Koori-ERP-Deploy-System / deploy-production (push) Has been skipped
feat: 倉庫業務屬性、庫存成本追蹤與採購單功能更新
1. 倉庫管理:新增業務類型 (Owned/External/Customer) 與車牌資訊與司機欄位。
2. 庫存管理:實作成本追蹤 (unit_cost, total_value),更新列表與撥補單顯示。
3. 採購單:新增採購日期 (order_date),調整欄位名稱與順序。
4. 前端優化:更新相關 TS Type 定義與 UI 顯示。
2026-01-26 17:27:34 +08:00

197 lines
6.8 KiB
TypeScript

/**
* 倉庫卡片元件
* 顯示單個倉庫的資訊和統計
*/
import { useState } from "react";
import {
Package,
AlertTriangle,
MapPin,
Edit,
Info,
FileText,
} from "lucide-react";
import { Warehouse, WarehouseStats } from "@/types/warehouse";
import { Button } from "@/Components/ui/button";
import { Badge } from "@/Components/ui/badge";
import { Card, CardContent } from "@/Components/ui/card";
import {
Dialog,
DialogContent,
DialogDescription,
DialogHeader,
DialogTitle,
} from "@/Components/ui/dialog";
interface WarehouseCardProps {
warehouse: Warehouse;
stats: WarehouseStats;
hasWarning: boolean;
onViewInventory: (warehouseId: string) => void;
onEdit: (warehouse: Warehouse) => void;
}
const WAREHOUSE_TYPE_LABELS: Record<string, string> = {
standard: "標準倉",
production: "生產倉",
retail: "門市倉",
vending: "販賣機",
transit: "在途倉",
quarantine: "瑕疵倉",
};
export default function WarehouseCard({
warehouse,
stats,
hasWarning,
onViewInventory,
onEdit,
}: WarehouseCardProps) {
const [showInfoDialog, setShowInfoDialog] = useState(false);
return (
<Card
className={`relative overflow-hidden transition-all hover:shadow-lg flex flex-col ${hasWarning
? "border-orange-400 border-2 bg-orange-50/50"
: "border-gray-200"
}`}
>
{/* 警告橫幅 */}
{hasWarning && (
<div className="absolute top-0 left-0 right-0 bg-orange-500 text-white px-4 py-1 flex items-center gap-2 text-sm">
<AlertTriangle className="h-4 w-4" />
<span></span>
</div>
)}
<CardContent className={`p-6 flex flex-col flex-1 ${hasWarning ? "pt-12" : "pt-6"}`}>
{/* 上半部:資訊區域 */}
<div className="flex-1">
{/* 標題區塊 */}
<div className="flex items-start justify-between mb-4">
<div>
<div className="flex items-center gap-2 mb-1">
<h3 className="text-2xl font-bold">{warehouse.name}</h3>
<button
onClick={() => setShowInfoDialog(true)}
className="text-gray-400 hover:text-gray-600 transition-colors"
>
<Info className="h-5 w-5" />
</button>
</div>
<div className="flex gap-2 mt-1">
<Badge variant="outline" className="text-xs font-normal">
{WAREHOUSE_TYPE_LABELS[warehouse.type || 'standard'] || '標準倉'}
</Badge>
{warehouse.type === 'transit' && warehouse.license_plate && (
<Badge variant="secondary" className="text-xs font-normal bg-yellow-100 text-yellow-800 border-yellow-200">
{warehouse.license_plate}
</Badge>
)}
</div>
</div>
</div>
<div className="text-sm text-gray-600 mb-4 line-clamp-2 min-h-[40px]">
{warehouse.description || "無描述"}
</div>
{/* 統計區塊 - 狀態標籤 */}
<div className="space-y-3">
{/* 銷售狀態 */}
<div className="flex items-center justify-between">
<span className="text-sm text-gray-500"></span>
<Badge variant={warehouse.is_sellable ? "default" : "secondary"} className={warehouse.is_sellable ? "bg-green-600" : "bg-gray-400"}>
{warehouse.is_sellable ? "可銷售" : "暫停銷售"}
</Badge>
</div>
{/* 低庫存警告狀態 */}
<div className="flex items-center justify-between p-3 rounded-lg bg-gray-50">
<div className="flex items-center gap-2 text-gray-600">
<AlertTriangle className="h-4 w-4" />
<span className="text-sm"></span>
</div>
<div>
{hasWarning ? (
<Badge className="bg-orange-500 text-white hover:bg-orange-600 border-none px-2 py-0.5">
{stats.lowStockCount}
</Badge>
) : (
<Badge variant="secondary" className="bg-green-100 text-green-700 hover:bg-green-100 border-green-200">
</Badge>
)}
</div>
</div>
{/* 移動倉司機資訊 */}
{warehouse.type === 'transit' && warehouse.driver_name && (
<div className="flex items-center justify-between pt-2 border-t border-gray-100">
<span className="text-sm text-gray-500"></span>
<span className="text-sm font-medium text-gray-900">{warehouse.driver_name}</span>
</div>
)}
</div>
</div>
{/* 下半部:操作按鈕 */}
<div className="mt-5 pt-3 border-t border-gray-200">
<div className="flex gap-2">
<Button
onClick={() => onViewInventory(warehouse.id)}
className="flex-1 button-filled-primary"
size="sm"
>
<Package className="h-4 w-4 mr-2" />
</Button>
<Button
variant="outline"
size="sm"
onClick={() => onEdit(warehouse)}
className="button-outlined-primary"
>
<Edit className="h-4 w-4" />
</Button>
</div>
</div>
</CardContent>
{/* 倉庫資訊對話框 */}
<Dialog open={showInfoDialog} onOpenChange={setShowInfoDialog}>
<DialogContent className="sm:max-w-[425px]">
<DialogHeader>
<DialogTitle>{warehouse.name}</DialogTitle>
<DialogDescription>
{warehouse.code}
</DialogDescription>
</DialogHeader>
<div className="space-y-4 py-4">
<div className="space-y-2">
<div className="flex items-start gap-2 text-gray-600">
<MapPin className="h-5 w-5 mt-0.5 flex-shrink-0" />
<div>
<p className="text-sm text-gray-500 mb-1"></p>
<p className="text-gray-900">{warehouse.address || "-"}</p>
</div>
</div>
</div>
<div className="space-y-2">
<div className="flex items-start gap-2 text-gray-600">
<FileText className="h-5 w-5 mt-0.5 flex-shrink-0" />
<div>
<p className="text-sm text-gray-500 mb-1"></p>
<p className="text-gray-900 whitespace-pre-wrap">{warehouse.description || "-"}</p>
</div>
</div>
</div>
</div>
</DialogContent>
</Dialog>
</Card>
);
}