Files
acmcc/src/pages/RVBoatLotsPage.tsx
T
admin f549f21c21 RV/Boat Lots: waitlist size + more vehicle types
- Waitlist now captures a free-form Size (requested_size column) in place of
  the type field in the internal form/table.
- Lot type selector (add + bulk edit) expanded to RV, Boat, Travel Trailer,
  Fifth Wheel, Camper, Car, Truck, Trailer, Other.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-07 21:01:18 -04:00

898 lines
50 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 GoogleMapPicker, { type MapPinData } from "@/components/association/GoogleMapPicker";
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; requested_size: string | null;
vehicle_description: string | null; notes: string | null;
status: string; source: string; joined_at: string;
fulfilled_at: string | null; fulfilled_lot_number: string | null;
};
const VEHICLE_TYPES = [
{ value: "rv", label: "RV" },
{ value: "boat", label: "Boat" },
{ value: "travel_trailer", label: "Travel Trailer" },
{ value: "fifth_wheel", label: "Fifth Wheel" },
{ value: "camper", label: "Camper" },
{ value: "car", label: "Car" },
{ value: "truck", label: "Truck" },
{ value: "trailer", label: "Trailer" },
{ value: "other", label: "Other" },
];
const vehicleTypeLabel = (v: string) => VEHICLE_TYPES.find(t => t.value === v)?.label || v;
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", requested_size: "", 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" });
// internal map state
const [mapPins, setMapPins] = useState<MapPinData[]>([]);
const [mapZoom, setMapZoom] = useState<number>(17);
const [mapLockedView, setMapLockedView] = useState<{ lat: number; lng: number; zoom: number } | null>(null);
const [savingMap, setSavingMap] = useState(false);
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 || []);
const { data: mapRow } = await supabase.from("rv_boat_lot_maps").select("config").eq("association_id", associationId).maybeSingle();
const cfg = (mapRow?.config as any) || {};
setMapPins(Array.isArray(cfg.pins) ? cfg.pins : []);
setMapZoom(typeof cfg.zoom === "number" ? cfg.zoom : 17);
setMapLockedView(cfg.locked_view && typeof cfg.locked_view.lat === "number" ? cfg.locked_view : null);
} 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 saveMap = async () => {
if (!associationId) return;
setSavingMap(true);
const { error } = await supabase.from("rv_boat_lot_maps").upsert({
association_id: associationId,
config: { pins: mapPins, zoom: mapZoom, locked_view: mapLockedView },
updated_at: new Date().toISOString(),
}, { onConflict: "association_id" });
setSavingMap(false);
if (error) { toast.error(error.message); return; }
toast.success("Map saved");
};
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,
requested_size: waitlistForm.requested_size || null,
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", requested_size: "", 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="map">Map</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>Size</Label><Input placeholder="e.g. 30 ft, 10x30" value={waitlistForm.requested_size} onChange={e => setWaitlistForm({ ...waitlistForm, requested_size: e.target.value })} /></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>Size</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 className="text-sm">{w.requested_size || "—"}</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>
{VEHICLE_TYPES.map(t => <SelectItem key={t.value} value={t.value}>{t.label}</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">{vehicleTypeLabel(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>
{(() => {
if (!r.insurance_carrier && !r.insurance_expiration_date) {
return <div className="text-xs text-amber-600 mt-0.5">No insurance on file</div>;
}
const exp = r.insurance_expiration_date ? new Date(r.insurance_expiration_date + "T00:00:00") : null;
const today = new Date(); today.setHours(0, 0, 0, 0);
const diffDays = exp ? Math.floor((exp.getTime() - today.getTime()) / 86400000) : null;
const isExpired = diffDays !== null && diffDays < 0;
const isExpiring = diffDays !== null && diffDays >= 0 && diffDays <= 90;
const tone = isExpired ? "text-destructive font-semibold" : isExpiring ? "text-amber-600" : "text-emerald-700";
return (
<div className={`text-xs mt-0.5 ${tone}`}>
Insured: {r.insurance_carrier || "—"}
{exp && (isExpired
? ` · Expired ${exp.toLocaleDateString()}`
: isExpiring
? ` · Expires in ${diffDays}d (${exp.toLocaleDateString()})`
: ` · exp ${exp.toLocaleDateString()}`)}
{r.insurance_document_url ? <> · <a href={r.insurance_document_url} target="_blank" rel="noreferrer" className="underline">COI</a></> : null}
</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>
{/* Map */}
<TabsContent value="map" className="space-y-4">
<Card>
<CardHeader className="flex flex-row items-center justify-between">
<div>
<CardTitle>Lot Map</CardTitle>
<p className="text-sm text-muted-foreground mt-1">
Click the map to drop a pin for each lot, label it, and optionally link a directory unit for reference. Internal use only.
</p>
</div>
<Button size="sm" onClick={saveMap} disabled={savingMap}>{savingMap ? "Saving…" : "Save Map"}</Button>
</CardHeader>
<CardContent>
<GoogleMapPicker
pins={mapPins}
onChange={setMapPins}
zoom={mapZoom}
onZoomChange={setMapZoom}
lockedView={mapLockedView}
onLockedViewChange={setMapLockedView}
linkLabel="unit"
allowLinkAnyStatus
formAmenities={units.map(u => ({ id: u.id, name: `Unit ${u.unit_number}` }))}
/>
</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>
{VEHICLE_TYPES.map(t => <SelectItem key={t.value} value={t.value}>{t.label}</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>
);
}