-
This commit is contained in:
204
frontend/src/modules/warehouse/pages/InventoryListPage.tsx
Normal file
204
frontend/src/modules/warehouse/pages/InventoryListPage.tsx
Normal file
@@ -0,0 +1,204 @@
|
||||
import { useState } from "react";
|
||||
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
|
||||
import { useNavigate } from "react-router-dom";
|
||||
import {
|
||||
Box,
|
||||
Typography,
|
||||
Button,
|
||||
Paper,
|
||||
Chip,
|
||||
IconButton,
|
||||
Tooltip,
|
||||
} from "@mui/material";
|
||||
import {
|
||||
DataGrid,
|
||||
GridColDef,
|
||||
GridRenderCellParams,
|
||||
GridToolbar,
|
||||
} from "@mui/x-data-grid";
|
||||
import {
|
||||
Add as AddIcon,
|
||||
Visibility as ViewIcon,
|
||||
PlayArrow as StartIcon,
|
||||
Cancel as CancelIcon,
|
||||
} from "@mui/icons-material";
|
||||
import { inventoryService } from "../services/warehouseService";
|
||||
import { InventoryCountDto, InventoryStatus } from "../types";
|
||||
import dayjs from "dayjs";
|
||||
|
||||
export default function InventoryListPage() {
|
||||
const navigate = useNavigate();
|
||||
const queryClient = useQueryClient();
|
||||
const [statusFilter] = useState<InventoryStatus | undefined>(
|
||||
undefined
|
||||
);
|
||||
|
||||
const { data: inventories = [], isLoading } = useQuery({
|
||||
queryKey: ["inventory-counts", statusFilter],
|
||||
queryFn: () => inventoryService.getAll(statusFilter),
|
||||
});
|
||||
|
||||
const cancelMutation = useMutation({
|
||||
mutationFn: (id: number) => inventoryService.cancel(id),
|
||||
onSuccess: () => {
|
||||
queryClient.invalidateQueries({ queryKey: ["inventory-counts"] });
|
||||
},
|
||||
});
|
||||
|
||||
const handleCreate = () => {
|
||||
navigate("/warehouse/inventory/new");
|
||||
};
|
||||
|
||||
const handleView = (id: number) => {
|
||||
navigate(`/warehouse/inventory/${id}`);
|
||||
};
|
||||
|
||||
const handleStart = (id: number) => {
|
||||
navigate(`/warehouse/inventory/${id}/count`);
|
||||
};
|
||||
|
||||
const getStatusChip = (status: InventoryStatus) => {
|
||||
switch (status) {
|
||||
case InventoryStatus.Draft:
|
||||
return <Chip label="Bozza" size="small" />;
|
||||
case InventoryStatus.InProgress:
|
||||
return <Chip label="In Corso" color="primary" size="small" />;
|
||||
case InventoryStatus.Completed:
|
||||
return <Chip label="Completato" color="info" size="small" />;
|
||||
case InventoryStatus.Confirmed:
|
||||
return <Chip label="Confermato" color="success" size="small" />;
|
||||
case InventoryStatus.Cancelled:
|
||||
return <Chip label="Annullato" color="error" size="small" />;
|
||||
default:
|
||||
return <Chip label={status} size="small" />;
|
||||
}
|
||||
};
|
||||
|
||||
const columns: GridColDef[] = [
|
||||
{ field: "code", headerName: "Codice", width: 120 },
|
||||
{ field: "description", headerName: "Descrizione", flex: 1, minWidth: 200 },
|
||||
{
|
||||
field: "inventoryDate",
|
||||
headerName: "Data Inventario",
|
||||
width: 150,
|
||||
valueFormatter: (value) =>
|
||||
value ? dayjs(value).format("DD/MM/YYYY") : "",
|
||||
},
|
||||
{ field: "warehouseName", headerName: "Magazzino", width: 180 },
|
||||
{ field: "categoryName", headerName: "Categoria", width: 150 },
|
||||
{
|
||||
field: "status",
|
||||
headerName: "Stato",
|
||||
width: 120,
|
||||
renderCell: (params: GridRenderCellParams<InventoryCountDto>) =>
|
||||
getStatusChip(params.row.status),
|
||||
},
|
||||
{
|
||||
field: "progress",
|
||||
headerName: "Progresso",
|
||||
width: 150,
|
||||
valueGetter: (_value, row) => {
|
||||
if (!row.lineCount) return "0%";
|
||||
const percentage = Math.round(
|
||||
(row.countedLineCount / row.lineCount) * 100
|
||||
);
|
||||
return `${row.countedLineCount}/${row.lineCount} (${percentage}%)`;
|
||||
},
|
||||
},
|
||||
{
|
||||
field: "actions",
|
||||
headerName: "Azioni",
|
||||
width: 180,
|
||||
sortable: false,
|
||||
renderCell: (params: GridRenderCellParams<InventoryCountDto>) => (
|
||||
<Box>
|
||||
<Tooltip title="Dettaglio">
|
||||
<IconButton size="small" onClick={() => handleView(params.row.id)}>
|
||||
<ViewIcon />
|
||||
</IconButton>
|
||||
</Tooltip>
|
||||
{params.row.status === InventoryStatus.Draft && (
|
||||
<Tooltip title="Avvia Conteggio">
|
||||
<IconButton
|
||||
size="small"
|
||||
color="primary"
|
||||
onClick={() => handleStart(params.row.id)}
|
||||
>
|
||||
<StartIcon />
|
||||
</IconButton>
|
||||
</Tooltip>
|
||||
)}
|
||||
{params.row.status === InventoryStatus.InProgress && (
|
||||
<Tooltip title="Continua Conteggio">
|
||||
<IconButton
|
||||
size="small"
|
||||
color="primary"
|
||||
onClick={() => handleStart(params.row.id)}
|
||||
>
|
||||
<StartIcon />
|
||||
</IconButton>
|
||||
</Tooltip>
|
||||
)}
|
||||
{params.row.status === InventoryStatus.Draft && (
|
||||
<Tooltip title="Annulla">
|
||||
<IconButton
|
||||
size="small"
|
||||
color="error"
|
||||
onClick={() => {
|
||||
if (confirm("Sei sicuro di voler annullare questo inventario?")) {
|
||||
cancelMutation.mutate(params.row.id);
|
||||
}
|
||||
}}
|
||||
>
|
||||
<CancelIcon />
|
||||
</IconButton>
|
||||
</Tooltip>
|
||||
)}
|
||||
</Box>
|
||||
),
|
||||
},
|
||||
];
|
||||
|
||||
return (
|
||||
<Box sx={{ p: 3 }}>
|
||||
<Box
|
||||
sx={{
|
||||
display: "flex",
|
||||
justifyContent: "space-between",
|
||||
alignItems: "center",
|
||||
mb: 3,
|
||||
}}
|
||||
>
|
||||
<Typography variant="h4">Inventari Fisici</Typography>
|
||||
<Button
|
||||
variant="contained"
|
||||
startIcon={<AddIcon />}
|
||||
onClick={handleCreate}
|
||||
>
|
||||
Nuovo Inventario
|
||||
</Button>
|
||||
</Box>
|
||||
|
||||
<Paper sx={{ height: 600, width: "100%" }}>
|
||||
<DataGrid
|
||||
rows={inventories}
|
||||
columns={columns}
|
||||
loading={isLoading}
|
||||
slots={{ toolbar: GridToolbar }}
|
||||
slotProps={{
|
||||
toolbar: {
|
||||
showQuickFilter: true,
|
||||
},
|
||||
}}
|
||||
disableRowSelectionOnClick
|
||||
initialState={{
|
||||
pagination: { paginationModel: { pageSize: 25 } },
|
||||
sorting: {
|
||||
sortModel: [{ field: "inventoryDate", sort: "desc" }],
|
||||
},
|
||||
}}
|
||||
/>
|
||||
</Paper>
|
||||
</Box>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user