feat(procurement): 實作採購退回單模組並修復商品選單報錯
All checks were successful
ERP-Deploy-Demo / deploy-demo (push) Successful in 58s

This commit is contained in:
2026-02-25 13:49:02 +08:00
parent deef3baacc
commit c4908533a8
21 changed files with 2409 additions and 1 deletions

View File

@@ -0,0 +1,100 @@
import { useState } from "react";
import { Pencil, Eye, Trash2 } from "lucide-react";
import { Button } from "@/Components/ui/button";
import { Link, useForm } from "@inertiajs/react";
import type { PurchaseReturn } from "@/types/purchase-return";
import { toast } from "sonner";
import { Can } from "@/Components/Permission/Can";
import {
AlertDialog,
AlertDialogAction,
AlertDialogCancel,
AlertDialogContent,
AlertDialogDescription,
AlertDialogFooter,
AlertDialogHeader,
AlertDialogTitle,
} from "@/Components/ui/alert-dialog";
export function PurchaseReturnActions({
purchaseReturn,
}: { purchaseReturn: PurchaseReturn }) {
const [showDeleteDialog, setShowDeleteDialog] = useState(false);
const { delete: destroy, processing } = useForm({});
const handleConfirmDelete = () => {
// @ts-ignore
destroy(route('purchase-returns.destroy', purchaseReturn.id), {
onSuccess: () => {
toast.success("採購退回單已成功刪除");
setShowDeleteDialog(false);
},
onError: (errors: any) => toast.error(errors.error || "刪除過程中發生錯誤"),
});
};
return (
<div className="flex justify-center gap-2">
<Link href={`/purchase-returns/${purchaseReturn.id}`}>
<Button
variant="outline"
size="sm"
className="button-outlined-primary"
title="查看詳情"
>
<Eye className="h-4 w-4" />
</Button>
</Link>
{purchaseReturn.status === 'draft' && (
<Can permission="purchase_returns.edit">
<Link href={`/purchase-returns/${purchaseReturn.id}/edit`}>
<Button
variant="outline"
size="sm"
className="button-outlined-primary"
title="編輯"
>
<Pencil className="h-4 w-4" />
</Button>
</Link>
</Can>
)}
{purchaseReturn.status === 'draft' && (
<Can permission="purchase_returns.delete">
<Button
variant="outline"
size="sm"
className="button-outlined-error"
title="刪除"
onClick={() => setShowDeleteDialog(true)}
disabled={processing}
>
<Trash2 className="h-4 w-4" />
</Button>
<AlertDialog open={showDeleteDialog} onOpenChange={setShowDeleteDialog}>
<AlertDialogContent>
<AlertDialogHeader>
<AlertDialogTitle>退</AlertDialogTitle>
<AlertDialogDescription>
退 {purchaseReturn.code}
</AlertDialogDescription>
</AlertDialogHeader>
<AlertDialogFooter>
<AlertDialogCancel className="button-outlined-primary"></AlertDialogCancel>
<AlertDialogAction
onClick={handleConfirmDelete}
className="button-filled-error"
>
</AlertDialogAction>
</AlertDialogFooter>
</AlertDialogContent>
</AlertDialog>
</Can>
)}
</div>
);
}

View File

@@ -0,0 +1,214 @@
import { Trash2 } from "lucide-react";
import { Button } from "@/Components/ui/button";
import { Input } from "@/Components/ui/input";
import { SearchableSelect } from "@/Components/ui/searchable-select";
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "@/Components/ui/table";
import {
AlertDialog,
AlertDialogAction,
AlertDialogCancel,
AlertDialogContent,
AlertDialogDescription,
AlertDialogFooter,
AlertDialogHeader,
AlertDialogTitle,
AlertDialogTrigger,
} from "@/Components/ui/alert-dialog";
import type { PurchaseReturnItem } from "@/types/purchase-return";
import type { Supplier } from "@/types/purchase-order";
import { formatCurrency } from "@/utils/format";
interface PurchaseReturnItemsTableProps {
items: PurchaseReturnItem[];
vendor?: Supplier;
isReadOnly?: boolean;
isDisabled?: boolean;
onRemoveItem?: (index: number) => void;
onItemChange?: (index: number, field: keyof PurchaseReturnItem, value: string | number) => void;
}
export function PurchaseReturnItemsTable({
items,
vendor,
isReadOnly = false,
isDisabled = false,
onRemoveItem,
onItemChange,
}: PurchaseReturnItemsTableProps) {
return (
<div className={`border rounded-lg overflow-hidden ${isDisabled ? "opacity-50 pointer-events-none grayscale" : ""}`}>
<Table>
<TableHeader>
<TableRow className="bg-gray-50 hover:bg-gray-50">
<TableHead className="w-[30%] text-left">退</TableHead>
<TableHead className="w-[15%] text-left">退</TableHead>
<TableHead className="w-[15%] text-left">退</TableHead>
<TableHead className="w-[15%] text-left">退</TableHead>
<TableHead className="w-[20%] text-left"> / </TableHead>
{!isReadOnly && <TableHead className="w-[5%]"></TableHead>}
</TableRow>
</TableHeader>
<TableBody>
{items.length === 0 ? (
<TableRow>
<TableCell
colSpan={isReadOnly ? 5 : 6}
className="text-center text-gray-400 py-12 italic"
>
{isDisabled ? "請先選擇供應商後才能加入退回商品" : "尚未新增任何退回品項"}
</TableCell>
</TableRow>
) : (
items.map((item, index) => {
return (
<TableRow key={index}>
{/* 商品選擇 */}
<TableCell>
{isReadOnly ? (
<span className="font-medium">{item.product?.name || item.product_name}</span>
) : (
<SearchableSelect
value={String(item.product_id)}
onValueChange={(value) =>
onItemChange?.(index, "product_id", value)
}
disabled={isDisabled}
options={vendor?.commonProducts?.map((p) => ({ label: p.productName, value: String(p.productId) })) || []}
placeholder="選擇退回商品"
searchPlaceholder="搜尋商品..."
emptyText="此供應商無可用商品"
className="w-full"
/>
)}
</TableCell>
{/* 數量 */}
<TableCell className="text-left">
{isReadOnly ? (
<span>{item.quantity_returned}</span>
) : (
<Input
type="number"
min="0.01"
step="any"
value={item.quantity_returned || ""}
onChange={(e) =>
onItemChange?.(index, "quantity_returned", Number(e.target.value))
}
disabled={isDisabled}
className="text-right w-full"
/>
)}
</TableCell>
{/* 單價 */}
<TableCell className="text-left">
{isReadOnly ? (
<span>{formatCurrency(item.unit_price)}</span>
) : (
<Input
type="number"
min="0"
step="any"
value={item.unit_price === 0 ? "" : item.unit_price}
onChange={(e) =>
onItemChange?.(index, "unit_price", Number(e.target.value))
}
disabled={isDisabled}
className="text-right w-full"
/>
)}
</TableCell>
{/* 總退款金額 */}
<TableCell className="text-left">
{isReadOnly ? (
<span className="font-bold text-primary">{formatCurrency(item.total_amount)}</span>
) : (
<div className="space-y-1">
<Input
type="number"
min="0"
step="any"
value={item.total_amount || ""}
onChange={(e) =>
onItemChange?.(index, "total_amount", Number(e.target.value))
}
disabled={isDisabled}
className={`text-right w-full ${item.quantity_returned > 0 && (!item.total_amount || item.total_amount <= 0)
? "border-red-400 bg-red-50 focus-visible:ring-red-500"
: ""
}`}
/>
</div>
)}
</TableCell>
{/* 批號 */}
<TableCell className="text-left">
{isReadOnly ? (
<span className="text-gray-500">{item.batch_number}</span>
) : (
<Input
type="text"
value={item.batch_number || ""}
onChange={(e) =>
onItemChange?.(index, "batch_number", e.target.value)
}
disabled={isDisabled}
className="w-full"
placeholder="輸入批號或備註..."
/>
)}
</TableCell>
{/* 刪除按鈕 */}
{!isReadOnly && onRemoveItem && (
<TableCell className="text-center">
<AlertDialog>
<AlertDialogTrigger asChild>
<Button
variant="outline"
size="sm"
className="button-outlined-error"
title="移除項目"
>
<Trash2 className="h-4 w-4" />
</Button>
</AlertDialogTrigger>
<AlertDialogContent>
<AlertDialogHeader>
<AlertDialogTitle>退</AlertDialogTitle>
<AlertDialogDescription>
退
</AlertDialogDescription>
</AlertDialogHeader>
<AlertDialogFooter>
<AlertDialogCancel></AlertDialogCancel>
<AlertDialogAction
onClick={() => onRemoveItem(index)}
className="bg-red-600 hover:bg-red-700"
>
</AlertDialogAction>
</AlertDialogFooter>
</AlertDialogContent>
</AlertDialog>
</TableCell>
)}
</TableRow>
);
})
)}
</TableBody>
</Table>
</div>
);
}

View File

@@ -0,0 +1,26 @@
import { Badge } from "@/Components/ui/badge";
import { PurchaseReturnStatus, PURCHASE_RETURN_STATUS_CONFIG } from "@/types/purchase-return";
interface PurchaseReturnStatusBadgeProps {
status: PurchaseReturnStatus | string;
className?: string;
}
export default function PurchaseReturnStatusBadge({
status,
className = "",
}: PurchaseReturnStatusBadgeProps) {
const config = PURCHASE_RETURN_STATUS_CONFIG[status as PurchaseReturnStatus] || {
label: status,
color: "bg-gray-100 text-gray-800",
};
return (
<Badge
variant="secondary"
className={`${config.color} whitespace-nowrap min-w-[72px] justify-center ${className}`}
>
{config.label}
</Badge>
);
}

View File

@@ -0,0 +1,213 @@
import { useState, useMemo } from "react";
import { ArrowUpDown, ArrowUp, ArrowDown } from "lucide-react";
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "@/Components/ui/table";
import { PurchaseReturnActions } from "./PurchaseReturnActions";
import PurchaseReturnStatusBadge from "./PurchaseReturnStatusBadge";
import CopyButton from "@/Components/shared/CopyButton";
import type { PurchaseReturn } from "@/types/purchase-return";
import { formatCurrency, formatDateTime } from "@/utils/format";
interface PurchaseReturnTableProps {
purchaseReturns: PurchaseReturn[];
}
type SortField = "code" | "warehouse_name" | "vendor_name" | "return_date" | "total_amount" | "status";
type SortDirection = "asc" | "desc" | null;
export default function PurchaseReturnTable({
purchaseReturns,
}: PurchaseReturnTableProps) {
const [sortField, setSortField] = useState<SortField | null>(null);
const [sortDirection, setSortDirection] = useState<SortDirection>(null);
// 處理排序
const handleSort = (field: SortField) => {
if (sortField === field) {
if (sortDirection === "asc") {
setSortDirection("desc");
} else if (sortDirection === "desc") {
setSortDirection(null);
setSortField(null);
} else {
setSortDirection("asc");
}
} else {
setSortField(field);
setSortDirection("asc");
}
};
// 排序後的退回單列表
const sortedReturns = useMemo(() => {
if (!sortField || !sortDirection || !purchaseReturns) {
return purchaseReturns || [];
}
return [...purchaseReturns].sort((a, b) => {
let aValue: string | number;
let bValue: string | number;
switch (sortField) {
case "code":
aValue = a.code;
bValue = b.code;
break;
case "warehouse_name":
aValue = a.warehouse_name || "";
bValue = b.warehouse_name || "";
break;
case "vendor_name":
aValue = a.vendor?.name || "";
bValue = b.vendor?.name || "";
break;
case "return_date":
aValue = a.return_date;
bValue = b.return_date;
break;
case "total_amount":
aValue = a.total_amount;
bValue = b.total_amount;
break;
case "status":
aValue = a.status;
bValue = b.status;
break;
default:
return 0;
}
if (typeof aValue === "string" && typeof bValue === "string") {
return sortDirection === "asc"
? aValue.localeCompare(bValue, "zh-TW")
: bValue.localeCompare(aValue, "zh-TW");
} else {
return sortDirection === "asc"
? (aValue as number) - (bValue as number)
: (bValue as number) - (aValue as number);
}
});
}, [purchaseReturns, sortField, sortDirection]);
const SortIcon = ({ field }: { field: SortField }) => {
if (sortField !== field) {
return <ArrowUpDown className="h-4 w-4 text-muted-foreground ml-1" />;
}
if (sortDirection === "asc") {
return <ArrowUp className="h-4 w-4 text-primary ml-1" />;
}
if (sortDirection === "desc") {
return <ArrowDown className="h-4 w-4 text-primary ml-1" />;
}
return <ArrowUpDown className="h-4 w-4 text-muted-foreground ml-1" />;
};
return (
<div className="bg-white rounded-lg border shadow-sm overflow-hidden">
<div className="overflow-x-auto">
<Table>
<TableHeader className="bg-gray-50/50">
<TableRow>
<TableHead className="w-[50px] text-center">#</TableHead>
<TableHead className="w-[180px]">
<button
onClick={() => handleSort("code")}
className="flex items-center gap-1 hover:text-foreground transition-colors"
>
退
<SortIcon field="code" />
</button>
</TableHead>
<TableHead className="w-[180px]">
<button
onClick={() => handleSort("vendor_name")}
className="flex items-center gap-1 hover:text-foreground transition-colors"
>
<SortIcon field="vendor_name" />
</button>
</TableHead>
<TableHead className="w-[150px]">
<button
onClick={() => handleSort("return_date")}
className="flex items-center gap-1 hover:text-foreground transition-colors"
>
退
<SortIcon field="return_date" />
</button>
</TableHead>
<TableHead className="w-[140px] text-right">
<button
onClick={() => handleSort("total_amount")}
className="flex items-center justify-end gap-1 w-full hover:text-foreground transition-colors"
>
()
<SortIcon field="total_amount" />
</button>
</TableHead>
<TableHead className="w-[120px] text-center">
<button
onClick={() => handleSort("status")}
className="flex items-center justify-center gap-1 w-full hover:text-foreground transition-colors"
>
<SortIcon field="status" />
</button>
</TableHead>
<TableHead className="text-center"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{!sortedReturns || sortedReturns.length === 0 ? (
<TableRow>
<TableCell colSpan={7} className="text-center text-muted-foreground py-12">
退
</TableCell>
</TableRow>
) : (
sortedReturns.map((item, index) => (
<TableRow key={item.id}>
<TableCell className="text-gray-500 font-medium text-center">
{index + 1}
</TableCell>
<TableCell>
<div className="flex items-center gap-1.5">
<span className="font-mono text-sm font-medium">{item.code}</span>
<CopyButton text={item.code} label="複製單號" />
</div>
</TableCell>
<TableCell>
<div className="space-y-0.5">
<span className="text-sm text-gray-700">{item.vendor?.name}</span>
<div className="text-xs text-gray-500">{item.user?.name}</div>
</div>
</TableCell>
<TableCell>
<span className="text-sm text-gray-500">{item.return_date}</span>
</TableCell>
<TableCell className="text-right">
<span className="font-semibold text-gray-900">{formatCurrency(item.total_amount)}</span>
</TableCell>
<TableCell className="text-center">
<PurchaseReturnStatusBadge status={item.status} />
</TableCell>
<TableCell className="text-center">
<PurchaseReturnActions
purchaseReturn={item}
/>
</TableCell>
</TableRow>
))
)}
</TableBody>
</Table>
</div>
</div>
);
}

View File

@@ -26,7 +26,8 @@ import {
ArrowLeftRight,
TrendingUp,
FileUp,
Store
Store,
RotateCcw
} from "lucide-react";
import { toast, Toaster } from "sonner";
import { useState, useEffect, useMemo, useRef } from "react";
@@ -168,6 +169,13 @@ export default function AuthenticatedLayout({
route: "/goods-receipts",
permission: "goods_receipts.view",
},
{
id: "purchase-return-list",
label: "採購退回單",
icon: <RotateCcw className="h-4 w-4" />,
route: "/purchase-returns",
permission: "purchase_returns.view",
},
{
id: "delivery-note-list",
label: "出貨單管理 (功能製作中)",

View File

@@ -0,0 +1,281 @@
import { ArrowLeft, Plus, Info, Package } from "lucide-react";
import { Button } from "@/Components/ui/button";
import { Input } from "@/Components/ui/input";
import { Textarea } from "@/Components/ui/textarea";
import { Alert, AlertDescription } from "@/Components/ui/alert";
import { SearchableSelect } from "@/Components/ui/searchable-select";
import AuthenticatedLayout from "@/Layouts/AuthenticatedLayout";
import { Head, Link, router, usePage } from "@inertiajs/react";
import { PurchaseReturnItemsTable } from "@/Components/PurchaseReturn/PurchaseReturnItemsTable";
import type { PurchaseReturn } from "@/types/purchase-return";
import type { Supplier } from "@/types/purchase-order";
import type { Warehouse } from "@/types/requester";
import { usePurchaseReturnForm } from "@/hooks/usePurchaseReturnForm";
import { formatCurrency } from "@/utils/format";
import { toast } from "sonner";
import { getCreateBreadcrumbs } from "@/utils/breadcrumb";
interface Props {
vendors: Supplier[];
warehouses: Warehouse[];
}
export default function CreatePurchaseReturn({
vendors,
warehouses,
}: Props) {
const { auth } = usePage<any>().props;
const permissions = auth.user?.permissions || [];
const isSuperAdmin = auth.user?.roles?.some((r: any) => r.name === 'super-admin');
const canCreate = isSuperAdmin || permissions.includes('purchase_returns.create');
const {
vendorId,
warehouseId,
returnDate,
items,
remarks,
selectedVendor,
setVendorId,
setWarehouseId,
setReturnDate,
setRemarks,
addItem,
removeItem,
updateItem,
} = usePurchaseReturnForm({ suppliers: vendors });
const totalAmount = items.reduce((sum, item) => sum + (Number(item.total_amount) || 0), 0);
const handleSave = () => {
if (!warehouseId) {
toast.error("請選擇退出的倉庫");
return;
}
if (!vendorId) {
toast.error("請選擇供應商");
return;
}
if (!returnDate) {
toast.error("請選擇退回日期");
return;
}
if (items.length === 0) {
toast.error("請至少新增一項退回商品");
return;
}
const itemsWithQuantity = items.filter(item => item.quantity_returned > 0);
if (itemsWithQuantity.length === 0) {
toast.error("請填寫有效的退回數量(必須大於 0");
return;
}
const itemsWithoutPrice = itemsWithQuantity.filter(item => !item.unit_price || item.unit_price < 0);
if (itemsWithoutPrice.length > 0) {
toast.error("單價不能為負數");
return;
}
const validItems = items.filter(i => i.product_id && i.quantity_returned > 0);
if (validItems.length === 0) {
toast.error("請確保所有商品都有正確選擇並填寫數量");
return;
}
const data = {
vendor_id: vendorId,
warehouse_id: warehouseId,
return_date: returnDate,
remarks: remarks,
items: validItems.map(item => ({
product_id: item.product_id,
quantity_returned: item.quantity_returned,
unit_price: item.unit_price,
total_amount: item.total_amount,
batch_number: item.batch_number || null,
})),
};
router.post("/purchase-returns", data, {
onSuccess: () => {
// Controller 會有 Flash Message 透過 Middleware 傳給 Toast
},
onError: (errors) => {
if (errors.items) {
toast.error("商品資料有誤,請檢查數量和單價是否正確填寫");
} else if (errors.error) {
toast.error(errors.error);
} else {
toast.error("建立失敗,請檢查輸入內容");
}
console.error(errors);
}
});
};
const hasVendor = !!vendorId;
return (
<AuthenticatedLayout breadcrumbs={getCreateBreadcrumbs("purchaseReturns")}>
<Head title="建立採購退回單" />
<div className="container mx-auto p-6 max-w-7xl">
{/* Header */}
<div className="mb-6">
<Link href="/purchase-returns">
<Button
variant="outline"
className="gap-2 button-outlined-primary mb-4"
>
<ArrowLeft className="h-4 w-4" />
</Button>
</Link>
<div className="mb-4">
<h1 className="text-2xl font-bold text-grey-0 flex items-center gap-2">
<Package className="h-6 w-6 text-primary-main" />
退
</h1>
<p className="text-gray-500 mt-1">
退
</p>
</div>
</div>
<div className="space-y-6">
{/* 步驟一:基本資訊 */}
<div className="bg-white rounded-lg border shadow-sm overflow-hidden">
<div className="p-6 bg-gray-50/50 border-b flex items-center gap-3">
<div className="w-8 h-8 rounded-full bg-primary text-white flex items-center justify-center font-bold">1</div>
<h2 className="text-lg font-bold"></h2>
</div>
<div className="p-6 space-y-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-8">
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">退 <span className="text-red-500">*</span></label>
<SearchableSelect
value={String(warehouseId)}
onValueChange={setWarehouseId}
options={warehouses.map((w) => ({ label: w.name, value: String(w.id) }))}
placeholder="請選擇倉庫"
searchPlaceholder="搜尋倉庫..."
/>
<p className="text-xs text-gray-500">退</p>
</div>
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">退 <span className="text-red-500">*</span></label>
<SearchableSelect
value={String(vendorId)}
onValueChange={setVendorId}
options={vendors.map((s) => ({ label: s.name, value: String(s.id) }))}
placeholder="選擇退回廠商"
searchPlaceholder="搜尋廠商..."
/>
</div>
</div>
<div className="grid grid-cols-1 md:grid-cols-2 gap-8">
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">
退 <span className="text-red-500">*</span>
</label>
<Input
type="date"
value={returnDate || ""}
onChange={(e) => setReturnDate(e.target.value)}
className="block w-full"
/>
</div>
</div>
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">退/</label>
<Textarea
value={remarks || ""}
onChange={(e) => setRemarks(e.target.value)}
placeholder="描述退回的原因 (如:商品瑕疵,與廠商協商同意退貨...)"
className="min-h-[100px]"
/>
</div>
</div>
</div>
{/* 步驟二:退回品項明細 */}
<div className={`bg-white rounded-lg border shadow-sm overflow-hidden transition-all duration-300 ${!hasVendor ? 'opacity-60 saturate-50' : ''}`}>
<div className="p-6 bg-gray-50/50 border-b flex items-center justify-between">
<div className="flex items-center gap-3">
<div className="w-8 h-8 rounded-full bg-primary text-white flex items-center justify-center font-bold">2</div>
<h2 className="text-lg font-bold">退</h2>
</div>
<Button
onClick={addItem}
disabled={!hasVendor}
className="button-filled-primary h-10 gap-2"
>
<Plus className="h-4 w-4" /> 退
</Button>
</div>
<div className="p-6">
{!hasVendor && (
<Alert className="mb-6 bg-amber-50 border-amber-200 text-amber-800">
<Info className="h-4 w-4 text-amber-600" />
<AlertDescription>
退退
</AlertDescription>
</Alert>
)}
<PurchaseReturnItemsTable
items={items}
vendor={selectedVendor}
isReadOnly={false}
isDisabled={!hasVendor}
onRemoveItem={removeItem}
onItemChange={updateItem}
/>
{hasVendor && items.length > 0 && (
<div className="mt-6 flex justify-end">
<div className="w-full max-w-sm bg-primary/5 px-6 py-4 rounded-xl border border-primary/10 flex flex-col gap-3">
<div className="flex justify-between items-center w-full">
<span className="text-sm text-gray-500 font-medium">退</span>
<span className="text-2xl font-black text-primary">
{formatCurrency(totalAmount)}
</span>
</div>
</div>
</div>
)}
</div>
</div>
</div>
{/* 底部按鈕 */}
<div className="flex items-center justify-end gap-4 py-6 border-t border-gray-100 mt-6">
<Link href="/purchase-returns">
<Button variant="ghost" className="h-11 px-6 text-gray-500 hover:text-gray-700">
</Button>
</Link>
<Button
size="xl"
className="bg-primary hover:bg-primary/90 text-white shadow-primary/20"
onClick={handleSave}
disabled={!canCreate}
title={!canCreate ? "您沒有建立退回單的權限" : ""}
>
稿
</Button>
</div>
</div>
</AuthenticatedLayout>
);
}

View File

@@ -0,0 +1,294 @@
import { ArrowLeft, Plus, Info, Package } from "lucide-react";
import { Button } from "@/Components/ui/button";
import { Input } from "@/Components/ui/input";
import { Textarea } from "@/Components/ui/textarea";
import { Alert, AlertDescription } from "@/Components/ui/alert";
import { SearchableSelect } from "@/Components/ui/searchable-select";
import AuthenticatedLayout from "@/Layouts/AuthenticatedLayout";
import { Head, Link, router, usePage } from "@inertiajs/react";
import { PurchaseReturnItemsTable } from "@/Components/PurchaseReturn/PurchaseReturnItemsTable";
import type { PurchaseReturn } from "@/types/purchase-return";
import type { Supplier } from "@/types/purchase-order";
import type { Warehouse } from "@/types/requester";
import { usePurchaseReturnForm } from "@/hooks/usePurchaseReturnForm";
import { formatCurrency } from "@/utils/format";
import { toast } from "sonner";
import { getEditBreadcrumbs } from "@/utils/breadcrumb";
interface Props {
purchaseReturn: PurchaseReturn;
vendors: Supplier[];
warehouses: Warehouse[];
}
export default function EditPurchaseReturn({
purchaseReturn,
vendors,
warehouses,
}: Props) {
const { auth } = usePage<any>().props;
const permissions = auth.user?.permissions || [];
const isSuperAdmin = auth.user?.roles?.some((r: any) => r.name === 'super-admin');
const canEdit = isSuperAdmin || permissions.includes('purchase_returns.edit');
const {
vendorId,
warehouseId,
returnDate,
items,
remarks,
selectedVendor,
setVendorId,
setWarehouseId,
setReturnDate,
setRemarks,
addItem,
removeItem,
updateItem,
} = usePurchaseReturnForm({ purchaseReturn, suppliers: vendors });
const totalAmount = items.reduce((sum, item) => sum + (Number(item.total_amount) || 0), 0);
const isDraft = purchaseReturn.status === 'draft';
const handleSave = () => {
if (!warehouseId) {
toast.error("請選擇退出的倉庫");
return;
}
if (!vendorId) {
toast.error("請選擇供應商");
return;
}
if (!returnDate) {
toast.error("請選擇退回日期");
return;
}
if (items.length === 0) {
toast.error("請至少新增一項退回商品");
return;
}
const itemsWithQuantity = items.filter(item => item.quantity_returned > 0);
if (itemsWithQuantity.length === 0) {
toast.error("請填寫有效的退回數量(必須大於 0");
return;
}
const itemsWithoutPrice = itemsWithQuantity.filter(item => !item.unit_price || item.unit_price < 0);
if (itemsWithoutPrice.length > 0) {
toast.error("單價不能為負數");
return;
}
const validItems = items.filter(i => i.product_id && i.quantity_returned > 0);
if (validItems.length === 0) {
toast.error("請確保所有商品都有正確選擇並填寫數量");
return;
}
const data = {
vendor_id: vendorId,
warehouse_id: warehouseId,
return_date: returnDate,
remarks: remarks,
items: validItems.map(item => ({
product_id: item.product_id,
quantity_returned: item.quantity_returned,
unit_price: item.unit_price,
total_amount: item.total_amount,
batch_number: item.batch_number || null,
})),
};
router.put(`/purchase-returns/${purchaseReturn.id}`, data, {
onSuccess: () => {
// Controller 會有 Flash Message
},
onError: (errors) => {
if (errors.items) {
toast.error("商品資料有誤,請檢查數量和單價是否正確填寫");
} else if (errors.error) {
toast.error(errors.error);
} else {
toast.error("更新失敗,請檢查輸入內容");
}
console.error(errors);
}
});
};
const hasVendor = !!vendorId;
return (
<AuthenticatedLayout breadcrumbs={getEditBreadcrumbs("purchaseReturns")}>
<Head title="編輯採購退回單" />
<div className="container mx-auto p-6 max-w-7xl">
{/* Header */}
<div className="mb-6">
<Link href={`/purchase-returns/${purchaseReturn.id}`}>
<Button
variant="outline"
className="gap-2 button-outlined-primary mb-4"
>
<ArrowLeft className="h-4 w-4" />
退
</Button>
</Link>
<div className="mb-4">
<h1 className="text-2xl font-bold text-grey-0 flex items-center gap-2">
<Package className="h-6 w-6 text-primary-main" />
退 {purchaseReturn.code}
</h1>
<p className="text-gray-500 mt-1">
退
</p>
</div>
</div>
{!isDraft && (
<Alert className="mb-6 bg-amber-50 border-amber-200 text-amber-800">
<Info className="h-4 w-4 text-amber-600" />
<AlertDescription>
退{purchaseReturn.status}稿
</AlertDescription>
</Alert>
)}
<div className={`space-y-6 ${!isDraft ? 'opacity-70 pointer-events-none grayscale-0' : ''}`}>
{/* 步驟一:基本資訊 */}
<div className="bg-white rounded-lg border shadow-sm overflow-hidden">
<div className="p-6 bg-gray-50/50 border-b flex items-center gap-3">
<div className="w-8 h-8 rounded-full bg-primary text-white flex items-center justify-center font-bold">1</div>
<h2 className="text-lg font-bold"></h2>
</div>
<div className="p-6 space-y-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-8">
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">退 <span className="text-red-500">*</span></label>
<SearchableSelect
value={String(warehouseId)}
onValueChange={setWarehouseId}
options={warehouses.map((w) => ({ label: w.name, value: String(w.id) }))}
placeholder="請選擇倉庫"
searchPlaceholder="搜尋倉庫..."
/>
</div>
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">退 <span className="text-red-500">*</span></label>
<SearchableSelect
value={String(vendorId)}
onValueChange={setVendorId}
options={vendors.map((s) => ({ label: s.name, value: String(s.id) }))}
placeholder="選擇退回廠商"
searchPlaceholder="搜尋廠商..."
/>
</div>
</div>
<div className="grid grid-cols-1 md:grid-cols-2 gap-8">
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">
退 <span className="text-red-500">*</span>
</label>
<Input
type="date"
value={returnDate || ""}
onChange={(e) => setReturnDate(e.target.value)}
className="block w-full"
/>
</div>
</div>
<div className="space-y-3">
<label className="text-sm font-bold text-gray-700">退/</label>
<Textarea
value={remarks || ""}
onChange={(e) => setRemarks(e.target.value)}
placeholder="描述退回的原因 (如:商品瑕疵,與廠商協商同意退貨...)"
className="min-h-[100px]"
/>
</div>
</div>
</div>
{/* 步驟二:退回品項明細 */}
<div className={`bg-white rounded-lg border shadow-sm overflow-hidden transition-all duration-300 ${!hasVendor ? 'opacity-60 saturate-50' : ''}`}>
<div className="p-6 bg-gray-50/50 border-b flex items-center justify-between">
<div className="flex items-center gap-3">
<div className="w-8 h-8 rounded-full bg-primary text-white flex items-center justify-center font-bold">2</div>
<h2 className="text-lg font-bold">退</h2>
</div>
<Button
onClick={addItem}
disabled={!hasVendor || !isDraft}
className="button-filled-primary h-10 gap-2"
>
<Plus className="h-4 w-4" /> 退
</Button>
</div>
<div className="p-6">
{!hasVendor && (
<Alert className="mb-6 bg-amber-50 border-amber-200 text-amber-800">
<Info className="h-4 w-4 text-amber-600" />
<AlertDescription>
退退
</AlertDescription>
</Alert>
)}
<PurchaseReturnItemsTable
items={items}
vendor={selectedVendor}
isReadOnly={!isDraft}
isDisabled={!hasVendor || !isDraft}
onRemoveItem={removeItem}
onItemChange={updateItem}
/>
{hasVendor && items.length > 0 && (
<div className="mt-6 flex justify-end">
<div className="w-full max-w-sm bg-primary/5 px-6 py-4 rounded-xl border border-primary/10 flex flex-col gap-3">
<div className="flex justify-between items-center w-full">
<span className="text-sm text-gray-500 font-medium">退</span>
<span className="text-2xl font-black text-primary">
{formatCurrency(totalAmount)}
</span>
</div>
</div>
</div>
)}
</div>
</div>
</div>
{/* 底部按鈕 */}
<div className="flex items-center justify-end gap-4 py-6 border-t border-gray-100 mt-6">
<Link href={`/purchase-returns/${purchaseReturn.id}`}>
<Button variant="ghost" className="h-11 px-6 text-gray-500 hover:text-gray-700">
</Button>
</Link>
{isDraft && (
<Button
size="xl"
className="bg-primary hover:bg-primary/90 text-white shadow-primary/20"
onClick={handleSave}
disabled={!canEdit}
title={!canEdit ? "您沒有編輯退回單的權限" : ""}
>
</Button>
)}
</div>
</div>
</AuthenticatedLayout>
);
}

View File

@@ -0,0 +1,172 @@
/**
* 採購退回單管理主頁面
*/
import { useState, useEffect } from "react";
import { Plus, Package, Search, RotateCcw } from 'lucide-react';
import { Button } from "@/Components/ui/button";
import AuthenticatedLayout from "@/Layouts/AuthenticatedLayout";
import { Head, router } from "@inertiajs/react";
import PurchaseReturnTable from "@/Components/PurchaseReturn/PurchaseReturnTable";
import type { PurchaseReturn } from "@/types/purchase-return";
import Pagination from "@/Components/shared/Pagination";
import { getBreadcrumbs } from "@/utils/breadcrumb";
import { Can } from "@/Components/Permission/Can";
import { Input } from "@/Components/ui/input";
import { Label } from "@/Components/ui/label";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/Components/ui/select";
import { PURCHASE_RETURN_STATUS_CONFIG } from "@/types/purchase-return";
interface Props {
purchaseReturns: {
data: PurchaseReturn[];
links: any[];
total: number;
from: number;
to: number;
};
filters: {
search?: string;
status?: string;
};
}
export default function PurchaseReturnIndex({ purchaseReturns, filters }: Props) {
const [search, setSearch] = useState(filters.search || "");
const [status, setStatus] = useState<string>(filters.status || "all");
// 同步 URL 參數
useEffect(() => {
setSearch(filters.search || "");
setStatus(filters.status || "all");
}, [filters]);
const handleFilter = () => {
router.get(
route('purchase-returns.index'),
{
search,
status: status === 'all' ? undefined : status,
},
{ preserveState: true, replace: true }
);
};
const handleReset = () => {
setSearch("");
setStatus("all");
router.get(route('purchase-returns.index'));
};
const handleNavigateToCreate = () => {
router.get(route('purchase-returns.create'));
};
const statusOptions = Object.entries(PURCHASE_RETURN_STATUS_CONFIG).map(([key, config]) => ({
value: key,
label: config.label
}));
return (
// TODO: 加入 Breadcrumbs 對應到 purchaseReturns (如果目前 getBreadcrumbs 尚未支援,可先傳入自訂或加入 utils 支援)
<AuthenticatedLayout breadcrumbs={getBreadcrumbs("purchaseReturns")}>
<Head title="採購退回單管理" />
<div className="container mx-auto p-6 max-w-7xl">
<div className="flex items-center justify-between mb-6">
<div>
<h1 className="text-2xl font-bold text-grey-0 flex items-center gap-2">
<Package className="h-6 w-6 text-primary-main" />
退
</h1>
<p className="text-gray-500 mt-1">
退
</p>
</div>
<div className="flex gap-2">
<Can permission="purchase_returns.create">
<Button
onClick={handleNavigateToCreate}
className="gap-2 button-filled-primary"
>
<Plus className="h-4 w-4" />
退
</Button>
</Can>
</div>
</div>
{/* 篩選區塊 */}
<div className="bg-white rounded-xl shadow-sm border border-grey-4 p-5 mb-6">
<div className="grid grid-cols-1 md:grid-cols-12 gap-4 items-end">
{/* Search */}
<div className="md:col-span-5 space-y-1">
<Label className="text-xs font-medium text-grey-1"></Label>
<div className="relative">
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 text-gray-400 h-4 w-4" />
<Input
placeholder="搜尋退回單號、廠商名稱..."
value={search}
onChange={(e) => setSearch(e.target.value)}
className="pl-10 h-9 block"
onKeyDown={(e) => e.key === 'Enter' && handleFilter()}
/>
</div>
</div>
{/* Status */}
<div className="md:col-span-3 space-y-1">
<Label className="text-xs font-medium text-grey-1"></Label>
<Select value={status} onValueChange={setStatus}>
<SelectTrigger className="h-9">
<SelectValue placeholder="選擇狀態" />
</SelectTrigger>
<SelectContent>
<SelectItem value="all"></SelectItem>
{statusOptions.map((option) => (
<SelectItem key={option.value} value={option.value}>
{option.label}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
{/* Actions */}
<div className="md:col-span-4 flex items-center justify-end gap-2">
<Button
variant="outline"
onClick={handleReset}
className="flex-1 flex items-center justify-center gap-2 button-outlined-primary h-9"
>
<RotateCcw className="h-4 w-4" />
</Button>
<Button
onClick={handleFilter}
className="flex-1 flex items-center justify-center gap-2 button-filled-primary h-9"
>
<Search className="h-4 w-4" />
</Button>
</div>
</div>
</div>
<PurchaseReturnTable
purchaseReturns={purchaseReturns.data}
/>
{/* 分頁元件 */}
<div className="mt-4 flex flex-col sm:flex-row items-center justify-between sm:justify-end gap-4 w-full">
<Pagination links={purchaseReturns.links} />
</div>
</div>
</AuthenticatedLayout>
);
}

View File

@@ -0,0 +1,218 @@
import { ArrowLeft, Package, CheckCircle, XCircle, FileEdit } from "lucide-react";
import { Button } from "@/Components/ui/button";
import AuthenticatedLayout from "@/Layouts/AuthenticatedLayout";
import { Head, Link, useForm, usePage, router } from "@inertiajs/react";
import CopyButton from "@/Components/shared/CopyButton";
import { PurchaseReturnItemsTable } from "@/Components/PurchaseReturn/PurchaseReturnItemsTable";
import type { PurchaseReturn } from "@/types/purchase-return";
import { formatCurrency, formatDateTime } from "@/utils/format";
import { getShowBreadcrumbs } from "@/utils/breadcrumb";
import { toast } from "sonner";
import { PageProps } from "@/types/global";
interface Props {
purchaseReturn: PurchaseReturn;
}
export default function ViewPurchaseReturnPage({ purchaseReturn }: Props) {
const isDraft = purchaseReturn.status === 'draft';
return (
<AuthenticatedLayout breadcrumbs={getShowBreadcrumbs("purchaseReturns", `詳情 (${purchaseReturn.code})`)}>
<Head title={`採購退回單詳情 - ${purchaseReturn.code}`} />
<div className="container mx-auto p-6 max-w-7xl">
{/* 返回按鈕 */}
<div className="mb-6">
<Link href="/purchase-returns">
<Button
variant="outline"
className="gap-2 button-outlined-primary"
>
<ArrowLeft className="h-4 w-4" />
退
</Button>
</Link>
</div>
{/* 頁面標題與操作 */}
<div className="flex flex-col md:flex-row items-center justify-between mb-6 gap-4">
<div>
<h1 className="text-2xl font-bold text-grey-0 flex items-center gap-2">
<Package className="h-6 w-6 text-primary-main" />
退
</h1>
<div className="flex items-center gap-2 mt-1">
<PurchaseReturnStatusBadge status={purchaseReturn.status} />
<span className="text-gray-500 text-sm">{purchaseReturn.code}</span>
</div>
</div>
<div className="flex items-center gap-3">
<PurchaseReturnActions purchaseReturn={purchaseReturn} />
</div>
</div>
<div className="space-y-8">
{/* 基本資訊卡片 */}
<div className="bg-white rounded-lg border shadow-sm p-6">
<div className="flex justify-between items-center mb-6">
<h2 className="text-lg font-bold text-gray-900"></h2>
{isDraft && (
<Link href={`/purchase-returns/${purchaseReturn.id}/edit`}>
<Button variant="outline" size="sm" className="button-outlined-primary">
<FileEdit className="h-4 w-4 mr-1" />
</Button>
</Link>
)}
</div>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-x-8 gap-y-6">
<div>
<span className="text-sm text-gray-500 block mb-1">退</span>
<div className="flex items-center gap-1.5">
<span className="font-mono font-medium text-gray-900">{purchaseReturn.code}</span>
<CopyButton text={purchaseReturn.code} label="複製單號" />
</div>
</div>
<div>
<span className="text-sm text-gray-500 block mb-1"></span>
<span className="font-medium text-gray-900">{purchaseReturn.vendor?.name}</span>
</div>
<div>
<span className="text-sm text-gray-500 block mb-1">退</span>
<span className="font-medium text-gray-900">
{purchaseReturn.warehouse_name}
</span>
</div>
<div>
<span className="text-sm text-gray-500 block mb-1"></span>
<span className="font-medium text-gray-900">
{purchaseReturn.user?.name}
</span>
</div>
<div>
<span className="text-sm text-gray-500 block mb-1"></span>
<span className="font-medium text-gray-900">{formatDateTime(purchaseReturn.created_at)}</span>
</div>
<div>
<span className="text-sm text-gray-500 block mb-1">退</span>
<span className="font-medium text-gray-900">{purchaseReturn.return_date || "-"}</span>
</div>
</div>
{purchaseReturn.remarks && (
<div className="mt-8 pt-6 border-t border-gray-100">
<span className="text-sm text-gray-500 block mb-2">退/</span>
<p className="text-sm text-gray-700 bg-gray-50 p-4 rounded-lg leading-relaxed whitespace-pre-wrap">
{purchaseReturn.remarks}
</p>
</div>
)}
</div>
{/* 退回項目卡片 */}
<div className="bg-white rounded-lg border shadow-sm overflow-hidden">
<div className="p-6 border-b border-gray-100">
<h2 className="text-lg font-bold text-gray-900">退</h2>
</div>
<div className="p-0 sm:p-6">
<PurchaseReturnItemsTable
items={purchaseReturn.items || []}
isReadOnly={true}
/>
<div className="mt-6 flex justify-end mr-6 sm:mr-0">
<div className="w-full max-w-sm bg-primary/5 px-6 py-4 rounded-xl border border-primary/10 flex flex-col gap-3">
<div className="flex justify-between items-center w-full">
<span className="text-sm text-gray-500 font-medium">退</span>
<span className="text-2xl font-black text-primary">
{formatCurrency(purchaseReturn.total_amount)}
</span>
</div>
{purchaseReturn.tax_amount > 0 && (
<div className="flex justify-between items-center w-full">
<span className="text-sm text-gray-500 font-medium"></span>
<span className="text-lg text-gray-700">
{formatCurrency(purchaseReturn.tax_amount)}
</span>
</div>
)}
</div>
</div>
</div>
</div>
</div>
</div>
</AuthenticatedLayout>
);
}
function PurchaseReturnStatusBadge({ status }: { status: string }) {
switch (status) {
case 'draft':
return <span className="px-2.5 py-1 bg-yellow-100 text-yellow-800 text-xs font-semibold rounded-full border border-yellow-200">稿</span>;
case 'completed':
return <span className="px-2.5 py-1 bg-green-100 text-green-800 text-xs font-semibold rounded-full border border-green-200"></span>;
case 'cancelled':
return <span className="px-2.5 py-1 bg-gray-100 text-gray-800 text-xs font-semibold rounded-full border border-gray-200"></span>;
default:
return <span className="px-2.5 py-1 bg-gray-100 text-gray-800 text-xs font-semibold rounded-full border border-gray-200">{status}</span>;
}
}
function PurchaseReturnActions({ purchaseReturn }: { purchaseReturn: PurchaseReturn }) {
const { auth } = usePage<PageProps>().props;
const permissions = auth.user?.permissions || [];
const { processing } = useForm({});
const handleAction = (routeUri: string, method: 'post' | 'put' | 'patch' | 'delete', confirmMessage?: string) => {
if (confirmMessage && !confirm(confirmMessage)) return;
router[method](routeUri, {}, {
onSuccess: () => {
// UI feedback handled by Flash Message toast via Middleware
},
onError: (errors: any) => {
console.error("Action Error:", errors);
toast.error(errors.error || "操作失敗");
}
});
};
const isSuperAdmin = auth.user?.roles?.some((r: any) => r.name === 'super-admin');
const canApprove = isSuperAdmin || permissions.includes('purchase_returns.approve');
const canCancel = isSuperAdmin || permissions.includes('purchase_returns.cancel');
const canDelete = isSuperAdmin || permissions.includes('purchase_returns.delete');
return (
<div className="flex items-center gap-2">
{purchaseReturn.status === 'draft' && canDelete && (
<Button
onClick={() => handleAction(`/purchase-returns/${purchaseReturn.id}`, 'delete', '確定要刪除這筆草稿嗎?此操作無法復原。')}
disabled={processing}
variant="outline"
className="button-outlined-error border-red-600 text-red-600 hover:bg-red-50"
>
<XCircle className="h-4 w-4 mr-1" /> 稿
</Button>
)}
{purchaseReturn.status === 'draft' && canApprove && (
<Button
onClick={() => handleAction(`/purchase-returns/${purchaseReturn.id}/submit`, 'post', '確定要完成此退貨?此步驟會立即扣減實際倉庫庫存,且無法復原。')}
disabled={processing}
className="button-filled-success shadow-green-600/20"
>
<CheckCircle className="h-4 w-4 mr-1" /> 退 ()
</Button>
)}
{purchaseReturn.status === 'completed' && canCancel && (
<Button
onClick={() => handleAction(`/purchase-returns/${purchaseReturn.id}/cancel`, 'post', '確定要取消此筆退貨?庫存不會自動回補。')}
disabled={processing}
variant="outline"
className="button-outlined-warning"
>
<XCircle className="h-4 w-4 mr-1" />
</Button>
)}
</div>
);
}

View File

@@ -0,0 +1,123 @@
import { useState, useEffect } from "react";
import type { PurchaseReturn, PurchaseReturnItem, PurchaseReturnStatus } from "@/types/purchase-return";
import type { Supplier } from "@/types/purchase-order";
interface UsePurchaseReturnFormProps {
purchaseReturn?: PurchaseReturn;
suppliers: Supplier[];
}
export function usePurchaseReturnForm({ purchaseReturn, suppliers }: UsePurchaseReturnFormProps) {
const [vendorId, setVendorId] = useState<string | number>(purchaseReturn?.vendor_id || "");
const [warehouseId, setWarehouseId] = useState<string | number>(purchaseReturn?.warehouse_id || "");
const [returnDate, setReturnDate] = useState(purchaseReturn?.return_date || new Date().toISOString().split('T')[0]);
const [items, setItems] = useState<PurchaseReturnItem[]>(purchaseReturn?.items || []);
const [remarks, setRemarks] = useState(purchaseReturn?.remarks || "");
const [status, setStatus] = useState<PurchaseReturnStatus>(purchaseReturn?.status || "draft");
const [taxAmount, setTaxAmount] = useState<string | number>(purchaseReturn?.tax_amount || 0);
// 同步外部傳入的 order 更新 (例如重新執行 edit 路由)
useEffect(() => {
if (purchaseReturn) {
setVendorId(purchaseReturn.vendor_id);
setWarehouseId(purchaseReturn.warehouse_id);
setReturnDate(purchaseReturn.return_date);
setItems(purchaseReturn.items || []);
setRemarks(purchaseReturn.remarks || "");
setStatus(purchaseReturn.status);
setTaxAmount(purchaseReturn.tax_amount || 0);
}
}, [purchaseReturn]);
const resetForm = () => {
setVendorId("");
setWarehouseId("");
setReturnDate(new Date().toISOString().split('T')[0]);
setItems([]);
setRemarks("");
setStatus("draft");
setTaxAmount(0);
};
const selectedVendor = suppliers.find((s) => String(s.id) === String(vendorId));
// 新增商品項目
const addItem = () => {
if (!selectedVendor) return;
setItems([
...items,
{
product_id: 0,
quantity_returned: 1,
unit_price: 0,
total_amount: 0,
batch_number: "",
},
]);
};
// 移除商品項目
const removeItem = (index: number) => {
setItems(items.filter((_, i) => i !== index));
};
// 更新商品項目
const updateItem = (index: number, field: keyof PurchaseReturnItem, value: any) => {
const newItems = [...items];
const item = { ...newItems[index] };
if (field === "product_id" && selectedVendor) {
const product = selectedVendor.commonProducts.find((p) => String(p.productId) === String(value));
if (product) {
// @ts-ignore
item.product_id = Number(value);
item.product_name = product.productName;
item.unit_price = product.lastPrice || 0;
item.total_amount = Number(item.quantity_returned) * Number(item.unit_price);
}
} else if (field === "total_amount") {
item.total_amount = Number(value);
if (item.quantity_returned > 0) {
item.unit_price = Number(item.total_amount) / Number(item.quantity_returned);
}
} else if (field === "quantity_returned" || field === "unit_price") {
// @ts-ignore
item[field] = Number(value);
if (field === "quantity_returned" && item.unit_price > 0) {
item.total_amount = Number(value) * item.unit_price;
} else if (field === "unit_price" && item.quantity_returned > 0) {
item.total_amount = item.quantity_returned * Number(value);
}
} else {
// @ts-ignore
item[field] = value;
}
newItems[index] = item;
setItems(newItems);
};
return {
vendorId,
warehouseId,
returnDate,
items,
remarks,
status,
selectedVendor,
taxAmount,
setVendorId,
setWarehouseId,
setReturnDate,
setRemarks,
setStatus,
setTaxAmount,
addItem,
removeItem,
updateItem,
resetForm,
};
}

View File

@@ -0,0 +1,48 @@
export type PurchaseReturnStatus = "draft" | "completed" | "cancelled";
export interface PurchaseReturnItem {
id?: number;
purchase_return_id?: number;
product_id: number;
quantity_returned: number;
unit_price: number;
total_amount: number;
batch_number?: string | null;
product_code?: string;
product_name?: string;
product?: any; // 當有關聯商品時
}
export interface PurchaseReturn {
id: number;
code: string;
vendor_id: number;
warehouse_id: number;
user_id: number;
return_date: string;
status: PurchaseReturnStatus;
total_amount: number;
tax_amount: number;
grand_total: number;
remarks?: string | null;
created_at: string;
updated_at: string;
// 關聯資料
vendor?: {
id: number;
name: string;
};
user?: {
id: number;
name: string;
};
items?: PurchaseReturnItem[];
warehouse_name?: string;
}
export const PURCHASE_RETURN_STATUS_CONFIG: Record<PurchaseReturnStatus, { label: string; color: string }> = {
draft: { label: "草稿", color: "bg-gray-100 text-gray-800" },
completed: { label: "已完成", color: "bg-green-100 text-green-800" },
cancelled: { label: "已作廢", color: "bg-red-100 text-red-800" },
};

View File

@@ -22,6 +22,10 @@ export const BREADCRUMB_MAP: Record<string, BreadcrumbItemType[]> = {
{ label: "供應鏈管理", href: '#' },
{ label: "採購單管理", href: "/purchase-orders", isPage: true }
],
purchaseReturns: [
{ label: "供應鏈管理", href: '#' },
{ label: "採購退回單管理", href: "/purchase-returns", isPage: true }
],
productionOrders: [
{ label: "生產管理" },
{ label: "生產工單", href: "/production-orders", isPage: true }