2026-04-24 14:36:34 +02:00
|
|
|
|
'use client';
|
|
|
|
|
|
|
|
|
|
|
|
import { useState } from 'react';
|
|
|
|
|
|
import Link from 'next/link';
|
|
|
|
|
|
import { useParams } from 'next/navigation';
|
|
|
|
|
|
import { Plus } from 'lucide-react';
|
|
|
|
|
|
|
|
|
|
|
|
import { Button } from '@/components/ui/button';
|
|
|
|
|
|
import {
|
|
|
|
|
|
Table,
|
|
|
|
|
|
TableHeader,
|
|
|
|
|
|
TableBody,
|
|
|
|
|
|
TableCell,
|
|
|
|
|
|
TableHead,
|
|
|
|
|
|
TableRow,
|
|
|
|
|
|
} from '@/components/ui/table';
|
|
|
|
|
|
import { EmptyState } from '@/components/shared/empty-state';
|
|
|
|
|
|
import { PermissionGate } from '@/components/shared/permission-gate';
|
|
|
|
|
|
import { YachtForm } from '@/components/yachts/yacht-form';
|
|
|
|
|
|
|
|
|
|
|
|
interface ClientYachtsTabProps {
|
|
|
|
|
|
clientId: string;
|
|
|
|
|
|
yachts: Array<{
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
name: string;
|
|
|
|
|
|
hullNumber: string | null;
|
|
|
|
|
|
registration: string | null;
|
|
|
|
|
|
lengthFt: string | null;
|
|
|
|
|
|
widthFt: string | null;
|
|
|
|
|
|
status: string;
|
|
|
|
|
|
}>;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function ClientYachtsTab({ clientId: _clientId, yachts }: ClientYachtsTabProps) {
|
|
|
|
|
|
const routeParams = useParams<{ portSlug: string }>();
|
|
|
|
|
|
const portSlug = routeParams?.portSlug ?? '';
|
|
|
|
|
|
const [createOpen, setCreateOpen] = useState(false);
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div className="space-y-4">
|
|
|
|
|
|
<div className="flex items-center justify-between">
|
|
|
|
|
|
<h3 className="text-sm font-medium">Client-owned yachts</h3>
|
|
|
|
|
|
<PermissionGate resource="yachts" action="create">
|
|
|
|
|
|
<Button size="sm" onClick={() => setCreateOpen(true)}>
|
|
|
|
|
|
<Plus className="mr-1.5 h-4 w-4" />
|
|
|
|
|
|
Add yacht
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</PermissionGate>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
{yachts.length === 0 ? (
|
fix(ux): popover collision padding, PWA manifest, webhook toasts, portal toast, dashboard error boundary, GDPR poll backoff, empty-state CTA
Grab-bag of UX gaps from audit-pass-#2 + #3. Each one is a small,
focused fix; bundled because they touch different surfaces.
- Popover: collisionPadding={16} + responsive
w-[min(calc(100vw-2rem),18rem)] so popovers stop clipping past the
viewport on iPhone 12 portrait.
- public/manifest.json (was missing) + manifest reference in
layout.tsx — PWA installability now works; icons (192/512/512-
maskable) were already present.
- Admin webhooks page: 4 silent `// ignore` catches in load/delete/
toggle/regenerate replaced with toast.error / toast.success. Users
no longer see a stale list with no feedback when an op fails.
- Portal document-download button: blocking alert() → toast.error().
- src/app/(dashboard)/error.tsx: branded error boundary with retry +
back-to-dashboard, replacing Next.js's default uncaught-error UI.
- GDPR export modal: refetchInterval was a flat 5s while the modal was
open. Switched to a function that only polls (every 15s) when a job
is actually pending/building; settled exports stop polling entirely.
- client-yachts-tab empty state gains a CTA wired to the existing
Add-yacht dialog, instead of just saying "No yachts".
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-06 14:59:27 +02:00
|
|
|
|
<EmptyState
|
|
|
|
|
|
title="No yachts yet"
|
|
|
|
|
|
description="Track every yacht this client owns or charters here. Linked yachts pre-fill EOIs and surface in the recommender."
|
|
|
|
|
|
action={{ label: 'Add yacht', onClick: () => setCreateOpen(true) }}
|
|
|
|
|
|
/>
|
2026-04-24 14:36:34 +02:00
|
|
|
|
) : (
|
|
|
|
|
|
<div className="rounded-md border">
|
|
|
|
|
|
<Table>
|
|
|
|
|
|
<TableHeader>
|
|
|
|
|
|
<TableRow>
|
|
|
|
|
|
<TableHead>Name</TableHead>
|
|
|
|
|
|
<TableHead>Dimensions</TableHead>
|
|
|
|
|
|
<TableHead>Hull Number</TableHead>
|
|
|
|
|
|
<TableHead>Status</TableHead>
|
|
|
|
|
|
</TableRow>
|
|
|
|
|
|
</TableHeader>
|
|
|
|
|
|
<TableBody>
|
|
|
|
|
|
{yachts.map((y) => (
|
|
|
|
|
|
<TableRow key={y.id}>
|
|
|
|
|
|
<TableCell>
|
|
|
|
|
|
<Link
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
|
|
href={`/${portSlug}/yachts/${y.id}` as any}
|
|
|
|
|
|
className="text-primary hover:underline"
|
|
|
|
|
|
>
|
|
|
|
|
|
{y.name}
|
|
|
|
|
|
</Link>
|
|
|
|
|
|
</TableCell>
|
|
|
|
|
|
<TableCell>
|
2026-05-04 22:57:01 +02:00
|
|
|
|
{y.lengthFt && y.widthFt ? `${y.lengthFt} × ${y.widthFt} ft` : '-'}
|
2026-04-24 14:36:34 +02:00
|
|
|
|
</TableCell>
|
2026-05-04 22:57:01 +02:00
|
|
|
|
<TableCell>{y.hullNumber ?? '-'}</TableCell>
|
2026-04-24 14:36:34 +02:00
|
|
|
|
<TableCell className="capitalize">{y.status.replace('_', ' ')}</TableCell>
|
|
|
|
|
|
</TableRow>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</TableBody>
|
|
|
|
|
|
</Table>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
{/*
|
|
|
|
|
|
TODO: YachtForm (Task 5.2) does not yet accept a preset owner prop.
|
|
|
|
|
|
When opened here, the user must manually pick this client in the owner
|
|
|
|
|
|
picker. Wire an `initialOwner` prop into YachtForm in a follow-up so
|
|
|
|
|
|
we can pre-select `{ type: 'client', id: clientId }`.
|
|
|
|
|
|
*/}
|
|
|
|
|
|
{createOpen && <YachtForm open={createOpen} onOpenChange={setCreateOpen} />}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|