import React, { useEffect, useState, useCallback, useMemo, useRef } from 'react'; import { useSearchParams } from 'react-router-dom'; import { Plus, ChevronRight, SlidersHorizontal, Search, Trash2, X } from 'lucide-react'; import { toast } from 'sonner'; import { Button } from '@/components/ui/button'; import { Input } from '@/components/ui/input'; import { Label } from '@/components/ui/label'; import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter, } from '@/components/ui/dialog'; import { AlertDialog, AlertDialogContent, AlertDialogHeader, AlertDialogTitle, AlertDialogDescription, AlertDialogFooter, AlertDialogCancel, AlertDialogAction, } from '@/components/ui/alert-dialog'; import { Select, SelectTrigger, SelectValue, SelectContent, SelectItem, } from '@/components/ui/select'; import { api } from '@/api'; import { cn } from '@/lib/utils'; import BillsTableInner from '@/components/BillsTableInner'; import BillModal from '@/components/BillModal'; import { makeBillDraft } from '@/lib/billDrafts'; const VISIBILITY_OPTIONS = [ { value: 'default', label: 'Default', description: 'Use the app default behavior for inactive bill history.' }, { value: 'all', label: 'Show all history', description: 'Show all historical tracker data for this inactive bill.' }, { value: 'none', label: 'Show no history', description: 'Hide historical tracker data for this inactive bill.' }, { value: 'ranges', label: 'Show only selected date ranges', description: 'Show history only for the ranges listed below.' }, ]; const MONTH_OPTIONS = [ ['1', 'Jan'], ['2', 'Feb'], ['3', 'Mar'], ['4', 'Apr'], ['5', 'May'], ['6', 'Jun'], ['7', 'Jul'], ['8', 'Aug'], ['9', 'Sep'], ['10', 'Oct'], ['11', 'Nov'], ['12', 'Dec'], ]; const FILTER_ALL = 'all'; const BUILT_IN_TEMPLATES = [ { id: 'builtin-utility', name: 'Utility', data: { name: 'Utility Bill', due_day: 15, expected_amount: 0, billing_cycle: 'monthly', cycle_type: 'monthly', cycle_day: '1', autopay_enabled: false, autodraft_status: 'none', has_2fa: false, notes: 'Utility service bill', }, categoryKeywords: ['utility', 'utilities', 'electric', 'water'], }, { id: 'builtin-credit-card', name: 'Credit Card', data: { name: 'Credit Card', due_day: 20, expected_amount: 0, billing_cycle: 'monthly', cycle_type: 'monthly', cycle_day: '1', autopay_enabled: false, autodraft_status: 'none', has_2fa: true, current_balance: 0, minimum_payment: 0, interest_rate: 0, snowball_include: true, notes: 'Credit card minimum payment', }, categoryKeywords: ['credit card', 'credit cards', 'debt'], }, { id: 'builtin-subscription', name: 'Subscription', data: { name: 'Subscription', due_day: 1, expected_amount: 0, billing_cycle: 'monthly', cycle_type: 'monthly', cycle_day: '1', autopay_enabled: true, autodraft_status: 'assumed_paid', has_2fa: false, notes: 'Recurring subscription', }, categoryKeywords: ['subscription', 'subscriptions', 'entertainment'], }, { id: 'builtin-loan', name: 'Loan', data: { name: 'Loan Payment', due_day: 1, expected_amount: 0, billing_cycle: 'monthly', cycle_type: 'monthly', cycle_day: '1', autopay_enabled: false, autodraft_status: 'none', has_2fa: false, current_balance: 0, minimum_payment: 0, interest_rate: 0, snowball_include: true, notes: 'Installment loan payment', }, categoryKeywords: ['loan', 'loans', 'debt'], }, ]; // BillModal is imported from @/components/BillModal function blankRange() { const year = String(new Date().getFullYear()); return { _key: `new-${Date.now()}-${Math.random().toString(16).slice(2)}`, start_year: year, start_month: String(new Date().getMonth() + 1), end_year: '', end_month: '', label: '', }; } function normalizeRange(range) { return { ...range, _key: range._key || String(range.id), start_year: String(range.start_year ?? ''), start_month: String(range.start_month ?? ''), end_year: range.end_year == null ? '' : String(range.end_year), end_month: range.end_month == null ? '' : String(range.end_month), label: range.label || '', }; } function rangePayload(range) { return { start_year: parseInt(range.start_year, 10), start_month: parseInt(range.start_month, 10), end_year: range.end_year ? parseInt(range.end_year, 10) : null, end_month: range.end_month ? parseInt(range.end_month, 10) : null, label: range.label?.trim() || null, }; } function validateRange(range) { const payload = rangePayload(range); if (!Number.isInteger(payload.start_year) || payload.start_year < 2000 || payload.start_year > 2100) { return 'Start year must be between 2000 and 2100.'; } if (!Number.isInteger(payload.start_month) || payload.start_month < 1 || payload.start_month > 12) { return 'Start month must be between 1 and 12.'; } if ((payload.end_year == null) !== (payload.end_month == null)) { return 'End year and end month must both be provided or both left blank.'; } if (payload.end_year != null) { if (payload.end_year < 2000 || payload.end_year > 2100) { return 'End year must be between 2000 and 2100.'; } if (payload.end_month < 1 || payload.end_month > 12) { return 'End month must be between 1 and 12.'; } if ((payload.end_year * 12 + payload.end_month) < (payload.start_year * 12 + payload.start_month)) { return 'Range end must be on or after the start.'; } } return null; } // ── Display preferences ─────────────────────────────────────────────────────── const PREFS_KEY = 'bills-display-prefs-v1'; const PREFS_DEFAULTS = { showCategory: true, showDueDay: true, showAmount: true, showCycle: true, showApr: true, showBalance: true, showMinPayment: true, showAutopay: true, show2fa: true, }; const PREFS_LABELS = [ ['showCategory', 'Category'], ['showDueDay', 'Due day'], ['showAmount', 'Amount'], ['showCycle', 'Billing cycle'], ['showApr', 'APR'], ['showBalance', 'Balance'], ['showMinPayment', 'Min payment'], ['showAutopay', 'Autopay badge'], ['show2fa', '2FA badge'], ]; function useDisplayPrefs() { const [prefs, setPrefs] = useState(() => { try { const raw = localStorage.getItem(PREFS_KEY); return raw ? { ...PREFS_DEFAULTS, ...JSON.parse(raw) } : PREFS_DEFAULTS; } catch { return PREFS_DEFAULTS; } }); const toggle = (key) => { setPrefs(prev => { const next = { ...prev, [key]: !prev[key] }; try { localStorage.setItem(PREFS_KEY, JSON.stringify(next)); } catch {} return next; }); }; return { prefs, toggle }; } function DisplayPrefsPanel({ prefs, onToggle }) { const [open, setOpen] = useState(false); const ref = useRef(null); useEffect(() => { if (!open) return; const handler = (e) => { if (ref.current && !ref.current.contains(e.target)) setOpen(false); }; document.addEventListener('mousedown', handler); return () => document.removeEventListener('mousedown', handler); }, [open]); return (
{open && (

Display options

{PREFS_LABELS.map(([key, label]) => ( ))}
)}
); } function amountSearchText(...values) { return values .filter(value => value !== null && value !== undefined && Number.isFinite(Number(value))) .flatMap(value => { const num = Number(value); return [String(num), num.toFixed(2), `$${num.toFixed(2)}`]; }) .join(' '); } function billIsDebt(bill) { const category = String(bill.category_name || '').toLowerCase(); return Number(bill.current_balance) > 0 || bill.minimum_payment != null || ['credit card', 'credit cards', 'loan', 'loans', 'debt', 'mortgage'].some(token => category.includes(token)); } function FilterChip({ active, children, onClick }) { return ( ); } // ───────────────────────────────────────────────────────────────────────────── function HistoryVisibilityDialog({ bill, onClose, onSaved }) { const [visibility, setVisibility] = useState(bill?.history_visibility || 'default'); const [ranges, setRanges] = useState([]); const [deletedIds, setDeletedIds] = useState([]); const [loading, setLoading] = useState(true); const [saving, setSaving] = useState(false); useEffect(() => { let mounted = true; setLoading(true); api.billHistoryRanges(bill.id) .then(data => { if (!mounted) return; setVisibility(data.history_visibility || bill.history_visibility || 'default'); setRanges((data.ranges || []).map(normalizeRange)); }) .catch(err => { toast.error(err.message || 'Failed to load history visibility.'); onClose(); }) .finally(() => mounted && setLoading(false)); return () => { mounted = false; }; }, [bill, onClose]); const updateRange = (key, patch) => { setRanges(prev => prev.map(r => r._key === key ? { ...r, ...patch } : r)); }; const deleteRange = (range) => { if (range.id) setDeletedIds(prev => [...prev, range.id]); setRanges(prev => prev.filter(r => r._key !== range._key)); }; const save = async () => { if (visibility === 'ranges') { if (ranges.length === 0) { toast.error('Add at least one date range or choose another visibility option.'); return; } const invalid = ranges.map(validateRange).find(Boolean); if (invalid) { toast.error(invalid); return; } } setSaving(true); try { await api.updateBill(bill.id, { history_visibility: visibility }); for (const id of deletedIds) { await api.deleteBillHistoryRange(bill.id, id); } if (visibility === 'ranges') { for (const range of ranges) { const payload = rangePayload(range); if (range.id) { await api.updateBillHistoryRange(bill.id, range.id, payload); } else { await api.createBillHistoryRange(bill.id, payload); } } } toast.success('History visibility saved.'); onSaved(); onClose(); } catch (err) { toast.error(err.message || 'Failed to save history visibility.'); } finally { setSaving(false); } }; return ( { if (!v && !saving) onClose(); }}> History Visibility: {bill.name} {loading ? (
Loading history visibility...
) : (

Inactive bill history

Choose whether this inactive bill should appear in historical tracker views. Active bill behavior is unchanged.

{VISIBILITY_OPTIONS.map(option => ( ))}
{visibility === 'ranges' && (

Selected date ranges

End month/year are optional for open-ended ranges.

{ranges.length === 0 ? (
No ranges yet. Add a range to use selected history.
) : (
{ranges.map(range => (
updateRange(range._key, { start_year: e.target.value })} className="h-8" />
updateRange(range._key, { end_year: e.target.value })} className="h-8" />
updateRange(range._key, { label: e.target.value })} placeholder="Optional" className="h-8" />
))}
)}
)}
)}
); } // ── Bills Page ───────────────────────────────────────────────────────────── export default function BillsPage() { const [searchParams] = useSearchParams(); const [bills, setBills] = useState([]); const [categories, setCategories] = useState([]); const [savedTemplates, setSavedTemplates] = useState([]); const [loading, setLoading] = useState(true); const [showInactive, setShowInactive] = useState(false); const [search, setSearch] = useState(''); const [filters, setFilters] = useState({ category: FILTER_ALL, cycle: FILTER_ALL, autopay: false, firstBucket: false, fifteenthBucket: false, debt: false, inactive: false, }); // modal state: null = closed, { bill: null } = add new, { bill: {...} } = edit const [modal, setModal] = useState(null); // Bill pending deactivation confirmation (AlertDialog replaces window.confirm) const [deactivate, setDeactivate] = useState(null); const [deleteTarget, setDeleteTarget] = useState(null); const [deleteConfirmed, setDeleteConfirmed] = useState(false); const [deleteBusy, setDeleteBusy] = useState(false); const [historyTarget, setHistoryTarget] = useState(null); const { prefs, toggle: togglePref } = useDisplayPrefs(); const load = useCallback(async () => { try { const [billsRes, catRes, templateRes] = await Promise.all([ api.allBills(), api.categories(), api.billTemplates(), ]); setBills(billsRes || []); setCategories(catRes || []); setSavedTemplates(templateRes || []); } catch (err) { toast.error(err.message); } finally { setLoading(false); } }, []); useEffect(() => { load(); }, [load]); useEffect(() => { const querySearch = searchParams.get('search') || ''; const includeInactive = searchParams.get('inactive') === '1' || searchParams.get('inactive') === 'true'; if (querySearch) setSearch(querySearch); if (includeInactive) { setShowInactive(true); setFilters(prev => ({ ...prev, inactive: true })); } }, [searchParams]); useEffect(() => { if (filters.inactive) setShowInactive(true); }, [filters.inactive]); const toggleFilter = (key) => setFilters(prev => ({ ...prev, [key]: !prev[key] })); const setFilterValue = (key, value) => setFilters(prev => ({ ...prev, [key]: value })); const hasFilters = !!( search.trim() || filters.category !== FILTER_ALL || filters.cycle !== FILTER_ALL || filters.autopay || filters.firstBucket || filters.fifteenthBucket || filters.debt || filters.inactive ); const resetFilters = () => { setSearch(''); setFilters({ category: FILTER_ALL, cycle: FILTER_ALL, autopay: false, firstBucket: false, fifteenthBucket: false, debt: false, inactive: false, }); }; async function handleEdit(billId) { try { const bill = await api.bill(billId); setModal({ bill }); } catch (err) { toast.error(err.message); } } function handleDuplicateBill(bill) { setModal({ bill: null, initialBill: makeBillDraft(bill, { copy: true, categories }) }); } function handleTemplateSelect(value) { if (!value || value === 'placeholder') return; const builtIn = BUILT_IN_TEMPLATES.find(template => template.id === value); if (builtIn) { setModal({ bill: null, initialBill: makeBillDraft(builtIn.data, { template: builtIn, categories }) }); return; } const saved = savedTemplates.find(template => `saved-${template.id}` === value); if (saved) { setModal({ bill: null, initialBill: makeBillDraft(saved.data, { template: saved, categories }) }); } } function handleToggle(bill) { if (bill.active) { // Prompt confirmation before deactivating setDeactivate(bill); } else { doToggle(bill); } } async function doToggle(bill) { if (!bill) return; try { await api.updateBill(bill.id, { active: bill.active ? 0 : 1 }); toast.success(bill.active ? 'Bill deactivated' : 'Bill activated'); load(); } catch (err) { toast.error(err.message); } } function handleDeleteRequest(bill) { setDeleteConfirmed(false); setDeleteTarget(bill); } async function handleDeleteConfirmed() { if (!deleteTarget || !deleteConfirmed) return; setDeleteBusy(true); try { const bill = deleteTarget; await api.deleteBill(bill.id); setBills(prev => prev.filter(b => b.id !== bill.id)); toast.success(`"${bill.name}" moved to recovery`, { description: 'It will be permanently purged after 30 days.', action: { label: 'Undo', onClick: async () => { try { await api.restoreBill(bill.id); toast.success(`"${bill.name}" restored`); load(); } catch (err) { toast.error(err.message || 'Failed to restore bill'); } }, }, }); setDeleteTarget(null); setDeleteConfirmed(false); load(); } catch (err) { toast.error(err.message || 'Failed to delete bill'); } finally { setDeleteBusy(false); } } async function handleDeleteAlternative() { if (!deleteTarget) return; const bill = deleteTarget; setDeleteTarget(null); setDeleteConfirmed(false); await doToggle(bill); } const cycleOptions = useMemo(() => ( Array.from(new Set(bills.map(b => b.billing_cycle || 'monthly'))).sort() ), [bills]); const filteredBills = useMemo(() => { const q = search.trim().toLowerCase(); return bills.filter(bill => { if (filters.inactive && bill.active) return false; if (filters.category !== FILTER_ALL && String(bill.category_id ?? '') !== filters.category) return false; if (filters.cycle !== FILTER_ALL && String(bill.billing_cycle || 'monthly') !== filters.cycle) return false; if (filters.autopay && !bill.autopay_enabled) return false; if (filters.debt && !billIsDebt(bill)) return false; if (filters.firstBucket && !filters.fifteenthBucket && bill.bucket !== '1st') return false; if (filters.fifteenthBucket && !filters.firstBucket && bill.bucket !== '15th') return false; if (!q) return true; const haystack = [ bill.name, bill.category_name, bill.notes, bill.billing_cycle, bill.bucket, amountSearchText(bill.expected_amount, bill.current_balance, bill.minimum_payment, bill.interest_rate), ].filter(Boolean).join(' ').toLowerCase(); return haystack.includes(q); }); }, [bills, filters, search]); const active = filteredBills.filter(b => b.active); const inactive = filteredBills.filter(b => !b.active); const totalActive = bills.filter(b => b.active).length; const totalInactive = bills.filter(b => !b.active).length; return (
{/* ── Header ── */}

Manage

Bills

{totalActive} active {totalInactive > 0 && ( · {totalInactive} inactive )}

toggleFilter('autopay')}>Autopay toggleFilter('firstBucket')}>1st bucket toggleFilter('fifteenthBucket')}>15th bucket toggleFilter('debt')}>Debt toggleFilter('inactive')}>Inactive {filteredBills.length} of {bills.length} shown
{/* ── Active Bills ── */} {!filters.inactive && (
Active {active.length}
{loading ? (
{[...Array(4)].map((_, i) => (
))}
) : active.length === 0 ? (
{hasFilters ? ( <>No active bills match your filters. ) : ( <> No active bills.{' '} )}
) : ( )}
)} {/* ── Inactive Bills ── */} {!loading && inactive.length > 0 && (
{(showInactive || filters.inactive) && (
Inactive {inactive.length}
)}
)} {!loading && filters.inactive && inactive.length === 0 && (
No inactive bills match your filters.
)} {/* ── Bill Modal ── */} {modal && ( setModal(null)} onSave={load} onDuplicate={handleDuplicateBill} /> )} {historyTarget && ( setHistoryTarget(null)} onSaved={load} /> )} {/* ── Deactivate confirmation (replaces window.confirm) ── */} { if (!open) setDeactivate(null); }}> Deactivate "{deactivate?.name}"? This bill will be hidden from the tracker. You can reactivate it at any time. setDeactivate(null)}>Cancel { const b = deactivate; setDeactivate(null); doToggle(b); }} > Deactivate {/* ── Soft delete confirmation ── */} { if (!open && !deleteBusy) { setDeleteTarget(null); setDeleteConfirmed(false); } }} > Move "{deleteTarget?.name}" to recovery? This hides the bill from normal tracking and keeps its payments, monthly history, notes, and history ranges recoverable for 30 days.
Deactivate is still the best option if you want to retain the bill long-term but hide it from active tracking.
Cancel
); }