/* ClubLedger – bar page */ let barMember = null; (async function init() { await loadConfig(); await loadStaffInto('barStaff'); document.getElementById('barSearch').addEventListener('keydown', e => { if (e.key === 'Enter') barSearchMembers(); }); })(); // --------------------------------------------------------------------------- // Member selection // --------------------------------------------------------------------------- async function barSearchMembers() { const q = document.getElementById('barSearch').value.trim(); const url = q ? `/members?q=${encodeURIComponent(q)}` : '/members'; try { const members = await apiFetch(url); const list = document.getElementById('barMemberList'); list.innerHTML = members.map(m => `
${esc(m.name)}
#${esc(m.member_number)}
${esc(m.balance_display)}
`).join(''); } catch (e) { console.error(e); } } function selectBarMember(id, name, number, balance, balanceDisplay) { barMember = { id, name, number }; document.getElementById('barMemberList').innerHTML = ''; document.getElementById('barSelected').innerHTML = `${esc(name)}   #${esc(number)}   Balance: ${esc(balanceDisplay)}`; document.getElementById('barForm').classList.remove('hidden'); document.getElementById('barProductSearch').value = ''; document.getElementById('barProductResults').innerHTML = ''; setMsg('barMsg', '', ''); } function clearBarSelection() { barMember = null; document.getElementById('barForm').classList.add('hidden'); document.getElementById('barAmount').value = ''; document.getElementById('barPin').value = ''; document.getElementById('barNote').value = ''; document.getElementById('barProductSearch').value = ''; document.getElementById('barProductResults').innerHTML = ''; setMsg('barMsg', '', ''); } // --------------------------------------------------------------------------- // Product search // --------------------------------------------------------------------------- let productTimer = null; async function barProductLookup() { clearTimeout(productTimer); productTimer = setTimeout(async () => { const q = document.getElementById('barProductSearch').value.trim(); if (!q) { document.getElementById('barProductResults').innerHTML = ''; return; } try { const products = await apiFetch(`/products?q=${encodeURIComponent(q)}`); const div = document.getElementById('barProductResults'); if (!products.length) { div.innerHTML = '
No products found
'; return; } div.innerHTML = products.map(p => `
${esc(p.name)}${p.brand ? ` – ${esc(p.brand)}` : ''} ${p.search_tags ? `
${esc(p.search_tags)}
` : ''}
${esc(p.price_display)} ${p.member_price_display ? `mbr: ${esc(p.member_price_display)}` : ''}
`).join(''); } catch (e) { console.error(e); } }, 250); } function selectProduct(price, memberPrice, label) { document.getElementById('barAmount').value = memberPrice; document.getElementById('barNote').value = label; document.getElementById('barProductResults').innerHTML = ''; document.getElementById('barProductSearch').value = ''; } // --------------------------------------------------------------------------- // Charge // --------------------------------------------------------------------------- async function doCharge() { if (!barMember) return; const amount = parseInt(document.getElementById('barAmount').value, 10); const pin = document.getElementById('barPin').value; const staff = document.getElementById('barStaff').value; const note = document.getElementById('barNote').value.trim(); if (!amount || isNaN(amount) || amount <= 0) { setMsg('barMsg', 'Enter a valid amount.', 'err'); return; } if (!pin) { setMsg('barMsg', 'PIN required.', 'err'); return; } if (!staff) { setMsg('barMsg', 'Select a staff member.', 'err'); return; } try { const r = await apiFetch('/charge', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ member_id: barMember.id, amount, pin, staff_name: staff, note: note || null }) }); window.open(`/receipt/${r.entry_id}`, '_blank'); setMsg('barMsg', `Charge complete. New balance: ${r.new_balance_display}`, 'ok'); clearBarSelection(); } catch (e) { setMsg('barMsg', e.message, 'err'); } }