- Implemented TemplateCheckpointEditDialog for creating and editing inspection checkpoints. - Added VendorActions component for managing vendor actions including edit, activate/deactivate, and delete. - Created VendorContext for managing vendor state across components. - Developed VendorGeneralInfo component to display detailed vendor information. - Introduced AedSymbol and Money components for consistent currency representation. - Added PromptDialog for user input prompts throughout the application. - Implemented RelationLink component for unified related-data display in CRUD tables. - Created InspectionTemplatesClient for API interactions related to inspection templates.
325 lines
12 KiB
TypeScript
325 lines
12 KiB
TypeScript
"use client"
|
|
|
|
import { useState } from "react"
|
|
import { AlertTriangle, Plus, Save, Trash2 } from "lucide-react"
|
|
import { useFieldArray } from "react-hook-form"
|
|
|
|
import { Button } from "@/shared/components/ui/button"
|
|
import { Input } from "@/shared/components/ui/input"
|
|
import { Label } from "@/shared/components/ui/label"
|
|
import { Alert, AlertTitle } from "@/shared/components/ui/alert"
|
|
import { FieldGroup } from "@/shared/components/ui/field"
|
|
import {
|
|
Rhform,
|
|
RhfTextField,
|
|
RhfTextareaField,
|
|
RhfAsyncSelectField,
|
|
RhfAutoGenerateField,
|
|
RhfSelectField,
|
|
type InlineCreateFormProps,
|
|
} from "@/shared/components/form"
|
|
import { toast } from "sonner"
|
|
import { useAuthApi } from "@/shared/useApi"
|
|
import { useResourceForm } from "@/shared/hooks/use-resource-form"
|
|
import { useFormMutation } from "@/shared/hooks/use-form-mutation"
|
|
import { toRelation, toId } from "@/shared/lib/utils"
|
|
|
|
import {
|
|
inventoryAdjustmentFormSchema,
|
|
type InventoryAdjustmentFormValues,
|
|
} from "./inventory-adjustment.schema"
|
|
import { INVENTORY_ADJUSTMENT_ROUTES, REASON_ROUTES, PARTS_ROUTES } from "@garage/api"
|
|
|
|
// ── Props ──
|
|
|
|
export type InventoryAdjustmentFormProps = {
|
|
resourceId?: string | null
|
|
initialData?: unknown
|
|
onSuccess?: () => void
|
|
}
|
|
|
|
// ── Default values ──
|
|
|
|
const OPERATION_OPTIONS = [
|
|
{ value: "add", label: "Increase" },
|
|
{ value: "subtract", label: "Decrease" },
|
|
]
|
|
|
|
const DEFAULT_VALUES: InventoryAdjustmentFormValues = {
|
|
reference_number: "",
|
|
date: "",
|
|
chart_of_account: undefined,
|
|
reason: null,
|
|
notes: "",
|
|
parts: [{ part: null, quantity: 1, operation: "add", rate: 0 }],
|
|
}
|
|
|
|
// ── Mapping helpers ──
|
|
|
|
function mapToFormValues(data: unknown): InventoryAdjustmentFormValues {
|
|
const d = (data as any)?.data ?? data ?? {}
|
|
// Laravel serializes the `inventoryAdjustmentParts` relation as
|
|
// `inventory_adjustment_parts` in JSON; fall back through the other
|
|
// shapes for resilience.
|
|
const rawParts: any[] =
|
|
(Array.isArray(d.inventory_adjustment_parts) && d.inventory_adjustment_parts)
|
|
|| (Array.isArray(d.inventoryAdjustmentParts) && d.inventoryAdjustmentParts)
|
|
|| (Array.isArray(d.parts) && d.parts)
|
|
|| []
|
|
|
|
// Backend casts `date` to Carbon → serializes as an ISO string
|
|
// ("2026-05-18T00:00:00.000000Z"). HTML <input type="date"> only accepts
|
|
// "YYYY-MM-DD", so slice the first 10 chars; pass through plain dates.
|
|
const rawDate = d.date ?? ""
|
|
const date = typeof rawDate === "string" && rawDate.length >= 10
|
|
? rawDate.slice(0, 10)
|
|
: ""
|
|
|
|
return {
|
|
reference_number: d.reference_number || "",
|
|
date,
|
|
chart_of_account: d.chart_of_account != null && d.chart_of_account !== ""
|
|
? Number(d.chart_of_account)
|
|
: undefined,
|
|
reason: toRelation(d.reason_id, d.reason?.title ?? d.reason?.name ?? d.reason_name),
|
|
notes: d.notes || "",
|
|
parts: rawParts.length > 0
|
|
? rawParts.map((p: any) => ({
|
|
part: toRelation(p.part_id, p.part?.title ?? p.part?.name ?? p.part_name),
|
|
quantity: p.quantity ?? 1,
|
|
operation: (p.operation === "subtract" ? "subtract" : "add") as "add" | "subtract",
|
|
rate: p.rate ?? 0,
|
|
}))
|
|
: [{ part: null, quantity: 1, operation: "add", rate: 0 }],
|
|
}
|
|
}
|
|
|
|
function mapFormToPayload(values: InventoryAdjustmentFormValues) {
|
|
return {
|
|
reference_number: values.reference_number || undefined,
|
|
date: values.date || undefined,
|
|
chart_of_account: values.chart_of_account ?? undefined,
|
|
reason_id: toId(values.reason) ? Number(toId(values.reason)) : undefined,
|
|
notes: values.notes || undefined,
|
|
parts: values.parts.map((p) => ({
|
|
part_id: toId(p.part) ? Number(toId(p.part)) : undefined,
|
|
quantity: p.quantity,
|
|
operation: p.operation,
|
|
rate: p.rate,
|
|
})),
|
|
}
|
|
}
|
|
|
|
const mapLookupOption = (item: any) => ({ value: String(item.id), label: item.name ?? item.title })
|
|
const STORE_OBJECT = { getOptionValue: (o: any) => o, getOptionLabel: (o: any) => o.label }
|
|
|
|
function ReasonInlineCreateForm({ onSuccess }: InlineCreateFormProps) {
|
|
const api = useAuthApi()
|
|
const [title, setTitle] = useState("")
|
|
const [isSaving, setIsSaving] = useState(false)
|
|
|
|
const handleSave = async () => {
|
|
const trimmed = title.trim()
|
|
if (!trimmed) {
|
|
toast.error("Reason title is required.")
|
|
return
|
|
}
|
|
setIsSaving(true)
|
|
try {
|
|
const response = (await api.reasons.create({ title: trimmed } as never)) as any
|
|
const created = response?.data ?? response
|
|
toast.success("Reason added.")
|
|
onSuccess(created?.id ? { value: String(created.id), label: created.title ?? trimmed } : undefined)
|
|
} catch {
|
|
toast.error("Failed to add reason.")
|
|
} finally {
|
|
setIsSaving(false)
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div className="flex flex-col gap-3">
|
|
<div className="flex flex-col gap-1.5">
|
|
<Label htmlFor="reason-title">Title</Label>
|
|
<Input
|
|
id="reason-title"
|
|
value={title}
|
|
onChange={(e) => setTitle(e.target.value)}
|
|
placeholder="e.g. Stock count correction"
|
|
autoFocus
|
|
/>
|
|
</div>
|
|
<div className="flex justify-end">
|
|
<Button type="button" onClick={handleSave} disabled={isSaving}>
|
|
<Plus className="size-4" />
|
|
{isSaving ? "Adding..." : "Add Reason"}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
// ── Component ──
|
|
|
|
export function InventoryAdjustmentForm({ resourceId, initialData, onSuccess }: InventoryAdjustmentFormProps) {
|
|
const api = useAuthApi()
|
|
|
|
const { form, isEditing } = useResourceForm<InventoryAdjustmentFormValues, any>({
|
|
schema: inventoryAdjustmentFormSchema,
|
|
defaultValues: DEFAULT_VALUES,
|
|
resourceId,
|
|
initialData,
|
|
mapToFormValues,
|
|
})
|
|
|
|
const { fields, append, remove } = useFieldArray({
|
|
control: form.control,
|
|
name: "parts",
|
|
})
|
|
|
|
const { mutate, error, isPending } = useFormMutation(form, {
|
|
mutationFn: (values: InventoryAdjustmentFormValues) => {
|
|
const payload = mapFormToPayload(values)
|
|
const promise = isEditing && resourceId
|
|
? api.inventoryAdjustments.update(resourceId, payload as never)
|
|
: api.inventoryAdjustments.create(payload as never)
|
|
return promise
|
|
},
|
|
onSuccess: () => {
|
|
toast.success(isEditing ? "Adjustment updated." : "Adjustment created.")
|
|
onSuccess?.()
|
|
},
|
|
})
|
|
|
|
return (
|
|
<Rhform form={form} onSubmit={mutate} className="flex flex-col gap-4">
|
|
{error && (
|
|
<Alert variant="destructive">
|
|
<AlertTriangle className="size-4" />
|
|
<AlertTitle>
|
|
{isEditing ? "Failed to update adjustment" : "Failed to create adjustment"}
|
|
</AlertTitle>
|
|
{error.message}
|
|
</Alert>
|
|
)}
|
|
|
|
<FieldGroup>
|
|
<div className="grid grid-cols-1 gap-4 sm:grid-cols-2">
|
|
<RhfAutoGenerateField
|
|
name="reference_number"
|
|
label="Reference Number"
|
|
placeholder="ADJ-0001"
|
|
table="inventory_adjustments"
|
|
autoFetch={!isEditing}
|
|
/>
|
|
<RhfTextField
|
|
name="date"
|
|
label="Date"
|
|
type="date"
|
|
/>
|
|
{/* TODO: replace with chart-of-accounts async select once the
|
|
module ships. Locked as integer + disabled for now. */}
|
|
<RhfTextField
|
|
name="chart_of_account"
|
|
label="Chart of Account"
|
|
type="number"
|
|
placeholder="Account number"
|
|
description="Pending chart-of-accounts module — disabled."
|
|
disabled
|
|
/>
|
|
<RhfAsyncSelectField
|
|
name="reason"
|
|
label="Reason"
|
|
placeholder="Select reason..."
|
|
queryKey={[REASON_ROUTES.INDEX]}
|
|
listFn={() => api.reasons.list()}
|
|
mapOption={mapLookupOption}
|
|
createLabel="Reason"
|
|
createForm={(props) => <ReasonInlineCreateForm {...props} />}
|
|
{...STORE_OBJECT}
|
|
/>
|
|
</div>
|
|
|
|
<RhfTextareaField
|
|
name="notes"
|
|
label="Notes"
|
|
placeholder="Additional notes..."
|
|
rows={3}
|
|
/>
|
|
</FieldGroup>
|
|
|
|
<div className="flex flex-col gap-2">
|
|
<div className="flex items-center justify-between">
|
|
<span className="text-sm font-medium">Parts</span>
|
|
<Button
|
|
type="button"
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={() => append({ part: null, quantity: 1, operation: "add", rate: 0 })}
|
|
>
|
|
<Plus className="size-4" />
|
|
Add Part
|
|
</Button>
|
|
</div>
|
|
|
|
{fields.length === 0 && (
|
|
<p className="text-sm text-muted-foreground">No parts added. Click "Add Part" to begin.</p>
|
|
)}
|
|
|
|
{fields.map((field, index) => (
|
|
<div
|
|
key={field.id}
|
|
className="grid items-end gap-2 rounded-lg border p-3 grid-cols-[minmax(0,1fr)_9rem_6rem_6rem_auto]"
|
|
>
|
|
<div className="min-w-0">
|
|
<RhfAsyncSelectField
|
|
name={`parts.${index}.part`}
|
|
label="Part"
|
|
placeholder="Select part..."
|
|
queryKey={[PARTS_ROUTES.INDEX]}
|
|
listFn={() => api.parts.list()}
|
|
mapOption={mapLookupOption}
|
|
{...STORE_OBJECT}
|
|
/>
|
|
</div>
|
|
<RhfSelectField
|
|
name={`parts.${index}.operation`}
|
|
label="Operation"
|
|
options={OPERATION_OPTIONS}
|
|
/>
|
|
<RhfTextField
|
|
name={`parts.${index}.quantity`}
|
|
label="Qty"
|
|
type="number"
|
|
placeholder="1"
|
|
/>
|
|
<RhfTextField
|
|
name={`parts.${index}.rate`}
|
|
label="Rate"
|
|
type="number"
|
|
placeholder="0"
|
|
/>
|
|
<Button
|
|
type="button"
|
|
variant="ghost"
|
|
size="icon"
|
|
className="mb-0.5"
|
|
onClick={() => remove(index)}
|
|
disabled={fields.length === 1}
|
|
>
|
|
<Trash2 className="size-4 text-destructive" />
|
|
</Button>
|
|
</div>
|
|
))}
|
|
</div>
|
|
|
|
<div className="flex justify-end">
|
|
<Button type="submit" disabled={isPending}>
|
|
<Save className="size-4" />
|
|
{isPending ? "Saving..." : isEditing ? "Update Adjustment" : "Create Adjustment"}
|
|
</Button>
|
|
</div>
|
|
</Rhform>
|
|
)
|
|
}
|