mirror of
https://github.com/ForFarmTeam/ForFarm.git
synced 2025-12-19 22:14:08 +01:00
feat: update inventory item structure and improve error handling
This commit is contained in:
parent
01bc37a136
commit
f037bf766f
@ -100,7 +100,6 @@ type HarvestUnit struct {
|
|||||||
|
|
||||||
type CreateInventoryItemInput struct {
|
type CreateInventoryItemInput struct {
|
||||||
Header string `header:"Authorization" required:"true" example:"Bearer token"`
|
Header string `header:"Authorization" required:"true" example:"Bearer token"`
|
||||||
UserID string `header:"userId" required:"true" example:"user-uuid"`
|
|
||||||
Body struct {
|
Body struct {
|
||||||
Name string `json:"name" required:"true"`
|
Name string `json:"name" required:"true"`
|
||||||
CategoryID int `json:"categoryId" required:"true"`
|
CategoryID int `json:"categoryId" required:"true"`
|
||||||
@ -119,7 +118,6 @@ type CreateInventoryItemOutput struct {
|
|||||||
|
|
||||||
type UpdateInventoryItemInput struct {
|
type UpdateInventoryItemInput struct {
|
||||||
Header string `header:"Authorization" required:"true" example:"Bearer token"`
|
Header string `header:"Authorization" required:"true" example:"Bearer token"`
|
||||||
UserID string `header:"userId" required:"true" example:"user-uuid"`
|
|
||||||
ID string `path:"id"`
|
ID string `path:"id"`
|
||||||
Body struct {
|
Body struct {
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
@ -137,7 +135,6 @@ type UpdateInventoryItemOutput struct {
|
|||||||
|
|
||||||
type GetInventoryItemsInput struct {
|
type GetInventoryItemsInput struct {
|
||||||
Header string `header:"Authorization" required:"true" example:"Bearer token"`
|
Header string `header:"Authorization" required:"true" example:"Bearer token"`
|
||||||
UserID string `header:"userId" required:"true" example:"user-uuid"`
|
|
||||||
CategoryID int `query:"categoryId"`
|
CategoryID int `query:"categoryId"`
|
||||||
StatusID int `query:"statusId"`
|
StatusID int `query:"statusId"`
|
||||||
StartDate time.Time `query:"startDate" format:"date-time"`
|
StartDate time.Time `query:"startDate" format:"date-time"`
|
||||||
@ -186,8 +183,9 @@ type GetHarvestUnitsOutput struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInventoryItemInput) (*CreateInventoryItemOutput, error) {
|
func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInventoryItemInput) (*CreateInventoryItemOutput, error) {
|
||||||
|
userID, err := a.getUserIDFromHeader(input.Header)
|
||||||
item := &domain.InventoryItem{
|
item := &domain.InventoryItem{
|
||||||
UserID: input.UserID,
|
UserID: userID,
|
||||||
Name: input.Body.Name,
|
Name: input.Body.Name,
|
||||||
CategoryID: input.Body.CategoryID,
|
CategoryID: input.Body.CategoryID,
|
||||||
Quantity: input.Body.Quantity,
|
Quantity: input.Body.Quantity,
|
||||||
@ -200,7 +198,7 @@ func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInven
|
|||||||
return nil, huma.Error422UnprocessableEntity(err.Error())
|
return nil, huma.Error422UnprocessableEntity(err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
err := a.inventoryRepo.CreateOrUpdate(ctx, item)
|
err = a.inventoryRepo.CreateOrUpdate(ctx, item)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -211,8 +209,9 @@ func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInven
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (a *api) getInventoryItemsByUserHandler(ctx context.Context, input *GetInventoryItemsInput) (*GetInventoryItemsOutput, error) {
|
func (a *api) getInventoryItemsByUserHandler(ctx context.Context, input *GetInventoryItemsInput) (*GetInventoryItemsOutput, error) {
|
||||||
|
userID, err := a.getUserIDFromHeader(input.Header)
|
||||||
filter := domain.InventoryFilter{
|
filter := domain.InventoryFilter{
|
||||||
UserID: input.UserID,
|
UserID: userID,
|
||||||
CategoryID: input.CategoryID,
|
CategoryID: input.CategoryID,
|
||||||
StatusID: input.StatusID,
|
StatusID: input.StatusID,
|
||||||
StartDate: input.StartDate,
|
StartDate: input.StartDate,
|
||||||
@ -225,7 +224,7 @@ func (a *api) getInventoryItemsByUserHandler(ctx context.Context, input *GetInve
|
|||||||
Direction: input.SortOrder,
|
Direction: input.SortOrder,
|
||||||
}
|
}
|
||||||
|
|
||||||
items, err := a.inventoryRepo.GetByUserID(ctx, input.UserID, filter, sort)
|
items, err := a.inventoryRepo.GetByUserID(ctx, userID, filter, sort)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -286,7 +285,8 @@ func (a *api) getInventoryItemHandler(ctx context.Context, input *GetInventoryIt
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (a *api) updateInventoryItemHandler(ctx context.Context, input *UpdateInventoryItemInput) (*UpdateInventoryItemOutput, error) {
|
func (a *api) updateInventoryItemHandler(ctx context.Context, input *UpdateInventoryItemInput) (*UpdateInventoryItemOutput, error) {
|
||||||
item, err := a.inventoryRepo.GetByID(ctx, input.ID, input.UserID)
|
userID, err := a.getUserIDFromHeader(input.Header)
|
||||||
|
item, err := a.inventoryRepo.GetByID(ctx, input.ID, userID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -319,7 +319,7 @@ func (a *api) updateInventoryItemHandler(ctx context.Context, input *UpdateInven
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
updatedItem, err := a.inventoryRepo.GetByID(ctx, input.ID, input.UserID)
|
updatedItem, err := a.inventoryRepo.GetByID(ctx, input.ID, userID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|||||||
@ -6,6 +6,7 @@ import type {
|
|||||||
CreateInventoryItemInput,
|
CreateInventoryItemInput,
|
||||||
UpdateInventoryItemInput,
|
UpdateInventoryItemInput,
|
||||||
} from "@/types";
|
} from "@/types";
|
||||||
|
import { AxiosError } from "axios";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simulates an API call to fetch inventory items.
|
* Simulates an API call to fetch inventory items.
|
||||||
@ -46,7 +47,6 @@ export async function fetchInventoryItems(): Promise<InventoryItem[]> {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error while fetching inventory items! " + error);
|
console.error("Error while fetching inventory items! " + error);
|
||||||
throw error;
|
throw error;
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,9 +59,32 @@ export async function createInventoryItem(
|
|||||||
item
|
item
|
||||||
);
|
);
|
||||||
return response.data;
|
return response.data;
|
||||||
} catch (error) {
|
} catch (error: unknown) {
|
||||||
console.error("Error while creating Inventory Item! " + error);
|
// Cast error to AxiosError to safely access response properties
|
||||||
throw new Error("Failed to create inventory item: " + error);
|
if (error instanceof AxiosError && error.response) {
|
||||||
|
// Log the detailed error message
|
||||||
|
console.error("Error while creating Inventory Item!");
|
||||||
|
console.error("Response Status:", error.response.status); // e.g., 422
|
||||||
|
console.error("Error Detail:", error.response.data?.detail); // Custom error message from backend
|
||||||
|
console.error("Full Error Response:", error.response.data); // Entire error object (including details)
|
||||||
|
|
||||||
|
// Throw a new error with a more specific message
|
||||||
|
throw new Error(
|
||||||
|
`Failed to create inventory item: ${
|
||||||
|
error.response.data?.detail || error.message
|
||||||
|
}`
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// Handle other errors (e.g., network errors or unknown errors)
|
||||||
|
console.error(
|
||||||
|
"Error while creating Inventory Item, unknown error:",
|
||||||
|
error
|
||||||
|
);
|
||||||
|
throw new Error(
|
||||||
|
"Failed to create inventory item: " +
|
||||||
|
(error instanceof Error ? error.message : "Unknown error")
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -3,8 +3,7 @@
|
|||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { CalendarIcon } from "lucide-react";
|
import { CalendarIcon } from "lucide-react";
|
||||||
import { format } from "date-fns";
|
import { format } from "date-fns";
|
||||||
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query";
|
import { useMutation, useQueryClient } from "@tanstack/react-query";
|
||||||
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { Calendar } from "@/components/ui/calendar";
|
import { Calendar } from "@/components/ui/calendar";
|
||||||
import {
|
import {
|
||||||
@ -59,29 +58,51 @@ export function AddInventoryItem({
|
|||||||
const [itemQuantity, setItemQuantity] = useState(0);
|
const [itemQuantity, setItemQuantity] = useState(0);
|
||||||
const [itemUnit, setItemUnit] = useState("");
|
const [itemUnit, setItemUnit] = useState("");
|
||||||
const [itemStatus, setItemStatus] = useState("");
|
const [itemStatus, setItemStatus] = useState("");
|
||||||
|
const [isSubmitted, setIsSubmitted] = useState(false);
|
||||||
|
const [successMessage, setSuccessMessage] = useState("");
|
||||||
|
const [errorMessage, setErrorMessage] = useState("");
|
||||||
|
const queryClient = useQueryClient();
|
||||||
const mutation = useMutation({
|
const mutation = useMutation({
|
||||||
mutationFn: (item: CreateInventoryItemInput) => createInventoryItem(item),
|
mutationFn: (item: CreateInventoryItemInput) => createInventoryItem(item),
|
||||||
onSuccess: () => {
|
onSuccess: () => {
|
||||||
// invalidate queries to refresh inventory data.
|
// invalidate queries to refresh inventory data
|
||||||
const queryClient = useQueryClient();
|
|
||||||
queryClient.invalidateQueries({ queryKey: ["inventoryItems"] });
|
queryClient.invalidateQueries({ queryKey: ["inventoryItems"] });
|
||||||
// reset form fields and close dialog.
|
|
||||||
setItemName("");
|
setItemName("");
|
||||||
setItemCategory("");
|
setItemCategory("");
|
||||||
setItemQuantity(0);
|
setItemQuantity(0);
|
||||||
setItemUnit("");
|
setItemUnit("");
|
||||||
setDate(undefined);
|
setDate(undefined);
|
||||||
|
setIsSubmitted(true);
|
||||||
|
setSuccessMessage("Item created successfully!");
|
||||||
|
|
||||||
|
// reset success message after 3 seconds
|
||||||
|
setTimeout(() => {
|
||||||
|
setIsSubmitted(false);
|
||||||
|
setSuccessMessage("");
|
||||||
setOpen(false);
|
setOpen(false);
|
||||||
|
}, 3000);
|
||||||
|
},
|
||||||
|
onError: (error: any) => {
|
||||||
|
console.error("Error creating item: ", error);
|
||||||
|
setErrorMessage(
|
||||||
|
"There was an error creating the item. Please try again."
|
||||||
|
);
|
||||||
|
|
||||||
|
// reset success message after 3 seconds
|
||||||
|
setTimeout(() => {
|
||||||
|
setErrorMessage("");
|
||||||
|
}, 3000);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const inputStates = [itemName, itemCategory, itemUnit, itemStatus, date];
|
const inputStates = [itemName, itemCategory, itemUnit, itemStatus, date];
|
||||||
const isInputValid = inputStates.every((input) => input);
|
const isInputValid = inputStates.every((input) => input);
|
||||||
|
|
||||||
const handleSave = () => {
|
const handleSave = () => {
|
||||||
if (!isInputValid) {
|
if (!isInputValid) {
|
||||||
console.error("All fields are required");
|
setErrorMessage(
|
||||||
|
"There was an error creating the item. Please try again."
|
||||||
|
);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -93,13 +114,14 @@ export function AddInventoryItem({
|
|||||||
unitId: harvestUnits.find((item) => item.name === itemUnit)?.id || 0,
|
unitId: harvestUnits.find((item) => item.name === itemUnit)?.id || 0,
|
||||||
statusId:
|
statusId:
|
||||||
inventoryStatus.find((item) => item.name === itemStatus)?.id || 0,
|
inventoryStatus.find((item) => item.name === itemStatus)?.id || 0,
|
||||||
lastUpdated: date ? date.toISOString() : new Date().toISOString(),
|
dateAdded: date ? date.toISOString() : new Date().toISOString(),
|
||||||
};
|
};
|
||||||
console.table(newItem);
|
// console.table(newItem);
|
||||||
mutation.mutate(newItem);
|
mutation.mutate(newItem);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
<Dialog open={open} onOpenChange={setOpen}>
|
<Dialog open={open} onOpenChange={setOpen}>
|
||||||
<DialogTrigger asChild>
|
<DialogTrigger asChild>
|
||||||
<Button>Add New Item</Button>
|
<Button>Add New Item</Button>
|
||||||
@ -134,8 +156,11 @@ export function AddInventoryItem({
|
|||||||
<SelectContent>
|
<SelectContent>
|
||||||
<SelectGroup>
|
<SelectGroup>
|
||||||
<SelectLabel>Category</SelectLabel>
|
<SelectLabel>Category</SelectLabel>
|
||||||
{inventoryCategory.map((categoryItem, _) => (
|
{inventoryCategory.map((categoryItem) => (
|
||||||
<SelectItem key={categoryItem.id} value={categoryItem.name}>
|
<SelectItem
|
||||||
|
key={categoryItem.id}
|
||||||
|
value={categoryItem.name}
|
||||||
|
>
|
||||||
{categoryItem.name}
|
{categoryItem.name}
|
||||||
</SelectItem>
|
</SelectItem>
|
||||||
))}
|
))}
|
||||||
@ -154,7 +179,7 @@ export function AddInventoryItem({
|
|||||||
<SelectContent>
|
<SelectContent>
|
||||||
<SelectGroup>
|
<SelectGroup>
|
||||||
<SelectLabel>Status</SelectLabel>
|
<SelectLabel>Status</SelectLabel>
|
||||||
{inventoryStatus.map((statusItem, _) => (
|
{inventoryStatus.map((statusItem) => (
|
||||||
<SelectItem key={statusItem.id} value={statusItem.name}>
|
<SelectItem key={statusItem.id} value={statusItem.name}>
|
||||||
{statusItem.name}
|
{statusItem.name}
|
||||||
</SelectItem>
|
</SelectItem>
|
||||||
@ -186,7 +211,7 @@ export function AddInventoryItem({
|
|||||||
<SelectContent>
|
<SelectContent>
|
||||||
<SelectGroup>
|
<SelectGroup>
|
||||||
<SelectLabel>Unit</SelectLabel>
|
<SelectLabel>Unit</SelectLabel>
|
||||||
{harvestUnits.map((unit, _) => (
|
{harvestUnits.map((unit) => (
|
||||||
<SelectItem key={unit.id} value={unit.name}>
|
<SelectItem key={unit.id} value={unit.name}>
|
||||||
{unit.name}
|
{unit.name}
|
||||||
</SelectItem>
|
</SelectItem>
|
||||||
@ -224,11 +249,26 @@ export function AddInventoryItem({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<DialogFooter>
|
<DialogFooter>
|
||||||
<Button type="submit" onClick={handleSave}>
|
<div className="flex flex-col items-center w-full space-y-2">
|
||||||
|
<Button type="button" onClick={handleSave} className="w-full">
|
||||||
Save Item
|
Save Item
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<div className="flex flex-col items-center space-y-2">
|
||||||
|
{isSubmitted && (
|
||||||
|
<p className="text-green-500 text-sm">
|
||||||
|
{successMessage} You may close this window.
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{errorMessage && (
|
||||||
|
<p className="text-red-500 text-sm">{errorMessage}</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</DialogFooter>
|
</DialogFooter>
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -34,9 +34,9 @@ import type { UpdateInventoryItemInput } from "@/types";
|
|||||||
export interface EditInventoryItemProps {
|
export interface EditInventoryItemProps {
|
||||||
id: string;
|
id: string;
|
||||||
name: string;
|
name: string;
|
||||||
categoryId: string;
|
categoryId: number;
|
||||||
statusId: string;
|
statusId: number;
|
||||||
unitId: string;
|
unitId: number;
|
||||||
quantity: number;
|
quantity: number;
|
||||||
fetchedInventoryStatus: InventoryStatus[];
|
fetchedInventoryStatus: InventoryStatus[];
|
||||||
fetchedInventoryCategory: InventoryItemCategory[];
|
fetchedInventoryCategory: InventoryItemCategory[];
|
||||||
@ -58,19 +58,16 @@ export function EditInventoryItem({
|
|||||||
const [itemName, setItemName] = useState(name);
|
const [itemName, setItemName] = useState(name);
|
||||||
const [itemCategory, setItemCategory] = useState(
|
const [itemCategory, setItemCategory] = useState(
|
||||||
fetchedInventoryCategory.find(
|
fetchedInventoryCategory.find(
|
||||||
(categoryItem) => categoryItem.id.toString() === categoryId
|
(categoryItem) => categoryItem.id === categoryId
|
||||||
)?.name
|
)?.name
|
||||||
);
|
);
|
||||||
const [itemQuantity, setItemQuantity] = useState(quantity);
|
const [itemQuantity, setItemQuantity] = useState(quantity);
|
||||||
const [itemUnit, setItemUnit] = useState(
|
const [itemUnit, setItemUnit] = useState(
|
||||||
fetchedHarvestUnits.find(
|
fetchedHarvestUnits.find((harvestItem) => harvestItem.id === unitId)?.name
|
||||||
(harvestItem) => harvestItem.id.toString() === unitId
|
|
||||||
)?.name
|
|
||||||
);
|
);
|
||||||
const [itemStatus, setItemStatus] = useState(
|
const [itemStatus, setItemStatus] = useState(
|
||||||
fetchedInventoryStatus.find(
|
fetchedInventoryStatus.find((statusItem) => statusItem.id === statusId)
|
||||||
(statusItem) => statusItem.id.toString() === statusId
|
?.name
|
||||||
)?.name
|
|
||||||
);
|
);
|
||||||
const [error, setError] = useState<string | null>(null);
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
|
||||||
@ -137,7 +134,7 @@ export function EditInventoryItem({
|
|||||||
statusId:
|
statusId:
|
||||||
fetchedInventoryStatus.find((status) => status.name === itemStatus)
|
fetchedInventoryStatus.find((status) => status.name === itemStatus)
|
||||||
?.id ?? 0,
|
?.id ?? 0,
|
||||||
lastUpdated: new Date().toISOString(),
|
dateAdded: new Date().toISOString(),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@ -63,6 +63,8 @@ export default function InventoryPage() {
|
|||||||
queryFn: fetchInventoryItems,
|
queryFn: fetchInventoryItems,
|
||||||
staleTime: 60 * 1000,
|
staleTime: 60 * 1000,
|
||||||
});
|
});
|
||||||
|
// console.table(inventoryItems);
|
||||||
|
// console.log(inventoryItems);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
data: inventoryStatus = [],
|
data: inventoryStatus = [],
|
||||||
@ -73,6 +75,8 @@ export default function InventoryPage() {
|
|||||||
queryFn: fetchInventoryStatus,
|
queryFn: fetchInventoryStatus,
|
||||||
staleTime: 60 * 1000,
|
staleTime: 60 * 1000,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// console.log(inventoryStatus);
|
||||||
const {
|
const {
|
||||||
data: inventoryCategory = [],
|
data: inventoryCategory = [],
|
||||||
isLoading: isLoadingCategory,
|
isLoading: isLoadingCategory,
|
||||||
@ -101,24 +105,23 @@ export default function InventoryPage() {
|
|||||||
return inventoryItems
|
return inventoryItems
|
||||||
.map((item) => ({
|
.map((item) => ({
|
||||||
...item,
|
...item,
|
||||||
status:
|
status: item.status.name,
|
||||||
inventoryStatus.find(
|
category: item.category.name,
|
||||||
(statusItem) => statusItem.id.toString() === item.statusId.toString()
|
categoryId: item.categoryId,
|
||||||
)?.name || "",
|
unit: item.unit.name,
|
||||||
category:
|
unitId: item.unitId,
|
||||||
inventoryCategory.find(
|
statusId: item.statusId,
|
||||||
(categoryItem) =>
|
|
||||||
categoryItem.id.toString() === item.categoryId.toString()
|
|
||||||
)?.name || "",
|
|
||||||
categoryId: item.categoryId.toString(),
|
|
||||||
unit:
|
|
||||||
harvestUnits.find((unit) => unit.id === item.unitId)
|
|
||||||
?.name || "",
|
|
||||||
unitId: item.unitId.toString(),
|
|
||||||
statusId: item.statusId.toString(),
|
|
||||||
fetchedInventoryStatus: inventoryStatus,
|
fetchedInventoryStatus: inventoryStatus,
|
||||||
fetchedInventoryCategory: inventoryCategory,
|
fetchedInventoryCategory: inventoryCategory,
|
||||||
fetchedHarvestUnits: harvestUnits,
|
fetchedHarvestUnits: harvestUnits,
|
||||||
|
lastUpdated: new Date(item.updatedAt).toLocaleString("en-US", {
|
||||||
|
year: "numeric",
|
||||||
|
month: "short",
|
||||||
|
day: "numeric",
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
hour12: true,
|
||||||
|
}),
|
||||||
}))
|
}))
|
||||||
.filter((item) =>
|
.filter((item) =>
|
||||||
item.name.toLowerCase().includes(searchTerm.toLowerCase())
|
item.name.toLowerCase().includes(searchTerm.toLowerCase())
|
||||||
@ -139,12 +142,18 @@ export default function InventoryPage() {
|
|||||||
cell: (info: { getValue: () => string }) => {
|
cell: (info: { getValue: () => string }) => {
|
||||||
const status = info.getValue();
|
const status = info.getValue();
|
||||||
|
|
||||||
let statusClass = ""; // default status class
|
let statusClass = "";
|
||||||
|
|
||||||
if (status === "Low Stock") {
|
if (status === "In Stock") {
|
||||||
statusClass = "bg-yellow-300"; // yellow for low stock
|
statusClass = "bg-green-500 hover:bg-green-600 text-white";
|
||||||
} else if (status === "Out Of Stock") {
|
} else if (status === "Low Stock") {
|
||||||
statusClass = "bg-red-500 text-white"; // red for out of stock
|
statusClass = "bg-yellow-300 hover:bg-yellow-400";
|
||||||
|
} else if (status === "Out of Stock") {
|
||||||
|
statusClass = "bg-red-500 hover:bg-red-600 text-white";
|
||||||
|
} else if (status === "Expired") {
|
||||||
|
statusClass = "bg-gray-500 hover:bg-gray-600 text-white";
|
||||||
|
} else if (status === "Reserved") {
|
||||||
|
statusClass = "bg-blue-500 hover:bg-blue-600 text-white";
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -201,6 +210,19 @@ export default function InventoryPage() {
|
|||||||
|
|
||||||
const isLoading = loadingStates.some((loading) => loading);
|
const isLoading = loadingStates.some((loading) => loading);
|
||||||
const isError = errorStates.some((error) => error);
|
const isError = errorStates.some((error) => error);
|
||||||
|
if (isLoading)
|
||||||
|
return (
|
||||||
|
<div className="flex min-h-screen items-center justify-center">
|
||||||
|
Loading...
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (isError)
|
||||||
|
return (
|
||||||
|
<div className="flex min-h-screen items-center justify-center">
|
||||||
|
Error loading inventory data.
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
if (inventoryItems.length === 0) {
|
if (inventoryItems.length === 0) {
|
||||||
return (
|
return (
|
||||||
@ -210,26 +232,23 @@ export default function InventoryPage() {
|
|||||||
<TriangleAlertIcon className="h-6 w-6 text-red-500 mb-2" />
|
<TriangleAlertIcon className="h-6 w-6 text-red-500 mb-2" />
|
||||||
<AlertTitle>No Inventory Data</AlertTitle>
|
<AlertTitle>No Inventory Data</AlertTitle>
|
||||||
<AlertDescription>
|
<AlertDescription>
|
||||||
|
<div>
|
||||||
You currently have no inventory items. Add a new item to get
|
You currently have no inventory items. Add a new item to get
|
||||||
started!
|
started!
|
||||||
|
</div>
|
||||||
|
<div className="mt-5">
|
||||||
|
<AddInventoryItem
|
||||||
|
inventoryCategory={inventoryCategory}
|
||||||
|
inventoryStatus={inventoryStatus}
|
||||||
|
harvestUnits={harvestUnits}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</AlertDescription>
|
</AlertDescription>
|
||||||
</div>
|
</div>
|
||||||
</Alert>
|
</Alert>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
if (isLoading)
|
|
||||||
return (
|
|
||||||
<div className="flex min-h-screen items-center justify-center">
|
|
||||||
Loading...
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
if (isError)
|
|
||||||
return (
|
|
||||||
<div className="flex min-h-screen items-center justify-center">
|
|
||||||
Error loading inventory data.
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex min-h-screen bg-background">
|
<div className="flex min-h-screen bg-background">
|
||||||
|
|||||||
@ -1,5 +1,9 @@
|
|||||||
lockfileVersion: '6.0'
|
lockfileVersion: '6.0'
|
||||||
|
|
||||||
|
settings:
|
||||||
|
autoInstallPeers: true
|
||||||
|
excludeLinksFromLockfile: false
|
||||||
|
|
||||||
dependencies:
|
dependencies:
|
||||||
'@hookform/resolvers':
|
'@hookform/resolvers':
|
||||||
specifier: ^4.0.0
|
specifier: ^4.0.0
|
||||||
@ -4889,7 +4893,3 @@ packages:
|
|||||||
/zod@3.24.2:
|
/zod@3.24.2:
|
||||||
resolution: {integrity: sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==}
|
resolution: {integrity: sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==}
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
settings:
|
|
||||||
autoInstallPeers: true
|
|
||||||
excludeLinksFromLockfile: false
|
|
||||||
|
|||||||
@ -156,7 +156,7 @@ export type CreateInventoryItemInput = {
|
|||||||
categoryId: number;
|
categoryId: number;
|
||||||
quantity: number;
|
quantity: number;
|
||||||
unitId: number;
|
unitId: number;
|
||||||
lastUpdated: string;
|
dateAdded: string;
|
||||||
statusId: number;
|
statusId: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -239,22 +239,32 @@ export interface SetOverlayAction {
|
|||||||
|
|
||||||
export type Action = ActionWithTypeOnly | SetOverlayAction;
|
export type Action = ActionWithTypeOnly | SetOverlayAction;
|
||||||
|
|
||||||
export function isCircle(overlay: OverlayGeometry): overlay is google.maps.Circle {
|
export function isCircle(
|
||||||
|
overlay: OverlayGeometry
|
||||||
|
): overlay is google.maps.Circle {
|
||||||
return (overlay as google.maps.Circle).getCenter !== undefined;
|
return (overlay as google.maps.Circle).getCenter !== undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isMarker(overlay: OverlayGeometry): overlay is google.maps.Marker {
|
export function isMarker(
|
||||||
|
overlay: OverlayGeometry
|
||||||
|
): overlay is google.maps.Marker {
|
||||||
return (overlay as google.maps.Marker).getPosition !== undefined;
|
return (overlay as google.maps.Marker).getPosition !== undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isPolygon(overlay: OverlayGeometry): overlay is google.maps.Polygon {
|
export function isPolygon(
|
||||||
|
overlay: OverlayGeometry
|
||||||
|
): overlay is google.maps.Polygon {
|
||||||
return (overlay as google.maps.Polygon).getPath !== undefined;
|
return (overlay as google.maps.Polygon).getPath !== undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isPolyline(overlay: OverlayGeometry): overlay is google.maps.Polyline {
|
export function isPolyline(
|
||||||
|
overlay: OverlayGeometry
|
||||||
|
): overlay is google.maps.Polyline {
|
||||||
return (overlay as google.maps.Polyline).getPath !== undefined;
|
return (overlay as google.maps.Polyline).getPath !== undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isRectangle(overlay: OverlayGeometry): overlay is google.maps.Rectangle {
|
export function isRectangle(
|
||||||
|
overlay: OverlayGeometry
|
||||||
|
): overlay is google.maps.Rectangle {
|
||||||
return (overlay as google.maps.Rectangle).getBounds !== undefined;
|
return (overlay as google.maps.Rectangle).getBounds !== undefined;
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user