311 lines
11 KiB
JavaScript
311 lines
11 KiB
JavaScript
'use client';
|
|
|
|
import { useState, useEffect, useRef, useCallback } from 'react';
|
|
import { toast } from 'sonner';
|
|
import {
|
|
AlertTriangle, Building2, CheckCircle2, Loader2, Plus, Tag, Trash2, X,
|
|
} from 'lucide-react';
|
|
import { api } from '@/api';
|
|
import { cn } from '@/lib/utils';
|
|
import { Button } from '@/components/ui/button';
|
|
import { Input } from '@/components/ui/input';
|
|
|
|
// Debounce helper
|
|
function useDebounce(value, delay) {
|
|
const [debounced, setDebounced] = useState(value);
|
|
useEffect(() => {
|
|
const t = setTimeout(() => setDebounced(value), delay);
|
|
return () => clearTimeout(t);
|
|
}, [value, delay]);
|
|
return debounced;
|
|
}
|
|
|
|
function RuleChip({ rule, onDelete, deleting }) {
|
|
return (
|
|
<span className="inline-flex items-center gap-1.5 rounded-full border border-primary/25 bg-primary/8 px-2.5 py-1 text-xs font-medium text-primary">
|
|
<Tag className="h-3 w-3 shrink-0" />
|
|
<span className="max-w-[12rem] truncate" title={rule.merchant}>{rule.merchant}</span>
|
|
<button
|
|
type="button"
|
|
onClick={() => onDelete(rule)}
|
|
disabled={deleting === rule.id}
|
|
className="ml-0.5 rounded-full p-0.5 opacity-60 transition-opacity hover:opacity-100 disabled:opacity-30"
|
|
aria-label={`Remove rule "${rule.merchant}"`}
|
|
>
|
|
{deleting === rule.id
|
|
? <Loader2 className="h-3 w-3 animate-spin" />
|
|
: <X className="h-3 w-3" />}
|
|
</button>
|
|
</span>
|
|
);
|
|
}
|
|
|
|
function ConflictWarning({ conflicts }) {
|
|
if (!conflicts?.length) return null;
|
|
return (
|
|
<div className="flex items-start gap-2 rounded-md border border-amber-500/30 bg-amber-500/8 px-3 py-2 text-xs text-amber-700 dark:text-amber-400">
|
|
<AlertTriangle className="mt-0.5 h-3.5 w-3.5 shrink-0" />
|
|
<span>
|
|
This pattern is already used by{' '}
|
|
{conflicts.map((c, i) => (
|
|
<span key={c.id}>
|
|
{i > 0 && ', '}
|
|
<strong>{c.name}</strong>
|
|
</span>
|
|
))}.
|
|
Transactions could match both bills — consider making your pattern more specific.
|
|
</span>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function PreviewBadge({ count, loading }) {
|
|
if (loading) return <Loader2 className="h-3.5 w-3.5 animate-spin text-muted-foreground" />;
|
|
if (count === null) return null;
|
|
return (
|
|
<span className={cn(
|
|
'rounded-full border px-2 py-0.5 text-[10px] font-semibold tabular-nums',
|
|
count > 0
|
|
? 'border-emerald-500/30 bg-emerald-500/10 text-emerald-600 dark:text-emerald-400'
|
|
: 'border-border/60 bg-muted/40 text-muted-foreground',
|
|
)}>
|
|
{count === 0 ? 'No matches' : `${count} match${count === 1 ? '' : 'es'}`}
|
|
</span>
|
|
);
|
|
}
|
|
|
|
export default function BillMerchantRules({ billId, onRulesChanged }) {
|
|
const [rules, setRules] = useState([]);
|
|
const [suggestions, setSuggestions] = useState([]);
|
|
const [loading, setLoading] = useState(true);
|
|
const [deleting, setDeleting] = useState(null);
|
|
const [adding, setAdding] = useState(false);
|
|
const [input, setInput] = useState('');
|
|
const [showSuggestions, setShowSuggestions] = useState(false);
|
|
const [previewCount, setPreviewCount] = useState(null);
|
|
const [previewLoading, setPreviewLoading] = useState(false);
|
|
const [conflicts, setConflicts] = useState([]);
|
|
const [retroFeedback, setRetroFeedback] = useState(null);
|
|
const inputRef = useRef(null);
|
|
const dropdownRef = useRef(null);
|
|
|
|
const debouncedInput = useDebounce(input.trim(), 380);
|
|
|
|
const load = useCallback(async () => {
|
|
if (!billId) return;
|
|
setLoading(true);
|
|
try {
|
|
const data = await api.billMerchantRules(billId);
|
|
setRules(data.rules || []);
|
|
setSuggestions(data.suggestions || []);
|
|
} catch {
|
|
// non-fatal
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
}, [billId]);
|
|
|
|
useEffect(() => { load(); }, [load]);
|
|
|
|
// Preview debounced input
|
|
useEffect(() => {
|
|
if (!debouncedInput || debouncedInput.length < 2) {
|
|
setPreviewCount(null);
|
|
setConflicts([]);
|
|
return;
|
|
}
|
|
let cancelled = false;
|
|
setPreviewLoading(true);
|
|
api.previewMerchantRule(billId, debouncedInput)
|
|
.then(data => {
|
|
if (cancelled) return;
|
|
setPreviewCount(data.match_count);
|
|
setConflicts(data.conflicts || []);
|
|
})
|
|
.catch(() => {})
|
|
.finally(() => { if (!cancelled) setPreviewLoading(false); });
|
|
return () => { cancelled = true; };
|
|
}, [debouncedInput, billId]);
|
|
|
|
// Close suggestion dropdown on outside click
|
|
useEffect(() => {
|
|
function handler(e) {
|
|
if (!dropdownRef.current?.contains(e.target) && !inputRef.current?.contains(e.target)) {
|
|
setShowSuggestions(false);
|
|
}
|
|
}
|
|
document.addEventListener('mousedown', handler);
|
|
return () => document.removeEventListener('mousedown', handler);
|
|
}, []);
|
|
|
|
async function handleAdd(merchantText) {
|
|
const text = (merchantText || input).trim();
|
|
if (!text) return;
|
|
setAdding(true);
|
|
setRetroFeedback(null);
|
|
try {
|
|
const result = await api.addMerchantRule(billId, text);
|
|
setRules(prev => {
|
|
if (prev.some(r => r.id === result.rule?.id)) return prev;
|
|
return [...prev, result.rule].filter(Boolean);
|
|
});
|
|
setInput('');
|
|
setPreviewCount(null);
|
|
setConflicts([]);
|
|
setShowSuggestions(false);
|
|
if (result.retroactive_matches > 0) {
|
|
setRetroFeedback(result.retroactive_matches);
|
|
toast.success(`Rule added — ${result.retroactive_matches} past payment${result.retroactive_matches === 1 ? '' : 's'} imported`);
|
|
} else {
|
|
toast.success('Rule added — will match future transactions automatically');
|
|
}
|
|
onRulesChanged?.();
|
|
} catch (err) {
|
|
toast.error(err.message || 'Failed to add rule');
|
|
} finally {
|
|
setAdding(false);
|
|
}
|
|
}
|
|
|
|
async function handleDelete(rule) {
|
|
setDeleting(rule.id);
|
|
try {
|
|
await api.deleteMerchantRule(billId, rule.id);
|
|
setRules(prev => prev.filter(r => r.id !== rule.id));
|
|
toast.success(`Rule "${rule.merchant}" removed`);
|
|
onRulesChanged?.();
|
|
} catch (err) {
|
|
toast.error(err.message || 'Failed to remove rule');
|
|
} finally {
|
|
setDeleting(null);
|
|
}
|
|
}
|
|
|
|
function pickSuggestion(s) {
|
|
setInput(s.label);
|
|
setShowSuggestions(false);
|
|
inputRef.current?.focus();
|
|
}
|
|
|
|
// Filter suggestions: not already a rule, and not already matched to something
|
|
const filteredSuggestions = suggestions.filter(s =>
|
|
!rules.some(r => r.merchant === s.normalized) &&
|
|
(input.length < 2 || s.label.toLowerCase().includes(input.toLowerCase()))
|
|
).slice(0, 8);
|
|
|
|
if (loading) {
|
|
return (
|
|
<div className="flex items-center gap-2 px-1 py-3 text-xs text-muted-foreground">
|
|
<Loader2 className="h-3.5 w-3.5 animate-spin" />
|
|
Loading matching rules…
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className="space-y-3">
|
|
|
|
{/* Existing rules */}
|
|
{rules.length > 0 && (
|
|
<div className="flex flex-wrap gap-2">
|
|
{rules.map(rule => (
|
|
<RuleChip
|
|
key={rule.id}
|
|
rule={rule}
|
|
onDelete={handleDelete}
|
|
deleting={deleting}
|
|
/>
|
|
))}
|
|
</div>
|
|
)}
|
|
|
|
{/* Retroactive feedback */}
|
|
{retroFeedback !== null && (
|
|
<div className="flex items-center gap-2 rounded-md border border-emerald-500/25 bg-emerald-500/8 px-3 py-2 text-xs text-emerald-700 dark:text-emerald-400">
|
|
<CheckCircle2 className="h-3.5 w-3.5 shrink-0" />
|
|
{retroFeedback} existing payment{retroFeedback === 1 ? '' : 's'} imported from your transaction history.
|
|
</div>
|
|
)}
|
|
|
|
{/* Add rule input */}
|
|
<div className="relative">
|
|
<div className="flex gap-2">
|
|
<div className="relative flex-1">
|
|
<Input
|
|
ref={inputRef}
|
|
value={input}
|
|
onChange={e => { setInput(e.target.value); setShowSuggestions(true); setRetroFeedback(null); }}
|
|
onFocus={() => setShowSuggestions(true)}
|
|
onKeyDown={e => {
|
|
if (e.key === 'Enter') { e.preventDefault(); handleAdd(); }
|
|
if (e.key === 'Escape') setShowSuggestions(false);
|
|
}}
|
|
placeholder="Type merchant name or pick from recent transactions…"
|
|
className="h-8 pr-20 text-xs"
|
|
disabled={adding}
|
|
/>
|
|
<div className="absolute right-2 top-1/2 flex -translate-y-1/2 items-center gap-1">
|
|
<PreviewBadge count={previewCount} loading={previewLoading} />
|
|
</div>
|
|
</div>
|
|
<Button
|
|
type="button"
|
|
size="sm"
|
|
className="h-8 gap-1.5 px-3 text-xs"
|
|
disabled={adding || input.trim().length < 2}
|
|
onClick={() => handleAdd()}
|
|
>
|
|
{adding ? <Loader2 className="h-3.5 w-3.5 animate-spin" /> : <Plus className="h-3.5 w-3.5" />}
|
|
Add
|
|
</Button>
|
|
</div>
|
|
|
|
{/* Conflict warning */}
|
|
{conflicts.length > 0 && input.trim().length >= 2 && (
|
|
<div className="mt-1.5">
|
|
<ConflictWarning conflicts={conflicts} />
|
|
</div>
|
|
)}
|
|
|
|
{/* Suggestions dropdown */}
|
|
{showSuggestions && filteredSuggestions.length > 0 && (
|
|
<div
|
|
ref={dropdownRef}
|
|
className="absolute left-0 right-14 top-full z-50 mt-1 overflow-hidden rounded-lg border border-border/80 bg-card shadow-md"
|
|
>
|
|
<p className="border-b border-border/50 px-3 py-1.5 text-[10px] font-semibold uppercase tracking-wide text-muted-foreground">
|
|
Recent unmatched transactions
|
|
</p>
|
|
<div className="max-h-48 overflow-y-auto">
|
|
{filteredSuggestions.map(s => {
|
|
const amountVal = Math.abs(Number(s.amount || 0)) / 100;
|
|
return (
|
|
<button
|
|
key={s.id}
|
|
type="button"
|
|
className="flex w-full items-center gap-3 px-3 py-2 text-left text-xs hover:bg-muted/50 focus:bg-muted/50 focus:outline-none"
|
|
onClick={() => pickSuggestion(s)}
|
|
>
|
|
<Building2 className="h-3.5 w-3.5 shrink-0 text-muted-foreground" />
|
|
<span className="min-w-0 flex-1 truncate font-medium">{s.label}</span>
|
|
<span className="shrink-0 font-mono text-muted-foreground tabular-nums">
|
|
${amountVal.toFixed(2)}
|
|
</span>
|
|
</button>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Empty state */}
|
|
{rules.length === 0 && !input && (
|
|
<p className="text-[11px] text-muted-foreground/70">
|
|
No rules yet. Type a merchant name or pick a recent transaction above to automatically import future payments for this bill.
|
|
</p>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|