Merge fix/back-front-validation-alignment into dev

This commit is contained in:
Najjar\NajjarV02 2026-05-12 17:34:16 +04:00
commit 464402f4d6
31 changed files with 205 additions and 119 deletions

View File

@ -10,20 +10,30 @@ const APPOINTMENT_STATUS_OPTIONS = AppointmentStatus.map((v) => ({
label: v.replace(/_/g, " ").replace(/\b\w/g, (c) => c.toUpperCase()),
}))
const appointmentFormSchema = z.object({
title: z.string().min(1, "Title is required"),
const appointmentFormSchema = z
.object({
title: z.string().min(1, "Title is required").max(255, "Title cannot exceed 255 characters"),
date: z.string().min(1, "Date is required"),
from_time: z.string().min(1, "Start time is required"),
to_time: z.string().min(1, "End time is required"),
customer: relationFieldSchema,
vehicle: relationFieldSchema,
service_writer: relationFieldSchema,
service_writer: relationFieldSchema.refine((val) => !!val?.value, "Service writer is required"),
technician: relationFieldSchema,
department: relationFieldSchema,
job_card: relationFieldSchema,
notes: z.string().optional(),
status: z.string().optional(),
})
})
.superRefine((val, ctx) => {
if (val.from_time && val.to_time && val.to_time <= val.from_time) {
ctx.addIssue({
code: z.ZodIssueCode.custom,
path: ["to_time"],
message: "End time must be after start time",
})
}
})
type AppointmentFormValues = z.infer<typeof appointmentFormSchema>

View File

@ -2,7 +2,7 @@ import { z } from "zod"
const loginFormSchema = z.object({
email: z.string().trim().email("Enter a valid email address"),
password: z.string().min(8, "Password must be at least 8 characters"),
password: z.string().min(6, "Password must be at least 6 characters"),
})
type LoginFormValues = z.infer<typeof loginFormSchema>

View File

@ -5,14 +5,14 @@ export const relationFieldSchema = z
.nullable()
export const expenseItemFormSchema = z.object({
item_type: z.string().min(1, "Item type is required"),
item_name: z.string().min(1, "Item name is required"),
sku: z.string().optional(),
item_type: z.string().min(1, "Item type is required").max(255, "Item type cannot exceed 255 characters"),
item_name: z.string().min(1, "Item name is required").max(255, "Item name cannot exceed 255 characters"),
sku: z.string().min(1, "SKU is required").max(255, "SKU cannot exceed 255 characters"),
item_code: z.string().optional(),
description: z.string().optional(),
category: relationFieldSchema,
unit_type: relationFieldSchema,
department: relationFieldSchema,
category: relationFieldSchema.refine((val) => !!val?.value, "Category is required"),
unit_type: relationFieldSchema.refine((val) => !!val?.value, "Unit type is required"),
department: relationFieldSchema.refine((val) => !!val?.value, "Department is required"),
// Purchase
purchase_information: z.boolean().default(true),
purchase_price: z.coerce.number().min(0).optional(),

View File

@ -5,17 +5,17 @@ const relationFieldSchema = z
.nullable()
const inspectionFormSchema = z.object({
customer: relationFieldSchema,
vehicle: relationFieldSchema,
department: relationFieldSchema,
inspection_category: relationFieldSchema,
employee: relationFieldSchema,
customer: relationFieldSchema.refine((val) => !!val?.value, "Customer is required"),
vehicle: relationFieldSchema.refine((val) => !!val?.value, "Vehicle is required"),
department: relationFieldSchema.refine((val) => !!val?.value, "Department is required"),
inspection_category: relationFieldSchema.refine((val) => !!val?.value, "Inspection category is required"),
employee: relationFieldSchema.refine((val) => !!val?.value, "Employee is required"),
job_card: relationFieldSchema.optional(),
labor_rate: relationFieldSchema.optional(),
title: z.string().min(1, "Title is required"),
order_number: z.string().optional(),
date: z.string().optional(),
time: z.string().optional(),
title: z.string().min(1, "Title is required").max(100, "Title cannot exceed 100 characters"),
order_number: z.string().min(1, "Order number is required").max(100, "Order number cannot exceed 100 characters"),
date: z.string().min(1, "Date is required"),
time: z.string().min(1, "Time is required"),
status: z.string().optional(),
note: z.string().optional(),
description: z.string().optional(),

View File

@ -11,8 +11,8 @@ const partLineSchema = z.object({
})
export const inventoryAdjustmentFormSchema = z.object({
reference_number: z.string().optional(),
date: z.string().optional(),
reference_number: z.string().min(1, "Reference number is required").max(255, "Reference number cannot exceed 255 characters"),
date: z.string().min(1, "Date is required"),
chart_of_account: z.string().optional(),
reason: relationFieldSchema,
notes: z.string().optional(),

View File

@ -22,13 +22,13 @@ import { z } from "zod"
// ── Schema for edit form (simplified, edit only) ──
const invoiceEditFormSchema = z.object({
subject: z.string().min(1, "Subject is required"),
subject: z.string().min(1, "Subject is required").max(255, "Subject cannot exceed 255 characters"),
due_date: z.string().optional(),
invoice_title: z.string().optional(),
invoice_title: z.string().max(255, "Invoice title cannot exceed 255 characters").optional(),
notes: z.string().optional(),
terms_and_conditions: z.string().optional(),
status: z.string().optional(),
discount: z.string().optional(),
status: z.enum(InvoiceStatus).optional(),
discount: z.enum(InvoiceDiscount).optional(),
})
type InvoiceEditFormValues = z.infer<typeof invoiceEditFormSchema>
@ -53,7 +53,7 @@ const DEFAULT_VALUES: InvoiceEditFormValues = {
invoice_title: "",
notes: "",
terms_and_conditions: "",
status: "",
status: undefined,
discount: "no",
}
@ -67,7 +67,7 @@ function mapToFormValues(data: unknown): InvoiceEditFormValues {
invoice_title: d.invoice_title || "",
notes: d.notes || "",
terms_and_conditions: d.terms_and_conditions || "",
status: d.status || "",
status: d.status || undefined,
discount: d.discount || "no",
}
}

View File

@ -36,7 +36,7 @@ const requiredRelationFieldSchema = relationFieldSchema.superRefine((value, ctx)
})
const jobCardExpenseItemFormSchema = z.object({
expense_item: relationFieldSchema,
expense_item: requiredRelationFieldSchema,
department: requiredRelationFieldSchema,
tax: relationFieldSchema.optional(),
quantity: z.coerce.number().min(1, "Quantity is required"),

View File

@ -27,8 +27,8 @@ import { useJobCard } from "./job-card-context"
const relationFieldSchema = z.object({ value: z.string(), label: z.string() }).nullable()
const jobCardPartFormSchema = z.object({
part: relationFieldSchema,
department: relationFieldSchema.optional(),
part: relationFieldSchema.refine((val) => !!val?.value, "Part is required"),
department: relationFieldSchema.refine((val) => !!val?.value, "Department is required"),
tax: relationFieldSchema.optional(),
quantity: z.coerce.number().min(1, "Quantity is required"),
rate: z.coerce.number().min(0, "Rate is required"),

View File

@ -11,7 +11,7 @@ import { toast } from "sonner"
import { useAuthApi } from "@/shared/useApi"
const schema = z.object({
recommendation: z.string().min(1, "Recommendation is required"),
recommendation: z.string().min(1, "Recommendation is required").max(255, "Recommendation cannot exceed 255 characters"),
})
type FormValues = z.infer<typeof schema>

View File

@ -28,8 +28,8 @@ import { useJobCard } from "./job-card-context"
const relationFieldSchema = z.object({ value: z.string(), label: z.string() }).nullable()
const jobCardServiceFormSchema = z.object({
service: relationFieldSchema,
department: z.object({ value: z.string(), label: z.string() }),
service: relationFieldSchema.refine((val) => !!val?.value, "Service is required"),
department: z.object({ value: z.string(), label: z.string() }, { error: "Department is required" }),
tax: relationFieldSchema.optional(),
rate_type: z.string().optional(),
labor_rate: relationFieldSchema.optional(),

View File

@ -16,7 +16,7 @@ const optionalStringMax255 = z.string().max(255).optional()
const optionalTimeString = z.string().max(50).optional()
const documentSchema = z.object({
document_type_id: z.coerce.number().int().optional(),
document_type_id: z.coerce.number({ error: "Document type is required" }).int(),
customer_id: z.coerce.number().int().optional(),
vehicle_id: z.coerce.number().int().optional(),
document_number: z.string().optional(),

View File

@ -5,12 +5,12 @@ export const relationFieldSchema = z
.nullable()
export const partFormSchema = z.object({
shop_type: relationFieldSchema,
category: relationFieldSchema,
unit_type: relationFieldSchema,
department: relationFieldSchema,
shop_type: relationFieldSchema.refine((val) => !!val?.value, "Shop type is required"),
category: relationFieldSchema.refine((val) => !!val?.value, "Category is required"),
unit_type: relationFieldSchema.refine((val) => !!val?.value, "Unit type is required"),
department: relationFieldSchema.refine((val) => !!val?.value, "Department is required"),
title: z.string().min(1, "Title is required"),
sku: z.string().optional(),
sku: z.string().min(1, "SKU is required"),
description: z.string().optional(),
selling_price: z.coerce.number().min(0).optional(),
purchase_price: z.coerce.number().min(0).optional(),

View File

@ -10,10 +10,11 @@ import { Rhform, RhfTextField, RhfSelectField, type InlineCreateFormProps } from
import { toast } from "sonner"
import { useAuthApi } from "@/shared/useApi"
import { DEPARTMENT_ASSIGNMENT_TYPE_OPTIONS } from "../department-assignment-types"
import { AssignmentType } from "@garage/api"
const schema = z.object({
name: z.string().min(1, "Name is required"),
assignment_type: z.string().optional(),
name: z.string().min(1, "Name is required").max(50, "Name cannot exceed 50 characters"),
assignment_type: z.enum(AssignmentType, { error: "Assignment type is required" }),
})
type FormValues = z.infer<typeof schema>

View File

@ -1,27 +1,53 @@
import { z } from "zod"
import { FirstDayOfWork } from "@garage/api"
export const relationFieldSchema = z
.object({ value: z.string(), label: z.string() })
.nullable()
const optionalLatitude = z
.string()
.optional()
.refine(
(v) => v == null || v === "" || (!Number.isNaN(Number(v)) && Number(v) >= -90 && Number(v) <= 90),
"Latitude must be between -90 and 90",
)
const optionalLongitude = z
.string()
.optional()
.refine(
(v) => v == null || v === "" || (!Number.isNaN(Number(v)) && Number(v) >= -180 && Number(v) <= 180),
"Longitude must be between -180 and 180",
)
const optionalUrl = z
.string()
.max(255, "Website cannot exceed 255 characters")
.optional()
.refine(
(v) => v == null || v === "" || z.string().url().safeParse(v).success,
"Enter a valid website URL",
)
export const settingsFormSchema = z.object({
name: z.string().min(1, "Name is required"),
name: z.string().min(1, "Name is required").max(100, "Name cannot exceed 100 characters"),
email: z.union([z.string().email("Invalid email address"), z.literal("")]).optional(),
phone: z.string().optional(),
alternative_phone: z.string().optional(),
website: z.string().optional(),
time_zone: z.string().optional(),
upi_id: z.string().optional(),
first_day_of_work: z.string().optional(),
latitude: z.string().optional(),
longitude: z.string().optional(),
phone: z.string().max(30, "Phone cannot exceed 30 characters").optional(),
alternative_phone: z.string().max(30, "Phone cannot exceed 30 characters").optional(),
website: optionalUrl,
time_zone: z.string().max(100, "Time zone cannot exceed 100 characters").optional(),
upi_id: z.string().max(100, "UPI cannot exceed 100 characters").optional(),
first_day_of_work: z.union([z.enum(FirstDayOfWork), z.literal("")]).optional(),
latitude: optionalLatitude,
longitude: optionalLongitude,
bank_details: z.string().optional(),
first_address_line: z.string().optional(),
second_address_line: z.string().optional(),
first_address_line: z.string().max(255, "Address cannot exceed 255 characters").optional(),
second_address_line: z.string().max(255, "Address cannot exceed 255 characters").optional(),
country: relationFieldSchema,
state: relationFieldSchema,
city: z.string().optional(),
zip_code: z.string().optional(),
city: z.string().max(100, "City cannot exceed 100 characters").optional(),
zip_code: z.string().max(20, "Zip code cannot exceed 20 characters").optional(),
description: z.string().optional(),
security: z.string().optional(),
privacy_policy: z.string().optional(),

View File

@ -1,24 +1,30 @@
import { z } from "zod"
import {
SellRatesTaxInclusive,
GiveDiscounts,
PurchaseRatesTaxInclusive,
ReceiveDiscounts,
} from "@garage/api"
export const salesConfigFormSchema = z.object({
sell_rates_tax_inclusive: z.string().optional(),
give_discounts: z.string().optional(),
sell_rates_tax_inclusive: z.enum(SellRatesTaxInclusive).optional(),
give_discounts: z.enum(GiveDiscounts).optional(),
})
export type SalesConfigFormValues = z.infer<typeof salesConfigFormSchema>
export const purchaseConfigFormSchema = z.object({
purchase_rates_tax_inclusive: z.string().optional(),
receive_discounts: z.string().optional(),
purchase_rates_tax_inclusive: z.enum(PurchaseRatesTaxInclusive).optional(),
receive_discounts: z.enum(ReceiveDiscounts).optional(),
})
export type PurchaseConfigFormValues = z.infer<typeof purchaseConfigFormSchema>
export const generalPreferencesFormSchema = z.object({
sell_rates_tax_inclusive: z.string().optional(),
give_discounts: z.string().optional(),
purchase_rates_tax_inclusive: z.string().optional(),
receive_discounts: z.string().optional(),
sell_rates_tax_inclusive: z.enum(SellRatesTaxInclusive).optional(),
give_discounts: z.enum(GiveDiscounts).optional(),
purchase_rates_tax_inclusive: z.enum(PurchaseRatesTaxInclusive).optional(),
receive_discounts: z.enum(ReceiveDiscounts).optional(),
})
export type GeneralPreferencesFormValues = z.infer<typeof generalPreferencesFormSchema>

View File

@ -23,10 +23,10 @@ const DISCOUNT_OPTIONS = DiscountType.map((v) => ({
}))
const DEFAULT_VALUES: GeneralPreferencesFormValues = {
sell_rates_tax_inclusive: "",
give_discounts: "",
purchase_rates_tax_inclusive: "",
receive_discounts: "",
sell_rates_tax_inclusive: undefined,
give_discounts: undefined,
purchase_rates_tax_inclusive: undefined,
receive_discounts: undefined,
}
export function GeneralPreferencesForm() {

View File

@ -23,8 +23,8 @@ const DISCOUNT_OPTIONS = DiscountType.map((v) => ({
}))
const DEFAULT_VALUES: PurchaseConfigFormValues = {
purchase_rates_tax_inclusive: "",
receive_discounts: "",
purchase_rates_tax_inclusive: undefined,
receive_discounts: undefined,
}
export function PurchaseConfigForm() {

View File

@ -23,8 +23,8 @@ const DISCOUNT_OPTIONS = DiscountType.map((v) => ({
}))
const DEFAULT_VALUES: SalesConfigFormValues = {
sell_rates_tax_inclusive: "",
give_discounts: "",
sell_rates_tax_inclusive: undefined,
give_discounts: undefined,
}
export function SalesConfigForm() {

View File

@ -1,8 +1,9 @@
import { z } from "zod"
import { AssignmentType } from "@garage/api"
export const departmentFormSchema = z.object({
name: z.string().min(1, "Name is required"),
assignment_type: z.string().optional(),
name: z.string().min(1, "Name is required").max(50, "Name cannot exceed 50 characters"),
assignment_type: z.enum(AssignmentType, { error: "Assignment type is required" }),
})
export type DepartmentFormValues = z.infer<typeof departmentFormSchema>

View File

@ -22,18 +22,21 @@ export type InsuranceTypeFormProps = {
const DEFAULT_VALUES: InsuranceTypeFormValues = {
title: "",
description: "",
}
function mapToFormValues(data: unknown): InsuranceTypeFormValues {
const d = (data as any)?.data ?? data ?? {}
return {
title: d.title ?? d.name ?? "",
description: d.description ?? "",
}
}
function mapFormToPayload(values: InsuranceTypeFormValues) {
return {
title: values.title,
description: values.description,
}
}
@ -90,6 +93,13 @@ export function InsuranceTypeForm({ resourceId, initialData, onSuccess }: Insura
required
/>
<RhfTextField
name="description"
label="Description"
placeholder="Short description"
required
/>
<Button type="submit" variant="default" disabled={isPending}>
{isEditing ? <Save /> : <Plus />}
{isPending

View File

@ -1,7 +1,8 @@
import { z } from "zod"
export const insuranceTypeFormSchema = z.object({
title: z.string().min(1, "Title is required"),
title: z.string().min(1, "Title is required").max(255, "Title cannot exceed 255 characters"),
description: z.string().min(1, "Description is required").max(255, "Description cannot exceed 255 characters"),
})
export type InsuranceTypeFormValues = z.infer<typeof insuranceTypeFormSchema>

View File

@ -3,13 +3,19 @@ import { z } from "zod"
const relationFieldSchema = z.object({ value: z.string(), label: z.string() }).nullable()
export const makeAndModelFormSchema = z.object({
make: z.string().min(1, "Make is required"),
model: z.string().min(1, "Model is required"),
year: z.string().optional(),
make: z.string().min(1, "Make is required").max(255, "Make cannot exceed 255 characters"),
model: z.string().min(1, "Model is required").max(255, "Model cannot exceed 255 characters"),
year: z
.string()
.min(1, "Year is required")
.refine((v) => {
const n = Number(v)
return Number.isInteger(n) && n >= 1900 && n <= 2100
}, "Year must be a whole number between 1900 and 2100"),
sub_model: z.string().optional(),
engine_size: z.string().optional(),
drivetrain: z.string().optional(),
shop_type: relationFieldSchema,
shop_type: relationFieldSchema.refine((val) => !!val?.value, "Shop type is required"),
body_type: relationFieldSchema,
fuel_type: relationFieldSchema,
transmission: relationFieldSchema,

View File

@ -1,9 +1,9 @@
import { z } from "zod"
export const shopTypeFormSchema = z.object({
title: z.string().min(1, "Title is required"),
shop_type: z.string().optional(),
note: z.string().optional(),
title: z.string().min(1, "Title is required").max(50, "Title cannot exceed 50 characters"),
shop_type: z.string().min(1, "Shop type is required"),
note: z.string().min(1, "Note is required").max(255, "Note cannot exceed 255 characters"),
is_default: z.boolean().optional(),
inspection: z.any().optional(),
image: z.any().optional(),

View File

@ -1,15 +1,21 @@
import { z } from "zod"
const timeRegex = /^([01]\d|2[0-3]):([0-5]\d):([0-5]\d)$/
const optionalTimeFormat = z
.string()
.optional()
.refine((v) => !v || timeRegex.test(v), "Time must be in HH:MM:SS format")
const shopTimingFormSchema = z.object({
title: z.string().min(1, "Title is required"),
in_time: z.string().min(1, "In time is required"),
out_time: z.string().min(1, "Out time is required"),
full_day_hours: z.string().optional(),
half_day_hours: z.string().optional(),
punch_in: z.string().optional(),
punch_out: z.string().optional(),
before_time: z.string().optional(),
after_time: z.string().optional(),
title: z.string().min(1, "Title is required").max(255, "Title cannot exceed 255 characters"),
in_time: z.string().min(1, "In time is required").regex(timeRegex, "In time must be in HH:MM:SS format"),
out_time: z.string().min(1, "Out time is required").regex(timeRegex, "Out time must be in HH:MM:SS format"),
full_day_hours: optionalTimeFormat,
half_day_hours: optionalTimeFormat,
punch_in: optionalTimeFormat,
punch_out: optionalTimeFormat,
before_time: optionalTimeFormat,
after_time: optionalTimeFormat,
is_default: z.boolean().default(false),
})

View File

@ -14,8 +14,11 @@ import { useEffect } from "react"
// ── Schema ──
const taskSectionSchema = z.object({
title: z.string().min(1, "Title is required"),
arrangement: z.string().optional(),
title: z.string().min(1, "Title is required").max(255, "Title cannot exceed 255 characters"),
arrangement: z
.string()
.min(1, "Arrangement is required")
.refine((v) => /^-?\d+$/.test(v), "Arrangement must be an integer"),
is_default: z.boolean().optional(),
})

View File

@ -5,10 +5,10 @@ export const relationFieldSchema = z
.nullable()
export const taskFormSchema = z.object({
subject: z.string().min(1, "Subject is required"),
subject: z.string().min(1, "Subject is required").max(255, "Subject cannot exceed 255 characters"),
description: z.string().optional(),
task_type: relationFieldSchema,
task_section: relationFieldSchema,
task_type: relationFieldSchema.refine((val) => !!val?.value, "Task type is required"),
task_section: relationFieldSchema.refine((val) => !!val?.value, "Task section is required"),
owner: relationFieldSchema,
department: relationFieldSchema,
priority: z.string().optional(),

View File

@ -1,7 +1,7 @@
"use client"
import { z } from "zod"
import { useForm } from "react-hook-form"
import { useForm, useFormContext } from "react-hook-form"
import { zodResolver } from "@hookform/resolvers/zod"
import { Plus } from "lucide-react"
import { Button } from "@/shared/components/ui/button"
@ -11,13 +11,14 @@ import { toast } from "sonner"
import { useAuthApi } from "@/shared/useApi"
const schema = z.object({
title: z.string().min(1, "Title is required"),
title: z.string().min(1, "Title is required").max(50, "Title cannot exceed 50 characters"),
})
type FormValues = z.infer<typeof schema>
export function BodyTypeInlineForm({ onSuccess }: InlineCreateFormProps) {
const api = useAuthApi()
const parentForm = useFormContext()
const form = useForm<FormValues>({
resolver: zodResolver(schema),
@ -25,8 +26,16 @@ export function BodyTypeInlineForm({ onSuccess }: InlineCreateFormProps) {
})
const handleSubmit = async (values: FormValues) => {
const parentShopType = parentForm?.getValues("shop_type_id") as { value?: string } | null | undefined
const shopTypeId = parentShopType?.value ? Number(parentShopType.value) : undefined
if (!shopTypeId) {
toast.error("Select a shop type before creating a body type")
return
}
try {
const result = await api.vehicleAttributes.createBodyType({ title: values.title })
const result = await api.vehicleAttributes.createBodyType({ title: values.title, shop_type_id: shopTypeId } as any)
toast.success("Body type created")
form.reset()
const item = (result as any)?.data ?? result

View File

@ -11,7 +11,8 @@ import { toast } from "sonner"
import { useAuthApi } from "@/shared/useApi"
const schema = z.object({
title: z.string().min(1, "Title is required"),
title: z.string().min(1, "Title is required").max(50, "Title cannot exceed 50 characters"),
code: z.string().min(1, "Code is required").max(50, "Code cannot exceed 50 characters"),
})
type FormValues = z.infer<typeof schema>
@ -21,12 +22,12 @@ export function ColorInlineForm({ onSuccess }: InlineCreateFormProps) {
const form = useForm<FormValues>({
resolver: zodResolver(schema),
defaultValues: { title: "" },
defaultValues: { title: "", code: "" },
})
const handleSubmit = async (values: FormValues) => {
try {
const result = await api.vehicleAttributes.createColor({ title: values.title })
const result = await api.vehicleAttributes.createColor({ title: values.title, code: values.code } as any)
toast.success("Color created")
form.reset()
const item = (result as any)?.data ?? result
@ -45,6 +46,12 @@ export function ColorInlineForm({ onSuccess }: InlineCreateFormProps) {
placeholder="e.g. Black"
required
/>
<RhfTextField
name="code"
label="Code"
placeholder="e.g. #000000"
required
/>
<Button type="submit" disabled={form.formState.isSubmitting}>
<Plus />
{form.formState.isSubmitting ? "Creating..." : "Create Color"}

View File

@ -18,9 +18,9 @@ import { toast } from "sonner"
import { useAuthApi } from "@/shared/useApi"
const schema = z.object({
title: z.string().min(1, "Title is required"),
shop_type: z.string().optional(),
note: z.string().optional(),
title: z.string().min(1, "Title is required").max(50, "Title cannot exceed 50 characters"),
shop_type: z.string().min(1, "Shop type is required"),
note: z.string().min(1, "Note is required").max(255, "Note cannot exceed 255 characters"),
is_default: z.boolean().optional(),
inspection: z.any().optional(),
image: z.any().optional(),
@ -28,8 +28,8 @@ const schema = z.object({
type FormValues = {
title: string
shop_type?: string
note?: string
shop_type: string
note: string
is_default?: boolean
inspection?: File | null
image?: File | null

View File

@ -3,7 +3,7 @@ import { z } from "zod"
const relationFieldSchema = z.object({ value: z.string(), label: z.string() }).nullable()
export const vehicleDocumentFormSchema = z.object({
document_type: relationFieldSchema,
document_type: relationFieldSchema.refine((val) => !!val?.value, "Document type is required"),
customer: relationFieldSchema,
document_number: z.string().optional(),
document_expire: z.string().optional(),

View File

@ -5,11 +5,11 @@ const relationFieldSchema = z
.nullable()
const vendorCreditFormSchema = z.object({
vendor: relationFieldSchema,
vendor: relationFieldSchema.refine((val) => !!val?.value, "Vendor is required"),
bill: relationFieldSchema,
department: relationFieldSchema,
subject: z.string().min(1, "Subject is required"),
vendor_credit_date: z.string().optional(),
vendor_credit_date: z.string().min(1, "Vendor credit date is required"),
status: z.string().optional(),
discount: z.string().optional(),
notes: z.string().optional(),