import { useEffect, useMemo, useState } from 'react'; import { Plus, Trash2, RotateCcw, Save, Download, Loader2, Search } from 'lucide-react'; import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter } from '@/components/ui/dialog'; import { Button } from '@/components/ui/button'; import { Input } from '@/components/ui/input'; import { Label } from '@/components/ui/label'; import { Checkbox } from '@/components/ui/checkbox'; import { Textarea } from '@/components/ui/textarea'; import { Tabs, TabsList, TabsTrigger, TabsContent } from '@/components/ui/tabs'; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select'; import { Combobox } from '@/components/ui/combobox'; import { cn } from '@/lib/utils'; import { GetAwardDefs, SaveAwardDefs, ResetAwardDefs, AwardFields, GetAwardReferenceMeta, UpdateAwardReferenceList, ListAwardReferences, SearchAwardReferences, SaveAwardReference, DeleteAwardReference, ImportAwardReferencesText, GetAwardPresets, ApplyAwardPreset, ListCountries, DXCCForCountry, DXCCName, PopulateBuiltinReferences, HasBuiltinReferences, } from '../../wailsjs/go/main/App'; // Above this many references the editor stops loading the whole list and // switches to search-only (mirrors Log4OM's "Too many items" behaviour). const REF_LIST_CAP = 1000; type RefMeta = { code: string; count: number; updated_at: string; can_update: boolean }; export type AwardDef = { code: string; name: string; description?: string; valid?: boolean; protected?: boolean; url?: string; download_url?: string; ref_url?: string; valid_from?: string; valid_to?: string; alias?: string; type?: string; field: string; match_by?: string; exact_match?: boolean; pattern: string; leading_str?: string; trailing_str?: string; multi?: boolean; dynamic?: boolean; add_prefixes?: string[]; dxcc_filter: number[] | null; valid_bands?: string[]; valid_modes?: string[]; emission?: string[]; confirm: string[] | null; validate?: string[] | null; grant_codes?: string; export_credit_granted?: boolean; total: number; builtin?: boolean; }; type AwardRef = { code: string; name: string; dxcc: number; group: string; subgrp: string; dxcc_list?: number[]; pattern?: string; valid: boolean; valid_from?: string; valid_to?: string; score?: number; bonus?: number; gridsquare?: string; alias?: string; }; type Preset = { key: string; name: string; field: string; dxcc: number; refs: AwardRef[] }; const AWARD_TYPES = ['QSOFIELDS', 'REFERENCE', 'DXCC', 'GRID']; const CONFIRM_SRC = [ { id: 'lotw', label: 'LoTW' }, { id: 'qsl', label: 'QSL' }, { id: 'eqsl', label: 'eQSL' }, { id: 'qrzcom', label: 'QRZ.com' }, { id: 'custom', label: 'Custom' }, ]; const BANDS = ['2190m','630m','160m','80m','60m','40m','30m','20m','17m','15m','12m','10m','6m','4m','2m','1.25m','70cm','23cm','13cm']; const MODES = ['CW','SSB','USB','LSB','AM','FM','RTTY','PSK31','FT8','FT4','JT65','JT9','MFSK','OLIVIA','DIGITALVOICE']; const EMISSIONS = ['CW', 'PHONE', 'DIGITAL']; const emptyAward = (): AwardDef => ({ code: 'NEW', name: 'New award', description: '', valid: true, type: 'QSOFIELDS', field: 'note', match_by: 'code', exact_match: true, pattern: '', dxcc_filter: null, confirm: ['lotw', 'qsl'], validate: ['lotw', 'qsl'], total: 0, }); interface Props { open: boolean; onClose: () => void; onSaved: () => void; } // Small reusable multi-toggle chip group. function Chips({ all, value, onToggle }: { all: string[]; value: string[]; onToggle: (v: string) => void }) { return (
{all.map((v) => { const on = value.includes(v); return ( ); })}
); } function Field2({ label, children }: { label: string; children: React.ReactNode }) { return (
{children}
); } // DxccFilter — pick the entities an award is scoped to by country name (like the // QSO editor), resolving each to its ADIF DXCC number. Stored as number[]. function DxccFilter({ value, onChange, countries }: { value: number[]; onChange: (v: number[]) => void; countries: string[] }) { const [names, setNames] = useState>({}); useEffect(() => { let live = true; (async () => { const miss = value.filter((n) => names[n] === undefined); if (miss.length === 0) return; const got: Record = {}; for (const n of miss) { try { got[n] = await DXCCName(n); } catch { got[n] = ''; } } if (live) setNames((m) => ({ ...m, ...got })); })(); return () => { live = false; }; }, [value]); // eslint-disable-line react-hooks/exhaustive-deps async function addCountry(name: string) { const n = await DXCCForCountry(name); if (n && n > 0 && !value.includes(n)) { setNames((m) => ({ ...m, [n]: name })); onChange([...value, n]); } } return (
{value.length > 0 && (
{value.map((n) => ( #{n} {names[n] || '…'} ))}
)}
); } export function AwardEditor({ open, onClose, onSaved }: Props) { const [defs, setDefs] = useState([]); const [fields, setFields] = useState([]); const [meta, setMeta] = useState>({}); const [presets, setPresets] = useState([]); const [countries, setCountries] = useState([]); const [sel, setSel] = useState(0); const [search, setSearch] = useState(''); const [updating, setUpdating] = useState(null); const [err, setErr] = useState(''); const loadMeta = () => GetAwardReferenceMeta() .then((m) => setMeta(Object.fromEntries(((m ?? []) as RefMeta[]).map((x) => [x.code.toUpperCase(), x])))) .catch(() => {}); useEffect(() => { if (!open) return; setErr(''); Promise.all([GetAwardDefs(), AwardFields(), GetAwardPresets(), ListCountries()]) .then(([d, f, p, c]) => { setDefs((d ?? []) as any); setFields((f ?? []) as any); setPresets((p ?? []) as any); setCountries((c ?? []) as any); }) .catch((e) => setErr(String(e?.message ?? e))); loadMeta(); }, [open]); const cur = defs[sel]; const patch = (p: Partial) => setDefs((ds) => ds.map((d, j) => (j === sel ? { ...d, ...p } : d))); const toggleIn = (key: keyof AwardDef, v: string) => { const arr = ((cur?.[key] as string[]) ?? []); patch({ [key]: arr.includes(v) ? arr.filter((x) => x !== v) : [...arr, v] } as any); }; function addAward() { setDefs((ds) => [...ds, emptyAward()]); setSel(defs.length); } function removeAward(i: number) { setDefs((ds) => ds.filter((_, j) => j !== i)); setSel((s) => Math.max(0, s >= i ? s - 1 : s)); } async function save() { setErr(''); try { const clean = defs.filter((d) => d.code.trim()) .map((d) => ({ ...d, code: d.code.trim().toUpperCase(), confirm: d.confirm ?? [], validate: d.validate ?? [] })); await SaveAwardDefs(clean as any); onSaved(); onClose(); } catch (e: any) { setErr(String(e?.message ?? e)); } } async function reset() { try { setDefs((await ResetAwardDefs()) as any); setSel(0); } catch (e: any) { setErr(String(e?.message ?? e)); } } async function updateList(code: string) { setUpdating(code); setErr(''); try { await UpdateAwardReferenceList(code); await loadMeta(); } catch (e: any) { setErr(`${code}: ${String(e?.message ?? e)}`); } finally { setUpdating(null); } } const filtered = useMemo(() => { const q = search.trim().toUpperCase(); return defs.map((d, i) => ({ d, i })).filter(({ d }) => !q || d.code.toUpperCase().includes(q) || d.name.toUpperCase().includes(q)); }, [defs, search]); return ( { if (!o) onClose(); }}> Award management
{/* Left: award list */}
setSearch(e.target.value)} />
{filtered.map(({ d, i }) => ( ))}
{/* Right: tabbed editor for selected award */}
{err &&
{err}
} {!cur ? (
Select or create an award.
) : ( Award info Award type Confirmation References
{/* ── Award info ── */}
patch({ code: e.target.value })} placeholder="CODE" /> patch({ name: e.target.value })} placeholder="Award name" />
patch({ description: e.target.value })} /> patch({ url: e.target.value })} /> patch({ ref_url: e.target.value })} placeholder="https://…/" />
patch({ valid_from: e.target.value })} /> patch({ valid_to: e.target.value })} />
patch({ dxcc_filter: v })} countries={countries} />
toggleIn('valid_bands', v)} />
toggleIn('emission', v)} />
toggleIn('valid_modes', v)} />
{/* ── Award type ── */}

QSO parameters (used by QSOFIELDS / REFERENCE types)

{['code', 'description', 'pattern'].map((m) => ( ))}
patch({ pattern: e.target.value })} placeholder="group 1 = reference (for match-by pattern / dynamic)" />
patch({ leading_str: e.target.value })} /> patch({ trailing_str: e.target.value })} />
{/* ── Confirmation ── */}
{CONFIRM_SRC.map((c) => ( ))}
{CONFIRM_SRC.map((c) => ( ))}
patch({ grant_codes: e.target.value })} />
{/* ── References ── */} updateList(cur.code.toUpperCase())} updating={updating === cur.code.toUpperCase()} onChanged={loadMeta} setErr={setErr} />
)}
); } // ReferencesPanel — manage the reference list of one award: search/list on the // left, a per-reference editor on the right, plus bulk paste/CSV, presets and // the online updater (POTA/SOTA/WWFF). function ReferencesPanel({ code, presets, meta, onUpdateOnline, updating, onChanged, setErr }: { code: string; presets: Preset[]; meta?: RefMeta; onUpdateOnline: () => void; updating: boolean; onChanged: () => void; setErr: (s: string) => void; }) { const [refs, setRefs] = useState([]); const [q, setQ] = useState(''); const [selCode, setSelCode] = useState(null); const [busy, setBusy] = useState(false); const [bulk, setBulk] = useState(''); const [showBulk, setShowBulk] = useState(false); const [hasBuiltin, setHasBuiltin] = useState(false); const total = meta?.count ?? 0; const large = total > REF_LIST_CAP; // Small lists are loaded whole and filtered client-side; large lists (POTA, // 85k parks) are search-only to stay responsive. const load = () => { if (!code) return; if (total > REF_LIST_CAP) { setRefs([]); return; } setBusy(true); ListAwardReferences(code).then((r) => setRefs((r ?? []) as any)).catch(() => {}).finally(() => setBusy(false)); }; useEffect(load, [code, total]); useEffect(() => { HasBuiltinReferences(code).then(setHasBuiltin).catch(() => setHasBuiltin(false)); }, [code]); // Server-side search for large lists (debounced, min 2 chars). useEffect(() => { if (!large) return; const s = q.trim(); if (s.length < 2) { setRefs([]); return; } const t = window.setTimeout(() => { setBusy(true); SearchAwardReferences(code, s, 0, 200).then((r) => setRefs((r ?? []) as any)).catch(() => setRefs([])).finally(() => setBusy(false)); }, 200); return () => window.clearTimeout(t); }, [code, q, large]); async function populateBuiltin() { try { const n = await PopulateBuiltinReferences(code); load(); onChanged(); setErr(`Populated ${n} built-in references.`); } catch (e: any) { setErr(String(e?.message ?? e)); } } const sel = refs.find((r) => r.code === selCode) || null; // Large lists are already filtered by the server; small lists filter locally. const filtered = useMemo(() => { if (large) return refs; const s = q.trim().toUpperCase(); return refs.filter((r) => !s || r.code.toUpperCase().includes(s) || (r.name ?? '').toUpperCase().includes(s)); }, [refs, q, large]); const patchSel = (p: Partial) => setRefs((rs) => rs.map((r) => (r.code === selCode ? { ...r, ...p } : r))); async function saveRef(r: AwardRef) { try { await SaveAwardReference(code, r as any); load(); onChanged(); } catch (e: any) { setErr(String(e?.message ?? e)); } } async function addRef() { const c = prompt('New reference code:')?.trim().toUpperCase(); if (!c) return; const r: AwardRef = { code: c, name: '', dxcc: 0, group: '', subgrp: '', valid: true }; await saveRef(r); setSelCode(c); } async function delRef(c: string) { try { await DeleteAwardReference(code, c); setSelCode(null); load(); onChanged(); } catch (e: any) { setErr(String(e?.message ?? e)); } } async function applyPreset(key: string) { if (!key) return; try { await ApplyAwardPreset(code, key); load(); onChanged(); } catch (e: any) { setErr(String(e?.message ?? e)); } } async function importBulk() { try { const n = await ImportAwardReferencesText(code, bulk); setBulk(''); setShowBulk(false); load(); onChanged(); setErr(`Imported ${n} references.`); } catch (e: any) { setErr(String(e?.message ?? e)); } } return (
{/* Toolbar */}
Reference count: {total.toLocaleString()}
{hasBuiltin && ( )} {meta?.can_update && ( )}
{showBulk && (

One reference per line: CODE,Description,Group,Subgroup,DXCC (comma/semicolon/tab). Replaces the whole list.