First slice of Phase 5 (verification + appraisal). The office queue for verifying site-entered records, per Crewing-Implementation-Spec §8.11/R11. Stacks on 4c. Behind NEXT_PUBLIC_CREWING_ENABLED. What's in - Schema: CrewActionType += RECORD_VERIFIED/RECORD_REJECTED (migration crewing_verification_actions). No model changes — SeafarerDocument/BankDetail/ EpfDetail already carry verificationStatus + verifiedById (3b/4a). - Actions (crewing/verification/actions.ts): verifyDocument (verify_site_records — MPO/Manager) and verifyBankEpf (verify_bank_epf — Accounts) set verificationStatus VERIFIED/REJECTED + verifiedById; rejection requires remarks; each writes a CrewAction. Already-decided records are guarded. - Screen: /crewing/verification — role-aware (MPO: pending documents with expiry flags; Accounts: pending bank/EPF), Verify / Reject-with-remarks. Leave is not here (Manager approval, R11). Verification added to nav (MPO + Accounts + SU, §7). Tests & docs - Integration: verification.test.ts (6) — doc verify/reject + already-decided guard, bank/EPF verify, permission gating (Accounts can't verify docs, MPO can't verify bank/EPF). type-check clean; full unit (241) + integration (201) green (verified with RESEND_API_KEY unset, mimicking CI). - CLAUDE.md updated. Deferred (per decision): PPE / next-of-kin verification gates. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
139 lines
8.3 KiB
TypeScript
139 lines
8.3 KiB
TypeScript
"use client";
|
|
|
|
import { useState } from "react";
|
|
import { useRouter } from "next/navigation";
|
|
import type { SeafarerDocType } from "@prisma/client";
|
|
import { AdminDialog } from "@/components/ui/admin-dialog";
|
|
import { verifyDocument, verifyBankEpf } from "./actions";
|
|
|
|
const label = (s: string) => s.replace(/_/g, " ").toLowerCase().replace(/\b\w/g, (m) => m.toUpperCase());
|
|
const fmt = (iso: string | null) => (iso ? new Date(iso).toLocaleDateString() : "—");
|
|
const isExpired = (iso: string | null) => Boolean(iso && new Date(iso) < new Date());
|
|
|
|
type Doc = { id: string; crewName: string; location: string; docType: SeafarerDocType; number: string | null; expiryDate: string | null; submitted: string };
|
|
type Bank = { crewMemberId: string; crewName: string; accountName: string | null; accountNumber: string | null; ifsc: string | null; bankName: string | null };
|
|
type Epf = { crewMemberId: string; crewName: string; uan: string | null; aadhaarLast4: string | null; pfNumber: string | null };
|
|
|
|
function Actions({ onVerify, onReject }: { onVerify: () => Promise<{ ok: true } | { error: string }>; onReject: (reason: string) => Promise<{ ok: true } | { error: string }> }) {
|
|
const router = useRouter();
|
|
const [pending, setPending] = useState(false);
|
|
const [error, setError] = useState("");
|
|
const [open, setOpen] = useState(false);
|
|
const [reason, setReason] = useState("");
|
|
|
|
async function verify() {
|
|
setPending(true); setError("");
|
|
const res = await onVerify();
|
|
setPending(false);
|
|
if ("error" in res) setError(res.error); else router.refresh();
|
|
}
|
|
async function reject(e: React.FormEvent) {
|
|
e.preventDefault();
|
|
setPending(true); setError("");
|
|
const res = await onReject(reason);
|
|
setPending(false);
|
|
if ("error" in res) setError(res.error); else { setOpen(false); router.refresh(); }
|
|
}
|
|
|
|
return (
|
|
<div className="text-right">
|
|
<div className="flex justify-end gap-2">
|
|
<button onClick={verify} disabled={pending} className="rounded-md bg-primary-600 px-3 py-1.5 text-xs font-semibold text-white hover:bg-primary-700 disabled:opacity-60">Verify</button>
|
|
<button onClick={() => setOpen(true)} disabled={pending} className="rounded-md border border-neutral-300 px-3 py-1.5 text-xs font-medium text-neutral-700 hover:bg-neutral-50">Reject</button>
|
|
</div>
|
|
{error && <p className="text-xs text-danger-700 mt-1">{error}</p>}
|
|
<AdminDialog title="Reject record" open={open} onClose={() => setOpen(false)}>
|
|
<form onSubmit={reject} className="space-y-4 text-left">
|
|
<textarea className="w-full rounded-lg border border-neutral-300 px-3 py-2 text-sm" rows={3} value={reason} onChange={(e) => setReason(e.target.value)} required placeholder="Reason for rejection" />
|
|
<div className="flex justify-end gap-3">
|
|
<button type="button" className="rounded-lg border border-neutral-300 px-4 py-2 text-sm font-medium text-neutral-700 hover:bg-neutral-50" onClick={() => setOpen(false)}>Cancel</button>
|
|
<button type="submit" disabled={pending} className="rounded-lg bg-primary-600 px-4 py-2 text-sm font-semibold text-white hover:bg-primary-700 disabled:opacity-60">Reject</button>
|
|
</div>
|
|
</form>
|
|
</AdminDialog>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function Card({ title, sub, empty, children }: { title: string; sub: string; empty: boolean; children: React.ReactNode }) {
|
|
return (
|
|
<div className="mb-8">
|
|
<h2 className="text-sm font-semibold text-neutral-900">{title}</h2>
|
|
<p className="text-xs text-neutral-500 mt-0.5 mb-3">{sub}</p>
|
|
<div className="rounded-lg border border-neutral-200 bg-white overflow-hidden">
|
|
{empty ? <p className="px-4 py-10 text-center text-sm text-neutral-400">Nothing awaiting verification.</p> : (
|
|
<table className="w-full text-sm">{children}</table>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export function VerificationManager({ docs, bank, epf, canDocs, canBankEpf }: { docs: Doc[]; bank: Bank[]; epf: Epf[]; canDocs: boolean; canBankEpf: boolean }) {
|
|
return (
|
|
<div className="max-w-4xl">
|
|
<div className="mb-6">
|
|
<h1 className="text-2xl font-semibold text-neutral-900">Verification</h1>
|
|
<p className="text-sm text-neutral-500 mt-0.5">Site-entered records awaiting office verification.</p>
|
|
</div>
|
|
|
|
{canDocs && (
|
|
<Card title="Documents" sub="Verify or reject crew documents (MPO)." empty={docs.length === 0}>
|
|
<thead><tr className="border-b border-neutral-200 bg-neutral-50 text-left text-xs font-semibold text-neutral-500 uppercase tracking-wide">
|
|
<th className="px-4 py-3">Crew</th><th className="px-4 py-3">Vessel / site</th><th className="px-4 py-3">Document</th><th className="px-4 py-3">Expiry</th><th className="px-4 py-3">Submitted</th><th className="px-4 py-3 w-32"></th>
|
|
</tr></thead>
|
|
<tbody>
|
|
{docs.map((d) => (
|
|
<tr key={d.id} className="border-b border-neutral-100 last:border-0 hover:bg-neutral-50">
|
|
<td className="px-4 py-3 font-medium text-neutral-900">{d.crewName}</td>
|
|
<td className="px-4 py-3 text-neutral-600">{d.location}</td>
|
|
<td className="px-4 py-3 text-neutral-700">{label(d.docType)}{d.number ? ` · ${d.number}` : ""}</td>
|
|
<td className="px-4 py-3">{d.expiryDate ? <span className={isExpired(d.expiryDate) ? "text-danger-700 font-medium" : "text-neutral-600"}>{fmt(d.expiryDate)}{isExpired(d.expiryDate) ? " · expired" : ""}</span> : "—"}</td>
|
|
<td className="px-4 py-3 text-neutral-500">{fmt(d.submitted)}</td>
|
|
<td className="px-4 py-3"><Actions onVerify={() => verifyDocument(d.id, true)} onReject={(r) => verifyDocument(d.id, false, r)} /></td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</Card>
|
|
)}
|
|
|
|
{canBankEpf && (
|
|
<Card title="Bank details" sub="Verify or reject crew bank details (Accounts)." empty={bank.length === 0}>
|
|
<thead><tr className="border-b border-neutral-200 bg-neutral-50 text-left text-xs font-semibold text-neutral-500 uppercase tracking-wide">
|
|
<th className="px-4 py-3">Crew</th><th className="px-4 py-3">Account</th><th className="px-4 py-3">IFSC</th><th className="px-4 py-3">Bank</th><th className="px-4 py-3 w-32"></th>
|
|
</tr></thead>
|
|
<tbody>
|
|
{bank.map((b) => (
|
|
<tr key={b.crewMemberId} className="border-b border-neutral-100 last:border-0 hover:bg-neutral-50">
|
|
<td className="px-4 py-3 font-medium text-neutral-900">{b.crewName}</td>
|
|
<td className="px-4 py-3 font-mono text-xs text-neutral-700">{b.accountNumber ?? "—"}{b.accountName ? ` (${b.accountName})` : ""}</td>
|
|
<td className="px-4 py-3 text-neutral-600">{b.ifsc ?? "—"}</td>
|
|
<td className="px-4 py-3 text-neutral-600">{b.bankName ?? "—"}</td>
|
|
<td className="px-4 py-3"><Actions onVerify={() => verifyBankEpf(b.crewMemberId, "bank", true)} onReject={(r) => verifyBankEpf(b.crewMemberId, "bank", false, r)} /></td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</Card>
|
|
)}
|
|
|
|
{canBankEpf && (
|
|
<Card title="EPF details" sub="Verify or reject crew EPF / identity details (Accounts)." empty={epf.length === 0}>
|
|
<thead><tr className="border-b border-neutral-200 bg-neutral-50 text-left text-xs font-semibold text-neutral-500 uppercase tracking-wide">
|
|
<th className="px-4 py-3">Crew</th><th className="px-4 py-3">UAN</th><th className="px-4 py-3">Aadhaar</th><th className="px-4 py-3">PF no.</th><th className="px-4 py-3 w-32"></th>
|
|
</tr></thead>
|
|
<tbody>
|
|
{epf.map((e) => (
|
|
<tr key={e.crewMemberId} className="border-b border-neutral-100 last:border-0 hover:bg-neutral-50">
|
|
<td className="px-4 py-3 font-medium text-neutral-900">{e.crewName}</td>
|
|
<td className="px-4 py-3 font-mono text-xs text-neutral-700">{e.uan ?? "—"}</td>
|
|
<td className="px-4 py-3 font-mono text-xs text-neutral-700">{e.aadhaarLast4 ?? "—"}</td>
|
|
<td className="px-4 py-3 text-neutral-600">{e.pfNumber ?? "—"}</td>
|
|
<td className="px-4 py-3"><Actions onVerify={() => verifyBankEpf(e.crewMemberId, "epf", true)} onReject={(r) => verifyBankEpf(e.crewMemberId, "epf", false, r)} /></td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</Card>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|