BillTracker/client/components/CommandPalette.jsx

217 lines
7.7 KiB
React
Raw Normal View History

2026-05-16 15:38:28 -05:00
import React, { useEffect, useMemo, useRef, useState } from 'react';
import { useNavigate } from 'react-router-dom';
import { Loader2, Receipt, Search, X } from 'lucide-react';
import { toast } from 'sonner';
import { api } from '@/api';
import { cn, fmt } from '@/lib/utils';
import { Button } from '@/components/ui/button';
import { Dialog, DialogContent, DialogHeader, DialogTitle } from '@/components/ui/dialog';
import { Input } from '@/components/ui/input';
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 billSearchText(bill) {
return [
bill.name,
bill.category_name,
bill.notes,
bill.billing_cycle,
bill.bucket,
bill.website,
amountSearchText(
bill.expected_amount,
bill.current_balance,
bill.minimum_payment,
bill.interest_rate,
),
].filter(Boolean).join(' ').toLowerCase();
}
function shortcutLabel() {
if (typeof navigator !== 'undefined' && /Mac|iPhone|iPad|iPod/i.test(navigator.platform)) {
return 'Cmd K';
}
return 'Ctrl K';
}
export default function CommandPalette() {
const navigate = useNavigate();
const inputRef = useRef(null);
const [open, setOpen] = useState(false);
const [query, setQuery] = useState('');
const [bills, setBills] = useState([]);
const [loading, setLoading] = useState(false);
const [loaded, setLoaded] = useState(false);
useEffect(() => {
const openPalette = () => setOpen(true);
const onKeyDown = (event) => {
if ((event.ctrlKey || event.metaKey) && event.key.toLowerCase() === 'k') {
event.preventDefault();
setOpen(value => !value);
}
};
window.addEventListener('command-palette:open', openPalette);
window.addEventListener('keydown', onKeyDown);
return () => {
window.removeEventListener('command-palette:open', openPalette);
window.removeEventListener('keydown', onKeyDown);
};
}, []);
useEffect(() => {
if (!open) return;
window.setTimeout(() => inputRef.current?.focus(), 0);
if (loaded || loading) return;
setLoading(true);
api.allBills({ inactive: true })
.then(rows => {
setBills(Array.isArray(rows) ? rows : []);
setLoaded(true);
})
.catch(err => {
toast.error(err.message || 'Failed to load bills');
})
.finally(() => setLoading(false));
}, [loaded, loading, open]);
const results = useMemo(() => {
const q = query.trim().toLowerCase();
const sorted = [...bills].sort((a, b) => {
if (!!a.active !== !!b.active) return a.active ? -1 : 1;
return String(a.name || '').localeCompare(String(b.name || ''));
});
if (!q) return sorted.slice(0, 8);
return sorted
.filter(bill => billSearchText(bill).includes(q))
.slice(0, 12);
}, [bills, query]);
const close = () => {
setOpen(false);
setQuery('');
};
const openBills = (bill) => {
const params = new URLSearchParams({ search: bill.name });
if (!bill.active) params.set('inactive', '1');
navigate(`/bills?${params.toString()}`);
close();
};
const openTracker = (bill) => {
const params = new URLSearchParams({ search: bill.name });
navigate(`/?${params.toString()}`);
close();
};
return (
<Dialog open={open} onOpenChange={setOpen}>
<DialogContent className="gap-0 overflow-hidden p-0 sm:max-w-2xl">
<DialogHeader className="sr-only">
<DialogTitle>Command palette</DialogTitle>
</DialogHeader>
<div className="flex items-center gap-2 border-b border-border/70 px-4 py-3">
<Search className="h-4 w-4 shrink-0 text-muted-foreground" />
<Input
ref={inputRef}
value={query}
onChange={event => setQuery(event.target.value)}
onKeyDown={event => {
if (event.key === 'Enter' && results[0]) openBills(results[0]);
}}
placeholder="Find a bill by name, category, notes, or amount"
className="h-9 border-0 bg-transparent px-0 text-base shadow-none focus-visible:ring-0"
/>
{query && (
<Button
type="button"
size="icon"
variant="ghost"
className="h-7 w-7 shrink-0"
onClick={() => setQuery('')}
aria-label="Clear search"
>
<X className="h-4 w-4" />
</Button>
)}
</div>
<div className="max-h-[min(28rem,70svh)] overflow-y-auto p-2">
{loading ? (
<div className="flex items-center justify-center gap-2 px-4 py-12 text-sm text-muted-foreground">
<Loader2 className="h-4 w-4 animate-spin" />
Loading bills...
</div>
) : results.length > 0 ? (
<div className="space-y-1">
{results.map(bill => (
<div
key={bill.id}
className={cn(
'group grid gap-2 rounded-lg border border-transparent p-2 transition-colors',
'hover:border-border/70 hover:bg-accent/65 sm:grid-cols-[1fr_auto]',
)}
>
<button
type="button"
onClick={() => openBills(bill)}
className="flex min-w-0 items-center gap-3 rounded-md text-left focus-visible:outline-none focus-visible:ring-[3px] focus-visible:ring-ring/50"
>
<span className="flex h-9 w-9 shrink-0 items-center justify-center rounded-md bg-primary/10 text-primary">
<Receipt className="h-4 w-4" />
</span>
<span className="min-w-0">
<span className="flex min-w-0 items-center gap-2">
<span className="truncate text-sm font-semibold text-foreground">{bill.name}</span>
{!bill.active && (
<span className="shrink-0 rounded-full border border-border px-1.5 py-0.5 text-[10px] font-medium text-muted-foreground">
Inactive
</span>
)}
</span>
<span className="mt-0.5 flex flex-wrap items-center gap-x-2 gap-y-0.5 text-xs text-muted-foreground">
<span>{bill.category_name || 'Uncategorized'}</span>
<span>Due {bill.due_day}</span>
<span>{fmt(bill.expected_amount || 0)}</span>
</span>
</span>
</button>
<div className="flex items-center justify-end gap-1 pl-12 sm:pl-0">
<Button type="button" size="sm" variant="ghost" className="h-8 px-2.5 text-xs" onClick={() => openTracker(bill)}>
Tracker
</Button>
<Button type="button" size="sm" variant="default" className="h-8 px-2.5 text-xs" onClick={() => openBills(bill)}>
Bills
</Button>
</div>
</div>
))}
</div>
) : (
<div className="px-4 py-12 text-center text-sm text-muted-foreground">
No bills found.
</div>
)}
</div>
<div className="flex items-center justify-between border-t border-border/70 px-4 py-2 text-xs text-muted-foreground">
<span>Enter opens Bills</span>
<span>{shortcutLabel()}</span>
</div>
</DialogContent>
</Dialog>
);
}