Files
acmcc/src/pages/RVBoatLotsPage.tsx
T
admin 308af20aa1 RV/Boat Lots: request renter insurance (vendor-style flow)
Phase 4. Mirror the vendor insurance request flow for RV/boat renters:
- Migration: insurance fields on rv_boat_lot_rentals + rv_renter_insurance_requests
  table + token-scoped lookup/submit SECURITY DEFINER RPCs (granted to anon).
- Edge fn send-rv-renter-insurance-request emails the renter a secure link
  (reuses the vendor-insurance-request email template).
- Public page /rv-insurance/:token to submit carrier/policy/expiration + COI upload.
- "Request Insurance" button on each active rental + insurance status display.

DB RPCs verified end-to-end (rolled-back txn): submit matches token, updates the
rental, marks the request submitted. Edge function deployed.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-07 20:16:31 -04:00

831 lines
46 KiB
TypeScript

import { useEffect, useMemo, useState } from "react";
import { supabase } from "@/integrations/supabase/client";
import { useAssociation } from "@/contexts/AssociationContext";
import { useAuth } from "@/contexts/AuthContext";
import { Button } from "@/components/ui/button";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
import { Input } from "@/components/ui/input";
import { Label } from "@/components/ui/label";
import { Textarea } from "@/components/ui/textarea";
import { Tabs, TabsList, TabsTrigger, TabsContent } from "@/components/ui/tabs";
import { Badge } from "@/components/ui/badge";
import { Dialog, DialogContent, DialogDescription, DialogFooter, DialogHeader, DialogTitle, DialogTrigger } from "@/components/ui/dialog";
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table";
import { Checkbox } from "@/components/ui/checkbox";
import { toast } from "sonner";
import { Plus, Trash2, Check, ArrowUp, ArrowDown, Copy, Pencil, X } from "lucide-react";
type Lot = {
id: string; association_id: string; lot_number: string; lot_type: string;
size: string | null; monthly_rate: number | null; status: string; notes: string | null;
};
type Rental = {
id: string; association_id: string; lot_id: string; renter_name: string;
renter_email: string | null; renter_phone: string | null; vehicle_description: string | null;
start_date: string; end_date: string | null; monthly_rate: number | null; status: string; notes: string | null;
is_owner?: boolean | null; user_id?: string | null; owner_id?: string | null; unit_id?: string | null;
insurance_carrier?: string | null; insurance_policy_number?: string | null;
insurance_expiration_date?: string | null; insurance_document_url?: string | null;
};
type WaitlistEntry = {
id: string; association_id: string; position: number;
requester_name: string; requester_email: string | null; requester_phone: string | null;
unit_address: string | null; requested_lot_type: string;
vehicle_description: string | null; notes: string | null;
status: string; source: string; joined_at: string;
fulfilled_at: string | null; fulfilled_lot_number: string | null;
};
export default function RVBoatLotsPage() {
const associationCtx = useAssociation() as any;
const ctxAssociation = associationCtx?.selectedAssociation ?? null;
const { user } = useAuth();
const [associations, setAssociations] = useState<{ id: string; name: string }[]>([]);
const [localAssociationId, setLocalAssociationId] = useState<string>("");
useEffect(() => {
supabase.from("associations").select("id, name").eq("status", "active").order("name").then(({ data }) => {
const list = data || [];
setAssociations(list);
setLocalAssociationId((prev) => prev || ctxAssociation?.id || list[0]?.id || "");
});
}, [ctxAssociation?.id]);
const associationId = localAssociationId || ctxAssociation?.id || "";
const [lots, setLots] = useState<Lot[]>([]);
const [rentals, setRentals] = useState<Rental[]>([]);
const [waitlist, setWaitlist] = useState<WaitlistEntry[]>([]);
const [owners, setOwners] = useState<any[]>([]);
const [units, setUnits] = useState<any[]>([]);
const [loading, setLoading] = useState(true);
// dialogs
const [lotDialogOpen, setLotDialogOpen] = useState(false);
const [rentalDialogOpen, setRentalDialogOpen] = useState(false);
const [waitlistDialogOpen, setWaitlistDialogOpen] = useState(false);
const [fulfillDialog, setFulfillDialog] = useState<{ open: boolean; entry: WaitlistEntry | null; lotNumber: string }>({ open: false, entry: null, lotNumber: "" });
// form state
const [lotForm, setLotForm] = useState({ lot_number: "", lot_type: "rv", size: "", monthly_rate: "", notes: "" });
const [rentalForm, setRentalForm] = useState({
lot_id: "", renter_name: "", renter_email: "", renter_phone: "",
vehicle_description: "", start_date: new Date().toISOString().slice(0, 10), monthly_rate: "", notes: "",
owner_id: "", unit_id: ""
});
const [waitlistForm, setWaitlistForm] = useState({
requester_name: "", requester_email: "", requester_phone: "",
unit_address: "", requested_lot_type: "either", vehicle_description: "", notes: ""
});
// batch edit state
const [selectedLotIds, setSelectedLotIds] = useState<Set<string>>(new Set());
const [lotBulkOpen, setLotBulkOpen] = useState(false);
const [savingLotBulk, setSavingLotBulk] = useState(false);
const [lotBulk, setLotBulk] = useState({ lot_type: "no_change", status: "no_change", monthly_rate: "" });
const [selectedRentalIds, setSelectedRentalIds] = useState<Set<string>>(new Set());
const [rentalBulkOpen, setRentalBulkOpen] = useState(false);
const [savingRentalBulk, setSavingRentalBulk] = useState(false);
const [rentalBulk, setRentalBulk] = useState({ status: "no_change", monthly_rate: "", is_owner: "no_change" });
const toggleSet = (setter: React.Dispatch<React.SetStateAction<Set<string>>>, id: string) =>
setter(prev => { const n = new Set(prev); n.has(id) ? n.delete(id) : n.add(id); return n; });
const toggleSetMany = (setter: React.Dispatch<React.SetStateAction<Set<string>>>, ids: string[], checked: boolean) =>
setter(prev => { const n = new Set(prev); ids.forEach(id => checked ? n.add(id) : n.delete(id)); return n; });
const publicUrl = useMemo(() => {
if (!associationId) return "";
return `${window.location.origin}/rv-boat-waitlist/${associationId}`;
}, [associationId]);
const load = async () => {
if (!associationId) return;
setLoading(true);
try {
const [lotsRes, rentalsRes, wlRes, ownersRes, unitsRes] = await Promise.all([
supabase.from("rv_boat_lots").select("*").eq("association_id", associationId).order("lot_number"),
supabase.from("rv_boat_lot_rentals").select("*").eq("association_id", associationId).order("start_date", { ascending: false }),
supabase.from("rv_boat_lot_waitlist").select("*").eq("association_id", associationId).order("position"),
supabase.from("owners").select("id, first_name, last_name, unit_id, email, phone").eq("association_id", associationId).eq("status", "active").order("last_name"),
supabase.from("units").select("id, unit_number").eq("association_id", associationId).order("unit_number"),
]);
if (lotsRes.error) throw lotsRes.error;
if (rentalsRes.error) throw rentalsRes.error;
if (wlRes.error) throw wlRes.error;
setLots(lotsRes.data || []);
setRentals(rentalsRes.data || []);
setWaitlist(wlRes.data || []);
setOwners(ownersRes.data || []);
setUnits(unitsRes.data || []);
} catch (e: any) {
toast.error(e.message || "Failed to load data");
} finally {
setLoading(false);
}
};
useEffect(() => { load(); }, [associationId]);
const createLot = async () => {
if (!associationId || !lotForm.lot_number.trim()) { toast.error("Lot number required"); return; }
const { error } = await supabase.from("rv_boat_lots").insert({
association_id: associationId,
lot_number: lotForm.lot_number.trim(),
lot_type: lotForm.lot_type,
size: lotForm.size || null,
monthly_rate: lotForm.monthly_rate ? Number(lotForm.monthly_rate) : null,
notes: lotForm.notes || null,
});
if (error) { toast.error(error.message); return; }
toast.success("Lot added");
setLotDialogOpen(false);
setLotForm({ lot_number: "", lot_type: "rv", size: "", monthly_rate: "", notes: "" });
load();
};
const deleteLot = async (id: string) => {
if (!confirm("Delete this lot?")) return;
const { error } = await supabase.from("rv_boat_lots").delete().eq("id", id);
if (error) { toast.error(error.message); return; }
toast.success("Lot deleted");
load();
};
const applyLotBulk = async () => {
const ids = [...selectedLotIds];
if (!ids.length) return;
const patch: Record<string, any> = {};
if (lotBulk.lot_type !== "no_change") patch.lot_type = lotBulk.lot_type;
if (lotBulk.status !== "no_change") patch.status = lotBulk.status;
if (lotBulk.monthly_rate.trim() !== "") patch.monthly_rate = Number(lotBulk.monthly_rate) || 0;
if (Object.keys(patch).length === 0) { toast.error("No changes selected"); return; }
setSavingLotBulk(true);
const { error } = await supabase.from("rv_boat_lots").update(patch).in("id", ids);
setSavingLotBulk(false);
if (error) { toast.error(error.message); return; }
toast.success(`Updated ${ids.length} lot${ids.length === 1 ? "" : "s"}`);
setLotBulkOpen(false); setSelectedLotIds(new Set()); load();
};
const deleteLots = async () => {
const ids = [...selectedLotIds];
if (!ids.length) return;
if (!confirm(`Delete ${ids.length} lot${ids.length === 1 ? "" : "s"}? Lots with rentals may fail to delete.`)) return;
const { error } = await supabase.from("rv_boat_lots").delete().in("id", ids);
if (error) { toast.error(error.message); return; }
toast.success(`Deleted ${ids.length} lot${ids.length === 1 ? "" : "s"}`);
setSelectedLotIds(new Set()); load();
};
const applyRentalBulk = async () => {
const ids = [...selectedRentalIds];
if (!ids.length) return;
const patch: Record<string, any> = {};
if (rentalBulk.status !== "no_change") {
patch.status = rentalBulk.status;
if (rentalBulk.status === "ended") patch.end_date = new Date().toISOString().slice(0, 10);
}
if (rentalBulk.monthly_rate.trim() !== "") patch.monthly_rate = Number(rentalBulk.monthly_rate) || 0;
if (rentalBulk.is_owner !== "no_change") patch.is_owner = rentalBulk.is_owner === "true";
if (Object.keys(patch).length === 0) { toast.error("No changes selected"); return; }
setSavingRentalBulk(true);
const { error } = await supabase.from("rv_boat_lot_rentals").update(patch).in("id", ids);
setSavingRentalBulk(false);
if (error) { toast.error(error.message); return; }
toast.success(`Updated ${ids.length} rental${ids.length === 1 ? "" : "s"}`);
setRentalBulkOpen(false); setSelectedRentalIds(new Set()); load();
};
const createRental = async () => {
if (!associationId || !rentalForm.lot_id || !rentalForm.renter_name.trim()) {
toast.error("Lot and renter name required"); return;
}
const { error } = await supabase.from("rv_boat_lot_rentals").insert({
association_id: associationId,
lot_id: rentalForm.lot_id,
renter_name: rentalForm.renter_name.trim(),
renter_email: rentalForm.renter_email || null,
renter_phone: rentalForm.renter_phone || null,
vehicle_description: rentalForm.vehicle_description || null,
start_date: rentalForm.start_date,
monthly_rate: rentalForm.monthly_rate ? Number(rentalForm.monthly_rate) : null,
notes: rentalForm.notes || null,
owner_id: rentalForm.owner_id || null,
unit_id: rentalForm.unit_id || null,
status: "active",
});
if (error) { toast.error(error.message); return; }
toast.success("Rental created");
setRentalDialogOpen(false);
setRentalForm({ lot_id: "", renter_name: "", renter_email: "", renter_phone: "", vehicle_description: "", start_date: new Date().toISOString().slice(0, 10), monthly_rate: "", notes: "", owner_id: "", unit_id: "" });
load();
};
const endRental = async (r: Rental) => {
if (!confirm("End this rental? The lot will be marked available.")) return;
const { error } = await supabase.from("rv_boat_lot_rentals")
.update({ status: "ended", end_date: new Date().toISOString().slice(0, 10) })
.eq("id", r.id);
if (error) { toast.error(error.message); return; }
toast.success("Rental ended");
load();
};
const invitePortal = async (r: Rental, asOwner = false) => {
const label = asOwner ? "owner" : "renter";
const email = window.prompt(`Email address for the ${label}'s portal account:`, r.renter_email || "");
if (!email) return;
const trimmed = email.trim();
if (!trimmed) return;
try {
const { data, error } = await supabase.functions.invoke("invite-rv-renter", {
body: { rental_id: r.id, email: trimmed, as_owner: asOwner },
});
if (error) throw error;
if ((data as any)?.error) throw new Error((data as any).error);
toast.success((data as any)?.invited ? "Invitation email sent" : "Existing user linked; password reset emailed");
load();
} catch (e: any) {
toast.error(e.message || "Failed to invite");
}
};
const requestInsurance = async (r: Rental) => {
if (!r.renter_email) { toast.error("Add a renter email first, then request insurance."); return; }
try {
const { data, error } = await supabase.functions.invoke("send-rv-renter-insurance-request", {
body: { rental_id: r.id },
});
if (error) throw error;
const res = data as any;
if (res?.error && !res?.link) throw new Error(res.error);
if (res?.ok) {
toast.success(`Insurance request emailed to ${res.sent_to}`);
} else if (res?.link) {
await navigator.clipboard.writeText(res.link).catch(() => {});
toast.message("Email didn't send — link copied", { description: "Share the submission link with the renter manually." });
}
load();
} catch (e: any) {
toast.error(e.message || "Failed to request insurance");
}
};
const toggleIsOwner = async (r: Rental) => {
const next = !r.is_owner;
const { error } = await supabase.from("rv_boat_lot_rentals")
.update({ is_owner: next })
.eq("id", r.id);
if (error) { toast.error(error.message); return; }
toast.success(next ? "Marked as owner" : "Marked as renter");
load();
};
const addWaitlist = async () => {
if (!associationId || !waitlistForm.requester_name.trim()) { toast.error("Name required"); return; }
const { error } = await supabase.from("rv_boat_lot_waitlist").insert({
association_id: associationId,
requester_name: waitlistForm.requester_name.trim(),
requester_email: waitlistForm.requester_email || null,
requester_phone: waitlistForm.requester_phone || null,
unit_address: waitlistForm.unit_address || null,
requested_lot_type: waitlistForm.requested_lot_type,
vehicle_description: waitlistForm.vehicle_description || null,
notes: waitlistForm.notes || null,
source: "manual",
status: "waiting",
position: 0,
});
if (error) { toast.error(error.message); return; }
toast.success("Added to waitlist");
setWaitlistDialogOpen(false);
setWaitlistForm({ requester_name: "", requester_email: "", requester_phone: "", unit_address: "", requested_lot_type: "either", vehicle_description: "", notes: "" });
load();
};
const movePosition = async (entry: WaitlistEntry, direction: "up" | "down") => {
const waiting = waitlist.filter(w => w.status === "waiting").sort((a, b) => a.position - b.position);
const idx = waiting.findIndex(w => w.id === entry.id);
const swapIdx = direction === "up" ? idx - 1 : idx + 1;
if (swapIdx < 0 || swapIdx >= waiting.length) return;
const other = waiting[swapIdx];
// Use temporary position to avoid unique-ish collisions if any (none enforced, but safe)
const tempPos = -Math.floor(Math.random() * 100000) - 1;
await supabase.from("rv_boat_lot_waitlist").update({ position: tempPos }).eq("id", entry.id);
await supabase.from("rv_boat_lot_waitlist").update({ position: entry.position }).eq("id", other.id);
await supabase.from("rv_boat_lot_waitlist").update({ position: other.position }).eq("id", entry.id);
load();
};
const removeWaitlist = async (id: string) => {
if (!confirm("Remove this entry from the waitlist?")) return;
const { error } = await supabase.from("rv_boat_lot_waitlist").update({ status: "removed" }).eq("id", id);
if (error) { toast.error(error.message); return; }
toast.success("Removed");
load();
};
const openFulfill = (entry: WaitlistEntry) => {
setFulfillDialog({ open: true, entry, lotNumber: "" });
};
const confirmFulfill = async () => {
const entry = fulfillDialog.entry;
if (!entry) return;
if (!fulfillDialog.lotNumber.trim()) { toast.error("Enter the lot number assigned"); return; }
const { error } = await supabase.from("rv_boat_lot_waitlist").update({
status: "fulfilled",
fulfilled_at: new Date().toISOString(),
fulfilled_lot_number: fulfillDialog.lotNumber.trim(),
fulfilled_by: user?.id || null,
}).eq("id", entry.id);
if (error) { toast.error(error.message); return; }
toast.success("Marked fulfilled");
setFulfillDialog({ open: false, entry: null, lotNumber: "" });
load();
};
const copyPublicLink = async () => {
await navigator.clipboard.writeText(publicUrl);
toast.success("Public waitlist link copied");
};
const waitingEntries = waitlist.filter(w => w.status === "waiting").sort((a, b) => a.position - b.position);
const fulfilledEntries = waitlist.filter(w => w.status === "fulfilled" || w.status === "removed");
return (
<div className="p-6 space-y-6">
<div className="flex items-center justify-between flex-wrap gap-3">
<div>
<h1 className="text-2xl font-bold">RV / Boat Lot Rentals</h1>
<p className="text-sm text-muted-foreground">Manage lot inventory, active rentals, and the waitlist.</p>
</div>
<div className="flex items-center gap-2">
<Select value={associationId} onValueChange={setLocalAssociationId}>
<SelectTrigger className="w-[260px]"><SelectValue placeholder="Select association" /></SelectTrigger>
<SelectContent>
{associations.map(a => <SelectItem key={a.id} value={a.id}>{a.name}</SelectItem>)}
</SelectContent>
</Select>
<Button variant="outline" size="sm" onClick={copyPublicLink} disabled={!associationId}>
<Copy className="h-4 w-4 mr-2" /> Copy public waitlist link
</Button>
</div>
</div>
{!associationId ? (
<p className="text-muted-foreground">Select an association to manage RV/Boat lots.</p>
) : null}
<Tabs defaultValue="waitlist" className="w-full">
<TabsList>
<TabsTrigger value="waitlist">Waitlist ({waitingEntries.length})</TabsTrigger>
<TabsTrigger value="lots">Lots ({lots.length})</TabsTrigger>
<TabsTrigger value="rentals">Active Rentals ({rentals.filter(r => r.status === "active").length})</TabsTrigger>
<TabsTrigger value="history">History</TabsTrigger>
</TabsList>
{/* Waitlist */}
<TabsContent value="waitlist" className="space-y-4">
<Card>
<CardHeader className="flex flex-row items-center justify-between">
<CardTitle>Current Waitlist</CardTitle>
<Dialog open={waitlistDialogOpen} onOpenChange={setWaitlistDialogOpen}>
<DialogTrigger asChild>
<Button size="sm"><Plus className="h-4 w-4 mr-1" /> Add to waitlist</Button>
</DialogTrigger>
<DialogContent>
<DialogHeader>
<DialogTitle>Add waitlist entry</DialogTitle>
<DialogDescription>New entries are added to the bottom of the list.</DialogDescription>
</DialogHeader>
<div className="space-y-3">
<div><Label>Name *</Label><Input value={waitlistForm.requester_name} onChange={e => setWaitlistForm({ ...waitlistForm, requester_name: e.target.value })} /></div>
<div className="grid grid-cols-2 gap-3">
<div><Label>Email</Label><Input type="email" value={waitlistForm.requester_email} onChange={e => setWaitlistForm({ ...waitlistForm, requester_email: e.target.value })} /></div>
<div><Label>Phone</Label><Input value={waitlistForm.requester_phone} onChange={e => setWaitlistForm({ ...waitlistForm, requester_phone: e.target.value })} /></div>
</div>
<div><Label>Unit / Address</Label><Input value={waitlistForm.unit_address} onChange={e => setWaitlistForm({ ...waitlistForm, unit_address: e.target.value })} /></div>
<div>
<Label>Lot type</Label>
<Select value={waitlistForm.requested_lot_type} onValueChange={v => setWaitlistForm({ ...waitlistForm, requested_lot_type: v })}>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="rv">RV</SelectItem>
<SelectItem value="boat">Boat</SelectItem>
<SelectItem value="either">Either</SelectItem>
</SelectContent>
</Select>
</div>
<div><Label>Vehicle description</Label><Input value={waitlistForm.vehicle_description} onChange={e => setWaitlistForm({ ...waitlistForm, vehicle_description: e.target.value })} /></div>
<div><Label>Notes</Label><Textarea value={waitlistForm.notes} onChange={e => setWaitlistForm({ ...waitlistForm, notes: e.target.value })} /></div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setWaitlistDialogOpen(false)}>Cancel</Button>
<Button onClick={addWaitlist}>Add</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</CardHeader>
<CardContent>
{loading ? <p className="text-sm text-muted-foreground">Loading</p> :
waitingEntries.length === 0 ? <p className="text-sm text-muted-foreground">No one on the waitlist.</p> : (
<Table>
<TableHeader>
<TableRow>
<TableHead className="w-16">#</TableHead>
<TableHead>Name</TableHead>
<TableHead>Contact</TableHead>
<TableHead>Type</TableHead>
<TableHead>Joined</TableHead>
<TableHead>Source</TableHead>
<TableHead className="text-right">Actions</TableHead>
</TableRow>
</TableHeader>
<TableBody>
{waitingEntries.map((w, i) => (
<TableRow key={w.id}>
<TableCell className="font-semibold">{i + 1}</TableCell>
<TableCell>
<div className="font-medium">{w.requester_name}</div>
{w.unit_address && <div className="text-xs text-muted-foreground">{w.unit_address}</div>}
</TableCell>
<TableCell className="text-xs">
{w.requester_email && <div>{w.requester_email}</div>}
{w.requester_phone && <div>{w.requester_phone}</div>}
</TableCell>
<TableCell><Badge variant="outline">{w.requested_lot_type}</Badge></TableCell>
<TableCell className="text-xs">{new Date(w.joined_at).toLocaleDateString()}</TableCell>
<TableCell><Badge variant="secondary">{w.source === "public_form" ? "Public" : "Manual"}</Badge></TableCell>
<TableCell className="text-right space-x-1">
<Button size="icon" variant="ghost" onClick={() => movePosition(w, "up")} disabled={i === 0}><ArrowUp className="h-4 w-4" /></Button>
<Button size="icon" variant="ghost" onClick={() => movePosition(w, "down")} disabled={i === waitingEntries.length - 1}><ArrowDown className="h-4 w-4" /></Button>
<Button size="sm" variant="default" onClick={() => openFulfill(w)}><Check className="h-4 w-4 mr-1" /> Check off</Button>
<Button size="icon" variant="ghost" onClick={() => removeWaitlist(w.id)}><Trash2 className="h-4 w-4" /></Button>
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
)}
</CardContent>
</Card>
</TabsContent>
{/* Lots */}
<TabsContent value="lots" className="space-y-4">
<Card>
<CardHeader className="flex flex-row items-center justify-between">
<CardTitle>Lot Inventory</CardTitle>
<Dialog open={lotDialogOpen} onOpenChange={setLotDialogOpen}>
<DialogTrigger asChild><Button size="sm"><Plus className="h-4 w-4 mr-1" /> Add lot</Button></DialogTrigger>
<DialogContent>
<DialogHeader>
<DialogTitle>Add lot</DialogTitle>
<DialogDescription>Add a new RV or boat lot to inventory.</DialogDescription>
</DialogHeader>
<div className="space-y-3">
<div><Label>Lot number *</Label><Input value={lotForm.lot_number} onChange={e => setLotForm({ ...lotForm, lot_number: e.target.value })} /></div>
<div>
<Label>Type</Label>
<Select value={lotForm.lot_type} onValueChange={v => setLotForm({ ...lotForm, lot_type: v })}>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="rv">RV</SelectItem>
<SelectItem value="boat">Boat</SelectItem>
<SelectItem value="either">Either</SelectItem>
</SelectContent>
</Select>
</div>
<div className="grid grid-cols-2 gap-3">
<div><Label>Size</Label><Input placeholder="10x30" value={lotForm.size} onChange={e => setLotForm({ ...lotForm, size: e.target.value })} /></div>
<div><Label>Monthly rate</Label><Input type="number" value={lotForm.monthly_rate} onChange={e => setLotForm({ ...lotForm, monthly_rate: e.target.value })} /></div>
</div>
<div><Label>Notes</Label><Textarea value={lotForm.notes} onChange={e => setLotForm({ ...lotForm, notes: e.target.value })} /></div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setLotDialogOpen(false)}>Cancel</Button>
<Button onClick={createLot}>Add</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</CardHeader>
<CardContent>
{selectedLotIds.size > 0 && (
<div className="flex items-center justify-between rounded-md border bg-muted/40 px-3 py-2 mb-3">
<span className="text-sm font-medium">{selectedLotIds.size} selected</span>
<div className="flex items-center gap-2">
<Button size="sm" variant="outline" onClick={() => { setLotBulk({ lot_type: "no_change", status: "no_change", monthly_rate: "" }); setLotBulkOpen(true); }}><Pencil className="h-3.5 w-3.5 mr-1" /> Edit</Button>
<Button size="sm" variant="outline" className="text-destructive hover:text-destructive" onClick={deleteLots}><Trash2 className="h-3.5 w-3.5 mr-1" /> Delete</Button>
<Button size="sm" variant="ghost" onClick={() => setSelectedLotIds(new Set())}><X className="h-3.5 w-3.5 mr-1" /> Clear</Button>
</div>
</div>
)}
{lots.length === 0 ? <p className="text-sm text-muted-foreground">No lots yet.</p> : (
<Table>
<TableHeader><TableRow>
<TableHead className="w-10"><Checkbox aria-label="Select all lots" checked={lots.length > 0 && lots.every(l => selectedLotIds.has(l.id))} onCheckedChange={(c) => toggleSetMany(setSelectedLotIds, lots.map(l => l.id), !!c)} /></TableHead>
<TableHead>Lot #</TableHead><TableHead>Type</TableHead><TableHead>Size</TableHead>
<TableHead>Rate</TableHead><TableHead>Status</TableHead><TableHead className="text-right">Actions</TableHead>
</TableRow></TableHeader>
<TableBody>
{lots.map(lot => (
<TableRow key={lot.id}>
<TableCell><Checkbox aria-label={`Select lot ${lot.lot_number}`} checked={selectedLotIds.has(lot.id)} onCheckedChange={() => toggleSet(setSelectedLotIds, lot.id)} /></TableCell>
<TableCell className="font-medium">{lot.lot_number}</TableCell>
<TableCell><Badge variant="outline">{lot.lot_type}</Badge></TableCell>
<TableCell>{lot.size || "-"}</TableCell>
<TableCell>{lot.monthly_rate ? `$${lot.monthly_rate}` : "-"}</TableCell>
<TableCell>
<Badge variant={lot.status === "available" ? "secondary" : lot.status === "occupied" ? "default" : "outline"}>{lot.status}</Badge>
</TableCell>
<TableCell className="text-right">
<Button size="icon" variant="ghost" onClick={() => deleteLot(lot.id)}><Trash2 className="h-4 w-4" /></Button>
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
)}
</CardContent>
</Card>
</TabsContent>
{/* Rentals */}
<TabsContent value="rentals" className="space-y-4">
<Card>
<CardHeader className="flex flex-row items-center justify-between">
<CardTitle>Active Rentals</CardTitle>
<Dialog open={rentalDialogOpen} onOpenChange={setRentalDialogOpen}>
<DialogTrigger asChild><Button size="sm"><Plus className="h-4 w-4 mr-1" /> New rental</Button></DialogTrigger>
<DialogContent>
<DialogHeader>
<DialogTitle>Create rental</DialogTitle>
<DialogDescription>Assign a lot to a renter.</DialogDescription>
</DialogHeader>
<div className="space-y-3">
<div>
<Label>Lot *</Label>
<Select value={rentalForm.lot_id} onValueChange={v => setRentalForm({ ...rentalForm, lot_id: v })}>
<SelectTrigger><SelectValue placeholder="Choose lot" /></SelectTrigger>
<SelectContent>
{lots.filter(l => l.status === "available").map(l => (
<SelectItem key={l.id} value={l.id}>Lot {l.lot_number} ({l.lot_type})</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="grid grid-cols-2 gap-3">
<div>
<Label>Link to Owner</Label>
<Select value={rentalForm.owner_id || "none"} onValueChange={v => {
if (v === "none") { setRentalForm(f => ({ ...f, owner_id: "" })); return; }
const o = owners.find(x => x.id === v);
setRentalForm(f => ({
...f, owner_id: v,
unit_id: o?.unit_id || f.unit_id,
renter_name: f.renter_name || `${o?.first_name || ""} ${o?.last_name || ""}`.trim(),
renter_email: f.renter_email || o?.email || "",
renter_phone: f.renter_phone || o?.phone || "",
}));
}}>
<SelectTrigger><SelectValue placeholder="Optional" /></SelectTrigger>
<SelectContent>
<SelectItem value="none">None</SelectItem>
{owners.map(o => <SelectItem key={o.id} value={o.id}>{`${o.last_name || ""}, ${o.first_name || ""}`.replace(/^, |, $/g, "").trim() || "Unnamed"}</SelectItem>)}
</SelectContent>
</Select>
</div>
<div>
<Label>Link to Unit</Label>
<Select value={rentalForm.unit_id || "none"} onValueChange={v => setRentalForm(f => ({ ...f, unit_id: v === "none" ? "" : v }))}>
<SelectTrigger><SelectValue placeholder="Optional" /></SelectTrigger>
<SelectContent>
<SelectItem value="none">None</SelectItem>
{units.map(u => <SelectItem key={u.id} value={u.id}>Unit {u.unit_number}</SelectItem>)}
</SelectContent>
</Select>
</div>
</div>
<div><Label>Renter name *</Label><Input value={rentalForm.renter_name} onChange={e => setRentalForm({ ...rentalForm, renter_name: e.target.value })} /></div>
<div className="grid grid-cols-2 gap-3">
<div><Label>Email</Label><Input type="email" value={rentalForm.renter_email} onChange={e => setRentalForm({ ...rentalForm, renter_email: e.target.value })} /></div>
<div><Label>Phone</Label><Input value={rentalForm.renter_phone} onChange={e => setRentalForm({ ...rentalForm, renter_phone: e.target.value })} /></div>
</div>
<div><Label>Vehicle description</Label><Input value={rentalForm.vehicle_description} onChange={e => setRentalForm({ ...rentalForm, vehicle_description: e.target.value })} /></div>
<div className="grid grid-cols-2 gap-3">
<div><Label>Start date</Label><Input type="date" value={rentalForm.start_date} onChange={e => setRentalForm({ ...rentalForm, start_date: e.target.value })} /></div>
<div><Label>Monthly rate</Label><Input type="number" value={rentalForm.monthly_rate} onChange={e => setRentalForm({ ...rentalForm, monthly_rate: e.target.value })} /></div>
</div>
<div><Label>Notes</Label><Textarea value={rentalForm.notes} onChange={e => setRentalForm({ ...rentalForm, notes: e.target.value })} /></div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setRentalDialogOpen(false)}>Cancel</Button>
<Button onClick={createRental}>Create</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</CardHeader>
<CardContent>
{(() => { const activeRentals = rentals.filter(r => r.status === "active" || r.status === "vacating"); return (
<>
{selectedRentalIds.size > 0 && (
<div className="flex items-center justify-between rounded-md border bg-muted/40 px-3 py-2 mb-3">
<span className="text-sm font-medium">{selectedRentalIds.size} selected</span>
<div className="flex items-center gap-2">
<Button size="sm" variant="outline" onClick={() => { setRentalBulk({ status: "no_change", monthly_rate: "", is_owner: "no_change" }); setRentalBulkOpen(true); }}><Pencil className="h-3.5 w-3.5 mr-1" /> Edit</Button>
<Button size="sm" variant="ghost" onClick={() => setSelectedRentalIds(new Set())}><X className="h-3.5 w-3.5 mr-1" /> Clear</Button>
</div>
</div>
)}
{activeRentals.length === 0 ? <p className="text-sm text-muted-foreground">No active rentals.</p> : (
<Table>
<TableHeader><TableRow>
<TableHead className="w-10"><Checkbox aria-label="Select all rentals" checked={activeRentals.length > 0 && activeRentals.every(r => selectedRentalIds.has(r.id))} onCheckedChange={(c) => toggleSetMany(setSelectedRentalIds, activeRentals.map(r => r.id), !!c)} /></TableHead>
<TableHead>Lot</TableHead><TableHead>Renter</TableHead><TableHead>Vehicle</TableHead>
<TableHead>Start</TableHead><TableHead>Rate</TableHead><TableHead className="text-right">Actions</TableHead>
</TableRow></TableHeader>
<TableBody>
{activeRentals.map(r => {
const lot = lots.find(l => l.id === r.lot_id);
return (
<TableRow key={r.id}>
<TableCell><Checkbox aria-label={`Select rental ${r.renter_name}`} checked={selectedRentalIds.has(r.id)} onCheckedChange={() => toggleSet(setSelectedRentalIds, r.id)} /></TableCell>
<TableCell className="font-medium">{lot?.lot_number || "—"}</TableCell>
<TableCell>
<div className="flex items-center gap-2">
<span>{r.renter_name}</span>
<Badge variant={r.is_owner ? "default" : "secondary"} className="text-[10px]">
{r.is_owner ? "Owner" : "Renter"}
</Badge>
{r.status === "vacating" && <Badge variant="outline" className="text-[10px] border-amber-300 text-amber-700">Vacating</Badge>}
</div>
<div className="text-xs text-muted-foreground">{r.renter_email}</div>
{r.insurance_carrier ? (
<div className="text-xs text-emerald-700 mt-0.5">
Insured: {r.insurance_carrier}
{r.insurance_expiration_date ? ` · exp ${new Date(r.insurance_expiration_date + "T00:00:00").toLocaleDateString()}` : ""}
{r.insurance_document_url ? <> · <a href={r.insurance_document_url} target="_blank" rel="noreferrer" className="underline">COI</a></> : null}
</div>
) : (
<div className="text-xs text-amber-600 mt-0.5">No insurance on file</div>
)}
</TableCell>
<TableCell className="text-sm">{r.vehicle_description || "-"}</TableCell>
<TableCell className="text-sm">{r.start_date}</TableCell>
<TableCell>{r.monthly_rate ? `$${r.monthly_rate}` : "-"}</TableCell>
<TableCell className="text-right space-x-2">
<Button size="sm" variant="ghost" onClick={() => toggleIsOwner(r)}>
{r.is_owner ? "Mark as Renter" : "Mark as Owner"}
</Button>
<Button size="sm" variant="secondary" onClick={() => invitePortal(r, false)}>Invite Renter</Button>
<Button size="sm" variant="secondary" onClick={() => invitePortal(r, true)}>Invite Owner</Button>
<Button size="sm" variant="secondary" onClick={() => requestInsurance(r)}>Request Insurance</Button>
<Button size="sm" variant="outline" onClick={() => endRental(r)}>End rental</Button>
</TableCell>
</TableRow>
);
})}
</TableBody>
</Table>
)}
</>
); })()}
</CardContent>
</Card>
</TabsContent>
{/* History */}
<TabsContent value="history" className="space-y-4">
<Card>
<CardHeader><CardTitle>Waitlist History</CardTitle></CardHeader>
<CardContent>
{fulfilledEntries.length === 0 ? <p className="text-sm text-muted-foreground">No history yet.</p> : (
<Table>
<TableHeader><TableRow>
<TableHead>Name</TableHead><TableHead>Status</TableHead>
<TableHead>Lot Assigned</TableHead><TableHead>Date</TableHead>
</TableRow></TableHeader>
<TableBody>
{fulfilledEntries.map(w => (
<TableRow key={w.id}>
<TableCell>{w.requester_name}</TableCell>
<TableCell><Badge variant={w.status === "fulfilled" ? "default" : "outline"}>{w.status}</Badge></TableCell>
<TableCell>{w.fulfilled_lot_number || "-"}</TableCell>
<TableCell className="text-sm">{w.fulfilled_at ? new Date(w.fulfilled_at).toLocaleDateString() : "-"}</TableCell>
</TableRow>
))}
</TableBody>
</Table>
)}
</CardContent>
</Card>
</TabsContent>
</Tabs>
{/* Fulfill dialog */}
<Dialog open={fulfillDialog.open} onOpenChange={o => setFulfillDialog({ ...fulfillDialog, open: o })}>
<DialogContent>
<DialogHeader>
<DialogTitle>Check off waitlist entry</DialogTitle>
<DialogDescription>Enter the lot number assigned to {fulfillDialog.entry?.requester_name}.</DialogDescription>
</DialogHeader>
<div className="space-y-3">
<div>
<Label>Lot number assigned *</Label>
<Input value={fulfillDialog.lotNumber} onChange={e => setFulfillDialog({ ...fulfillDialog, lotNumber: e.target.value })} />
</div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setFulfillDialog({ open: false, entry: null, lotNumber: "" })}>Cancel</Button>
<Button onClick={confirmFulfill}>Mark fulfilled</Button>
</DialogFooter>
</DialogContent>
</Dialog>
{/* Bulk edit lots */}
<Dialog open={lotBulkOpen} onOpenChange={setLotBulkOpen}>
<DialogContent>
<DialogHeader>
<DialogTitle>Edit {selectedLotIds.size} lot{selectedLotIds.size === 1 ? "" : "s"}</DialogTitle>
<DialogDescription>Only changed fields are applied. Leave a field on No change to keep existing values.</DialogDescription>
</DialogHeader>
<div className="space-y-3">
<div>
<Label>Type</Label>
<Select value={lotBulk.lot_type} onValueChange={v => setLotBulk({ ...lotBulk, lot_type: v })}>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="no_change">No change</SelectItem>
<SelectItem value="rv">RV</SelectItem>
<SelectItem value="boat">Boat</SelectItem>
<SelectItem value="either">Either</SelectItem>
</SelectContent>
</Select>
</div>
<div>
<Label>Status</Label>
<Select value={lotBulk.status} onValueChange={v => setLotBulk({ ...lotBulk, status: v })}>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="no_change">No change</SelectItem>
<SelectItem value="available">Available</SelectItem>
<SelectItem value="occupied">Occupied</SelectItem>
<SelectItem value="maintenance">Maintenance</SelectItem>
</SelectContent>
</Select>
</div>
<div><Label>Monthly rate <span className="text-muted-foreground text-xs">(blank = no change)</span></Label><Input type="number" min="0" step="0.01" value={lotBulk.monthly_rate} onChange={e => setLotBulk({ ...lotBulk, monthly_rate: e.target.value })} placeholder="No change" /></div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setLotBulkOpen(false)}>Cancel</Button>
<Button onClick={applyLotBulk} disabled={savingLotBulk}>Apply to {selectedLotIds.size}</Button>
</DialogFooter>
</DialogContent>
</Dialog>
{/* Bulk edit rentals */}
<Dialog open={rentalBulkOpen} onOpenChange={setRentalBulkOpen}>
<DialogContent>
<DialogHeader>
<DialogTitle>Edit {selectedRentalIds.size} rental{selectedRentalIds.size === 1 ? "" : "s"}</DialogTitle>
<DialogDescription>Only changed fields are applied. Leave a field on No change to keep existing values.</DialogDescription>
</DialogHeader>
<div className="space-y-3">
<div>
<Label>Status</Label>
<Select value={rentalBulk.status} onValueChange={v => setRentalBulk({ ...rentalBulk, status: v })}>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="no_change">No change</SelectItem>
<SelectItem value="active">Active</SelectItem>
<SelectItem value="ended">Ended (sets end date to today)</SelectItem>
</SelectContent>
</Select>
</div>
<div>
<Label>Owner / Renter</Label>
<Select value={rentalBulk.is_owner} onValueChange={v => setRentalBulk({ ...rentalBulk, is_owner: v })}>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="no_change">No change</SelectItem>
<SelectItem value="true">Owner</SelectItem>
<SelectItem value="false">Renter</SelectItem>
</SelectContent>
</Select>
</div>
<div><Label>Monthly rate <span className="text-muted-foreground text-xs">(blank = no change)</span></Label><Input type="number" min="0" step="0.01" value={rentalBulk.monthly_rate} onChange={e => setRentalBulk({ ...rentalBulk, monthly_rate: e.target.value })} placeholder="No change" /></div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setRentalBulkOpen(false)}>Cancel</Button>
<Button onClick={applyRentalBulk} disabled={savingRentalBulk}>Apply to {selectedRentalIds.size}</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</div>
);
}