/* global usScheduler */ (function () { 'use strict'; const app = document.getElementById('us-booking-app'); if (!app) return; const slotList = document.getElementById('us-slot-list'); const confirm = document.getElementById('us-booking-confirmation'); const errorBox = document.getElementById('us-booking-error'); const { restUrl, nonce } = usScheduler; function apiFetch(path, options = {}) { return fetch(restUrl + path, { ...options, headers: { 'Content-Type': 'application/json', 'X-WP-Nonce': nonce, ...(options.headers || {}), }, }).then(async (res) => { const data = await res.json(); if (!res.ok) throw new Error(data.message || 'Request failed'); return data; }); } function showError(message) { errorBox.textContent = message; errorBox.style.display = 'block'; } function clearError() { errorBox.style.display = 'none'; } function escHtml(str) { return String(str) .replace(/&/g, '&') .replace(//g, '>') .replace(/"/g, '"'); } const dayKey = (dt) => String(dt).slice(0, 10); const timeOf = (dt) => String(dt).slice(11, 16); function dayLabel(key) { const date = new Date(key + 'T00:00:00'); if (Number.isNaN(date.getTime())) return key; return date.toLocaleDateString(undefined, { weekday: 'long', year: 'numeric', month: 'long', day: 'numeric', }); } function groupByDay(slots) { const groups = new Map(); slots.forEach((slot) => { const key = dayKey(slot.start_dt); if (!groups.has(key)) groups.set(key, []); groups.get(key).push(slot); }); return [...groups.entries()].sort((a, b) => a[0].localeCompare(b[0])); } // Agenda-style calendar: available slots grouped by day. function renderSlots(slots) { if (!slots.length) { slotList.innerHTML = '

No available lesson slots at this time.

'; return; } slotList.innerHTML = groupByDay(slots).map(([key, daySlots]) => `

${escHtml(dayLabel(key))}

${daySlots.map((slot) => `
${escHtml(timeOf(slot.start_dt))}โ€“${escHtml(timeOf(slot.end_dt))} (${escHtml(String(slot.duration_minutes))} min)
`).join('')}
`).join(''); slotList.querySelectorAll('.us-book-btn').forEach((btn) => { const slot = slots.find((s) => String(s.id) === btn.dataset.slotId); btn.addEventListener('click', () => openRegistration(slot)); }); } function questionField(q) { const name = `q_${q.id}`; const required = q.is_required ? 'required' : ''; let input; if (q.field_type === 'textarea') { input = ``; } else if (q.field_type === 'select') { const opts = (q.options || []).map((o) => ``).join(''); input = ``; } else if (q.field_type === 'checkbox') { input = ``; } else { input = ``; } return `

`; } function policyField(p) { return `

${escHtml(p.title)}

${p.body || ''}
`; } function openRegistration(slot) { clearError(); const offeringId = Number(slot.offering_id) || 0; const qPath = offeringId ? `offerings/${offeringId}/questions` : null; Promise.all([ qPath ? apiFetch(qPath) : Promise.resolve([]), apiFetch('policies?scope=booking'), ]) .then(([questions, policies]) => { renderRegistration(slot, offeringId, questions, policies); }) .catch((err) => showError(err.message)); } function renderRegistration(slot, offeringId, questions, policies) { const weekly = slot.recurrence_group ? `

` : ''; slotList.innerHTML = `

${escHtml(dayLabel(dayKey(slot.start_dt)))} ยท ${escHtml(timeOf(slot.start_dt))}โ€“${escHtml(timeOf(slot.end_dt))}

${questions.map(questionField).join('')} ${policies.map(policyField).join('')} ${weekly}

`; document.getElementById('us-cancel').addEventListener('click', loadSlots); document.getElementById('us-register-form').addEventListener('submit', (e) => { e.preventDefault(); submitBooking(e.target, slot, offeringId, questions); }); } function submitBooking(form, slot, offeringId, questions) { clearError(); const answers = {}; questions.forEach((q) => { const field = form.elements[`q_${q.id}`]; if (!field) return; answers[q.id] = field.type === 'checkbox' ? (field.checked ? '1' : '0') : field.value; }); const accepted = [...form.querySelectorAll('.us-policy-accept:checked')].map((c) => Number(c.value)); const weeklyEl = document.getElementById('us-weekly'); apiFetch('bookings', { method: 'POST', body: JSON.stringify({ slot_id: slot.id, offering_id: offeringId, recurrence: weeklyEl && weeklyEl.checked ? 'weekly' : 'single', answers, accepted_policy_version_ids: accepted, }), }) .then(() => { slotList.style.display = 'none'; confirm.style.display = 'block'; }) .catch((err) => showError(err.message)); } function loadSlots() { clearError(); slotList.style.display = 'block'; confirm.style.display = 'none'; apiFetch('availability') .then(renderSlots) .catch((err) => showError(err.message)); } loadSlots(); }());