feat(berths): split Documents tab into Spec + Deal Documents
Berth detail page now has two tabs:
- Spec: the existing versioned berth-spec PDF surface (current panel,
version history, parser badge).
- Deal Documents: NEW. Lists EOIs / contracts / etc. attached to
interests currently linked to this berth via interest_berths.
New service helper listDealDocumentsForBerth joins documents →
interests → interest_berths with a port_id guard on both sides.
GET /api/v1/berths/[id]/deal-documents wraps it, gated on berths.view.
Read-only — title, type, status badge, and an Open link to the source
interest page. Edits / sends still happen on the interest's own page.
The Spec tab paragraph now points reps to the new Deal Documents tab
instead of telling them to navigate via Interests.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
24
src/app/api/v1/berths/[id]/deal-documents/route.ts
Normal file
24
src/app/api/v1/berths/[id]/deal-documents/route.ts
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import { NextResponse } from 'next/server';
|
||||||
|
|
||||||
|
import { withAuth, withPermission } from '@/lib/api/helpers';
|
||||||
|
import { errorResponse, NotFoundError } from '@/lib/errors';
|
||||||
|
import { listDealDocumentsForBerth } from '@/lib/services/documents.service';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/v1/berths/[id]/deal-documents
|
||||||
|
*
|
||||||
|
* Lists documents attached to interests currently linked to this berth.
|
||||||
|
* Same permission gate as the berth page itself (berths.view).
|
||||||
|
*/
|
||||||
|
export const GET = withAuth(
|
||||||
|
withPermission('berths', 'view', async (_req, ctx, params) => {
|
||||||
|
try {
|
||||||
|
const berthId = params.id;
|
||||||
|
if (!berthId) throw new NotFoundError('Berth');
|
||||||
|
const data = await listDealDocumentsForBerth(ctx.portId, berthId);
|
||||||
|
return NextResponse.json({ data });
|
||||||
|
} catch (error) {
|
||||||
|
return errorResponse(error);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
);
|
||||||
90
src/components/berths/berth-deal-documents-tab.tsx
Normal file
90
src/components/berths/berth-deal-documents-tab.tsx
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import Link from 'next/link';
|
||||||
|
import { useParams } from 'next/navigation';
|
||||||
|
import { useQuery } from '@tanstack/react-query';
|
||||||
|
import { FileText, ExternalLink } from 'lucide-react';
|
||||||
|
|
||||||
|
import { apiFetch } from '@/lib/api/client';
|
||||||
|
import { Badge } from '@/components/ui/badge';
|
||||||
|
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
|
|
||||||
|
interface BerthDealDoc {
|
||||||
|
id: string;
|
||||||
|
title: string;
|
||||||
|
documentType: string;
|
||||||
|
status: string;
|
||||||
|
createdAt: string;
|
||||||
|
interestId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const STATUS_TONE: Record<string, 'default' | 'secondary' | 'outline' | 'destructive'> = {
|
||||||
|
draft: 'outline',
|
||||||
|
sent: 'secondary',
|
||||||
|
partially_signed: 'secondary',
|
||||||
|
completed: 'default',
|
||||||
|
expired: 'destructive',
|
||||||
|
cancelled: 'destructive',
|
||||||
|
};
|
||||||
|
|
||||||
|
export function BerthDealDocumentsTab({ berthId }: { berthId: string }) {
|
||||||
|
const params = useParams<{ portSlug: string }>();
|
||||||
|
const portSlug = params?.portSlug ?? '';
|
||||||
|
|
||||||
|
const { data: docs = [], isLoading } = useQuery<BerthDealDoc[]>({
|
||||||
|
queryKey: ['berth-deal-documents', berthId],
|
||||||
|
queryFn: () =>
|
||||||
|
apiFetch<{ data: BerthDealDoc[] }>(`/api/v1/berths/${berthId}/deal-documents`).then(
|
||||||
|
(r) => r.data,
|
||||||
|
),
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="space-y-4">
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
EOIs, contracts, and other deal documents attached to interests currently linked to this
|
||||||
|
berth. Read-only — to send, sign, or edit, open the document on the linked interest's
|
||||||
|
page.
|
||||||
|
</p>
|
||||||
|
<Card>
|
||||||
|
<CardHeader className="pb-3">
|
||||||
|
<CardTitle className="text-sm font-medium">Linked deal documents</CardTitle>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="pt-0">
|
||||||
|
{isLoading ? (
|
||||||
|
<p className="text-sm text-muted-foreground">Loading…</p>
|
||||||
|
) : docs.length === 0 ? (
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
No deal documents yet. Documents created on a linked interest will appear here.
|
||||||
|
</p>
|
||||||
|
) : (
|
||||||
|
<ul className="divide-y">
|
||||||
|
{docs.map((doc) => (
|
||||||
|
<li
|
||||||
|
key={doc.id}
|
||||||
|
className="flex flex-wrap items-center justify-between gap-2 py-2.5 text-sm"
|
||||||
|
>
|
||||||
|
<div className="flex min-w-0 items-center gap-2">
|
||||||
|
<FileText className="h-4 w-4 shrink-0 text-muted-foreground" />
|
||||||
|
<span className="truncate font-medium">{doc.title}</span>
|
||||||
|
<span className="text-xs text-muted-foreground">{doc.documentType}</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Badge variant={STATUS_TONE[doc.status] ?? 'outline'}>{doc.status}</Badge>
|
||||||
|
<Link
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
href={`/${portSlug}/interests/${doc.interestId}` as any}
|
||||||
|
className="inline-flex items-center gap-1 text-xs text-primary hover:underline"
|
||||||
|
>
|
||||||
|
Open <ExternalLink className="h-3 w-3" />
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
)}
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -164,8 +164,8 @@ export function BerthDocumentsTab({ berthId }: { berthId: string }) {
|
|||||||
<div className="space-y-6">
|
<div className="space-y-6">
|
||||||
<p className="text-sm text-muted-foreground">
|
<p className="text-sm text-muted-foreground">
|
||||||
Berth-spec PDF: the dimensional drawing or surveyor sheet for this slip. Versioned so a
|
Berth-spec PDF: the dimensional drawing or surveyor sheet for this slip. Versioned so a
|
||||||
misparse can be rolled back. For documents tied to a prospect on this berth (EOI, contract,
|
misparse can be rolled back. Deal documents (EOI, contract, etc.) live on the “Deal
|
||||||
etc.), open the matching deal in the Interests tab.
|
Documents” tab.
|
||||||
</p>
|
</p>
|
||||||
<Card>
|
<Card>
|
||||||
<CardHeader className="flex flex-row items-center justify-between pb-3">
|
<CardHeader className="flex flex-row items-center justify-between pb-3">
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ import { BerthReservationsTab } from './berth-reservations-tab';
|
|||||||
import { BerthInterestsTab } from './berth-interests-tab';
|
import { BerthInterestsTab } from './berth-interests-tab';
|
||||||
import { BerthInterestPulse } from './berth-interest-pulse';
|
import { BerthInterestPulse } from './berth-interest-pulse';
|
||||||
import { BerthDocumentsTab } from './berth-documents-tab';
|
import { BerthDocumentsTab } from './berth-documents-tab';
|
||||||
|
import { BerthDealDocumentsTab } from './berth-deal-documents-tab';
|
||||||
|
|
||||||
type BerthData = {
|
type BerthData = {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -413,10 +414,15 @@ export function buildBerthTabs(berth: BerthData): DetailTab[] {
|
|||||||
content: <BerthReservationsTab berthId={berth.id} />,
|
content: <BerthReservationsTab berthId={berth.id} />,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'documents',
|
id: 'spec',
|
||||||
label: 'Documents',
|
label: 'Spec',
|
||||||
content: <BerthDocumentsTab berthId={berth.id} />,
|
content: <BerthDocumentsTab berthId={berth.id} />,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
id: 'deal-documents',
|
||||||
|
label: 'Deal Documents',
|
||||||
|
content: <BerthDealDocumentsTab berthId={berth.id} />,
|
||||||
|
},
|
||||||
// Waiting List + Maintenance Log tabs were stubs ("coming soon")
|
// Waiting List + Maintenance Log tabs were stubs ("coming soon")
|
||||||
// visible to every operator. Hidden here until the
|
// visible to every operator. Hidden here until the
|
||||||
// berth_waiting_list / berth_maintenance_log feature surfaces ship.
|
// berth_waiting_list / berth_maintenance_log feature surfaces ship.
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import {
|
|||||||
documentWatchers,
|
documentWatchers,
|
||||||
files,
|
files,
|
||||||
} from '@/lib/db/schema/documents';
|
} from '@/lib/db/schema/documents';
|
||||||
import { interests } from '@/lib/db/schema/interests';
|
import { interests, interestBerths } from '@/lib/db/schema/interests';
|
||||||
import { clients } from '@/lib/db/schema/clients';
|
import { clients } from '@/lib/db/schema/clients';
|
||||||
import { companies } from '@/lib/db/schema/companies';
|
import { companies } from '@/lib/db/schema/companies';
|
||||||
import { yachts } from '@/lib/db/schema/yachts';
|
import { yachts } from '@/lib/db/schema/yachts';
|
||||||
@@ -213,6 +213,65 @@ export async function listDocuments(
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─── Deal docs for a berth ────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
export interface BerthDealDoc {
|
||||||
|
id: string;
|
||||||
|
title: string;
|
||||||
|
documentType: string;
|
||||||
|
status: string;
|
||||||
|
createdAt: Date;
|
||||||
|
interestId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Documents attached to any interest currently linked to this berth via
|
||||||
|
* `interest_berths`. Used by the Deal Documents tab on the berth detail
|
||||||
|
* page so reps can see EOIs / contracts / etc. associated with active
|
||||||
|
* prospects on this slip without leaving the page.
|
||||||
|
*
|
||||||
|
* Read-only; visibility piggybacks on the interest tenancy (the
|
||||||
|
* permission-gate on the berth page guards entry, and we only return
|
||||||
|
* documents for interests in the same port). Edits / sends still happen
|
||||||
|
* from the interest's own page.
|
||||||
|
*/
|
||||||
|
export async function listDealDocumentsForBerth(
|
||||||
|
portId: string,
|
||||||
|
berthId: string,
|
||||||
|
): Promise<BerthDealDoc[]> {
|
||||||
|
const rows = await db
|
||||||
|
.select({
|
||||||
|
id: documents.id,
|
||||||
|
title: documents.title,
|
||||||
|
documentType: documents.documentType,
|
||||||
|
status: documents.status,
|
||||||
|
createdAt: documents.createdAt,
|
||||||
|
interestId: documents.interestId,
|
||||||
|
})
|
||||||
|
.from(documents)
|
||||||
|
.innerJoin(interestBerths, eq(interestBerths.interestId, documents.interestId))
|
||||||
|
.innerJoin(interests, eq(interests.id, documents.interestId))
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(interestBerths.berthId, berthId),
|
||||||
|
eq(documents.portId, portId),
|
||||||
|
eq(interests.portId, portId),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.orderBy(sql`${documents.createdAt} DESC`);
|
||||||
|
|
||||||
|
return rows
|
||||||
|
.filter((r): r is typeof r & { interestId: string } => Boolean(r.interestId))
|
||||||
|
.map((r) => ({
|
||||||
|
id: r.id,
|
||||||
|
title: r.title,
|
||||||
|
documentType: r.documentType,
|
||||||
|
status: r.status,
|
||||||
|
createdAt: r.createdAt,
|
||||||
|
interestId: r.interestId,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
// ─── Hub tab counts ───────────────────────────────────────────────────────────
|
// ─── Hub tab counts ───────────────────────────────────────────────────────────
|
||||||
|
|
||||||
export interface HubTabCounts {
|
export interface HubTabCounts {
|
||||||
|
|||||||
Reference in New Issue
Block a user