Action buttons in entity detail headers (Invite/GDPR/Archive on clients, similar sets elsewhere) overflowed off-screen at 393px because the actions row was flex without flex-wrap. Adds flex-wrap so buttons drop to a second/third row instead of clipping. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
209 lines
6.3 KiB
TypeScript
209 lines
6.3 KiB
TypeScript
'use client';
|
|
|
|
import { useState } from 'react';
|
|
import { Pencil, RefreshCw } from 'lucide-react';
|
|
import { useQueryClient } from '@tanstack/react-query';
|
|
import { toast } from 'sonner';
|
|
import { useForm } from 'react-hook-form';
|
|
import { zodResolver } from '@hookform/resolvers/zod';
|
|
|
|
import { Button } from '@/components/ui/button';
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
DialogFooter,
|
|
} from '@/components/ui/dialog';
|
|
import { Label } from '@/components/ui/label';
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectItem,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from '@/components/ui/select';
|
|
import { Textarea } from '@/components/ui/textarea';
|
|
import { DetailHeaderStrip } from '@/components/shared/detail-header-strip';
|
|
import { PermissionGate } from '@/components/shared/permission-gate';
|
|
import { BerthForm } from './berth-form';
|
|
import { apiFetch } from '@/lib/api/client';
|
|
import { updateBerthStatusSchema, type UpdateBerthStatusInput } from '@/lib/validators/berths';
|
|
import { BERTH_STATUSES } from '@/lib/constants';
|
|
|
|
type BerthDetailData = {
|
|
id: string;
|
|
mooringNumber: string;
|
|
area: string | null;
|
|
status: string;
|
|
portId: string;
|
|
lengthFt: string | null;
|
|
lengthM: string | null;
|
|
widthFt: string | null;
|
|
widthM: string | null;
|
|
draftFt: string | null;
|
|
draftM: string | null;
|
|
widthIsMinimum: boolean | null;
|
|
price: string | null;
|
|
priceCurrency: string;
|
|
tenureType: string;
|
|
tenureYears: number | null;
|
|
tenureStartDate: string | null;
|
|
tenureEndDate: string | null;
|
|
powerCapacity: string | null;
|
|
voltage: string | null;
|
|
mooringType: string | null;
|
|
access: string | null;
|
|
berthApproved: boolean | null;
|
|
tags: Array<{ id: string; name: string; color: string }>;
|
|
};
|
|
|
|
interface BerthDetailHeaderProps {
|
|
berth: BerthDetailData;
|
|
}
|
|
|
|
const STATUS_COLORS: Record<string, string> = {
|
|
available: 'bg-green-100 text-green-800 border-green-300',
|
|
under_offer: 'bg-yellow-100 text-yellow-800 border-yellow-300',
|
|
sold: 'bg-red-100 text-red-800 border-red-300',
|
|
};
|
|
|
|
const STATUS_LABELS: Record<string, string> = {
|
|
available: 'Available',
|
|
under_offer: 'Under Offer',
|
|
sold: 'Sold',
|
|
};
|
|
|
|
function StatusChangeDialog({
|
|
berthId,
|
|
currentStatus,
|
|
open,
|
|
onOpenChange,
|
|
}: {
|
|
berthId: string;
|
|
currentStatus: string;
|
|
open: boolean;
|
|
onOpenChange: (open: boolean) => void;
|
|
}) {
|
|
const queryClient = useQueryClient();
|
|
const {
|
|
register,
|
|
handleSubmit,
|
|
setValue,
|
|
watch,
|
|
reset,
|
|
formState: { isSubmitting },
|
|
} = useForm<UpdateBerthStatusInput>({
|
|
resolver: zodResolver(updateBerthStatusSchema),
|
|
defaultValues: { status: currentStatus as (typeof BERTH_STATUSES)[number], reason: '' },
|
|
});
|
|
|
|
const status = watch('status');
|
|
|
|
async function onSubmit(data: UpdateBerthStatusInput) {
|
|
try {
|
|
await apiFetch(`/api/v1/berths/${berthId}/status`, {
|
|
method: 'PATCH',
|
|
body: data,
|
|
});
|
|
queryClient.invalidateQueries({ queryKey: ['berths'] });
|
|
queryClient.invalidateQueries({ queryKey: ['berth', berthId] });
|
|
toast.success('Status updated');
|
|
reset();
|
|
onOpenChange(false);
|
|
} catch (err: unknown) {
|
|
const message = err instanceof Error ? err.message : 'Failed to update status';
|
|
toast.error(message);
|
|
}
|
|
}
|
|
|
|
return (
|
|
<Dialog open={open} onOpenChange={onOpenChange}>
|
|
<DialogContent className="sm:max-w-md">
|
|
<DialogHeader>
|
|
<DialogTitle>Change Status</DialogTitle>
|
|
</DialogHeader>
|
|
<form onSubmit={handleSubmit(onSubmit)} className="space-y-4">
|
|
<div className="space-y-2">
|
|
<Label>New Status</Label>
|
|
<Select
|
|
value={status}
|
|
onValueChange={(v) => setValue('status', v as (typeof BERTH_STATUSES)[number])}
|
|
>
|
|
<SelectTrigger>
|
|
<SelectValue />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
{BERTH_STATUSES.map((s) => (
|
|
<SelectItem key={s} value={s}>
|
|
{STATUS_LABELS[s]}
|
|
</SelectItem>
|
|
))}
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
<div className="space-y-2">
|
|
<Label>Reason *</Label>
|
|
<Textarea {...register('reason')} placeholder="Reason for status change..." rows={3} />
|
|
</div>
|
|
<DialogFooter>
|
|
<Button type="button" variant="outline" onClick={() => onOpenChange(false)}>
|
|
Cancel
|
|
</Button>
|
|
<Button type="submit" disabled={isSubmitting}>
|
|
{isSubmitting ? 'Saving...' : 'Update Status'}
|
|
</Button>
|
|
</DialogFooter>
|
|
</form>
|
|
</DialogContent>
|
|
</Dialog>
|
|
);
|
|
}
|
|
|
|
export function BerthDetailHeader({ berth }: BerthDetailHeaderProps) {
|
|
const [editOpen, setEditOpen] = useState(false);
|
|
const [statusOpen, setStatusOpen] = useState(false);
|
|
|
|
return (
|
|
<>
|
|
<DetailHeaderStrip>
|
|
<div className="flex items-start gap-4">
|
|
<div className="flex-1 min-w-0">
|
|
<div className="flex items-center gap-3 flex-wrap">
|
|
<h1 className="text-2xl font-bold text-foreground">Berth {berth.mooringNumber}</h1>
|
|
<span
|
|
className={`inline-flex items-center rounded-full border px-3 py-1 text-sm font-medium ${STATUS_COLORS[berth.status] ?? 'bg-muted text-muted-foreground border-muted'}`}
|
|
>
|
|
{STATUS_LABELS[berth.status] ?? berth.status}
|
|
</span>
|
|
</div>
|
|
{berth.area && <p className="text-muted-foreground mt-1">{berth.area}</p>}
|
|
</div>
|
|
|
|
<div className="flex flex-wrap items-center gap-2 shrink-0">
|
|
<PermissionGate resource="berths" action="edit">
|
|
<Button variant="outline" size="sm" onClick={() => setStatusOpen(true)}>
|
|
<RefreshCw className="mr-1.5 h-4 w-4" />
|
|
Change Status
|
|
</Button>
|
|
<Button size="sm" onClick={() => setEditOpen(true)}>
|
|
<Pencil className="mr-1.5 h-4 w-4" />
|
|
Edit
|
|
</Button>
|
|
</PermissionGate>
|
|
</div>
|
|
</div>
|
|
</DetailHeaderStrip>
|
|
|
|
<BerthForm berth={berth} open={editOpen} onOpenChange={setEditOpen} />
|
|
|
|
<StatusChangeDialog
|
|
berthId={berth.id}
|
|
currentStatus={berth.status}
|
|
open={statusOpen}
|
|
onOpenChange={setStatusOpen}
|
|
/>
|
|
</>
|
|
);
|
|
}
|