garage-erp/apps/dashboard/modules/estimates/estimate-general-info.tsx
2026-04-06 02:32:47 +03:00

198 lines
6.9 KiB
TypeScript

import {
FileText,
Hash,
Calendar,
User,
Car,
Building2,
Shield,
Tag,
MessageSquare,
} from "lucide-react"
import {
Card,
CardContent,
CardHeader,
CardTitle,
} from "@/shared/components/ui/card"
import { Badge } from "@/shared/components/ui/badge"
type EstimateData = {
id?: number
title?: string
estimate_number?: string
date?: string
customer_id?: number
vehicle_id?: number
department_id?: number
has_insurance?: boolean
enable_digital_authorisation?: boolean
insurance_type_id?: string | number | null
insurer_id?: string | number | null
service_writer_id?: number
footer?: string | null
created_at?: string
updated_at?: string
labels?: {
id?: number
title?: string
color_code?: string
}[]
customer_remarks?: {
id?: number
remark?: string
created_at?: string
}[]
// Joined fields that may come from the API
customer_name?: string
vehicle_name?: string
department_name?: string
}
type EstimateGeneralInfoProps = {
estimate: EstimateData
}
function InfoItem({
icon: Icon,
label,
value,
}: {
icon: React.ComponentType<{ className?: string }>
label: string
value?: string | null
}) {
return (
<div className="flex items-start gap-3">
<div className="flex size-9 shrink-0 items-center justify-center rounded-lg bg-muted text-muted-foreground">
<Icon className="size-4" />
</div>
<div className="flex flex-col gap-0.5">
<span className="text-xs text-muted-foreground">{label}</span>
<span className="text-sm font-medium">
{value || <span className="text-muted-foreground"></span>}
</span>
</div>
</div>
)
}
export function EstimateGeneralInfo({ estimate }: EstimateGeneralInfoProps) {
return (
<div className="grid gap-6 md:grid-cols-2">
<Card>
<CardHeader>
<CardTitle className="flex items-center gap-2">
<FileText className="size-4" /> Estimate Details
</CardTitle>
</CardHeader>
<CardContent className="grid gap-4">
<div className="grid gap-4 sm:grid-cols-2">
<InfoItem icon={FileText} label="Title" value={estimate.title} />
<InfoItem icon={Hash} label="Estimate #" value={estimate.estimate_number} />
</div>
<div className="grid gap-4 sm:grid-cols-2">
<InfoItem icon={Calendar} label="Date" value={estimate.date} />
<InfoItem
icon={Shield}
label="Insurance"
value={estimate.has_insurance ? "Yes" : "No"}
/>
</div>
<InfoItem
icon={Calendar}
label="Created"
value={
estimate.created_at
? new Date(estimate.created_at).toLocaleDateString()
: null
}
/>
</CardContent>
</Card>
<Card>
<CardHeader>
<CardTitle className="flex items-center gap-2">
<User className="size-4" /> Related Records
</CardTitle>
</CardHeader>
<CardContent className="grid gap-4">
<InfoItem
icon={User}
label="Customer"
value={estimate.customer_name || (estimate.customer_id ? `#${estimate.customer_id}` : null)}
/>
<InfoItem
icon={Car}
label="Vehicle"
value={estimate.vehicle_name || (estimate.vehicle_id ? `#${estimate.vehicle_id}` : null)}
/>
<InfoItem
icon={Building2}
label="Department"
value={estimate.department_name || (estimate.department_id ? `#${estimate.department_id}` : null)}
/>
</CardContent>
</Card>
{estimate.labels && estimate.labels.length > 0 && (
<Card>
<CardHeader>
<CardTitle className="flex items-center gap-2">
<Tag className="size-4" /> Labels
</CardTitle>
</CardHeader>
<CardContent>
<div className="flex flex-wrap gap-2">
{estimate.labels.map((label) => (
<Badge
key={label.id}
style={label.color_code ? { backgroundColor: label.color_code } : undefined}
>
{label.title}
</Badge>
))}
</div>
</CardContent>
</Card>
)}
{estimate.customer_remarks && estimate.customer_remarks.length > 0 && (
<Card>
<CardHeader>
<CardTitle className="flex items-center gap-2">
<MessageSquare className="size-4" /> Customer Remarks
</CardTitle>
</CardHeader>
<CardContent className="grid gap-3">
{estimate.customer_remarks.map((remark) => (
<div key={remark.id} className="rounded-md border p-3 text-sm">
<p>{remark.remark}</p>
{remark.created_at && (
<p className="mt-1 text-xs text-muted-foreground">
{new Date(remark.created_at).toLocaleDateString()}
</p>
)}
</div>
))}
</CardContent>
</Card>
)}
{estimate.footer && (
<Card className="md:col-span-2">
<CardHeader>
<CardTitle className="flex items-center gap-2">
<FileText className="size-4" /> Footer
</CardTitle>
</CardHeader>
<CardContent>
<p className="text-sm">{estimate.footer}</p>
</CardContent>
</Card>
)}
</div>
)
}