Pranav Nachanekar | afe52e8 | 2019-10-03 16:35:08 +0530 | [diff] [blame] | 1 | frappe.ready(async () => { |
Pranav Nachanekar | afe52e8 | 2019-10-03 16:35:08 +0530 | [diff] [blame] | 2 | initialise_select_date(); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 3 | }) |
Pranav Nachanekar | afe52e8 | 2019-10-03 16:35:08 +0530 | [diff] [blame] | 4 | |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 5 | window.holiday_list = []; |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 6 | |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 7 | async function initialise_select_date() { |
Pranav Nachanekar | a2dbd39 | 2019-09-12 10:48:26 +0530 | [diff] [blame] | 8 | navigate_to_page(1); |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 9 | await get_global_variables(); |
| 10 | setup_date_picker(); |
| 11 | setup_timezone_selector(); |
| 12 | hide_next_button(); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 13 | } |
| 14 | |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 15 | async function get_global_variables() { |
0Pranav | 793ba8f | 2019-11-14 11:25:49 +0530 | [diff] [blame] | 16 | // Using await through this file instead of then. |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 17 | window.appointment_settings = (await frappe.call({ |
Pranav Nachnekar | 6aa7632 | 2019-12-10 16:05:28 +0000 | [diff] [blame] | 18 | method: 'erpnext.www.book_appointment.index.get_appointment_settings' |
Pranav Nachanekar | afe52e8 | 2019-10-03 16:35:08 +0530 | [diff] [blame] | 19 | })).message; |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 20 | window.timezones = (await frappe.call({ |
Pranav Nachnekar | 6aa7632 | 2019-12-10 16:05:28 +0000 | [diff] [blame] | 21 | method:'erpnext.www.book_appointment.index.get_timezones' |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 22 | })).message; |
0Pranav | d1c530c | 2019-10-31 15:36:33 +0530 | [diff] [blame] | 23 | window.holiday_list = window.appointment_settings.holiday_list; |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 24 | } |
| 25 | |
| 26 | function setup_timezone_selector() { |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 27 | let timezones_element = document.getElementById('appointment-timezone'); |
0Pranav | 84ae2cc | 2019-12-18 15:44:04 +0530 | [diff] [blame] | 28 | let local_timezone = moment.tz.guess() |
| 29 | window.timezones.forEach(timezone => { |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 30 | let opt = document.createElement('option'); |
Pranav Nachanekar | c6da5fb | 2019-10-03 11:56:23 +0530 | [diff] [blame] | 31 | opt.value = timezone; |
0Pranav | 84ae2cc | 2019-12-18 15:44:04 +0530 | [diff] [blame] | 32 | if (timezone == local_timezone) { |
Pranav Nachanekar | c6da5fb | 2019-10-03 11:56:23 +0530 | [diff] [blame] | 33 | opt.selected = true; |
| 34 | } |
0Pranav | 84ae2cc | 2019-12-18 15:44:04 +0530 | [diff] [blame] | 35 | opt.innerHTML = timezone; |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 36 | timezones_element.appendChild(opt) |
| 37 | }); |
| 38 | } |
| 39 | |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 40 | function setup_date_picker() { |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 41 | let date_picker = document.getElementById('appointment-date'); |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 42 | let today = new Date(); |
| 43 | date_picker.min = today.toISOString().substr(0, 10); |
Pranav Nachanekar | 469247b | 2019-09-12 11:15:42 +0530 | [diff] [blame] | 44 | today.setDate(today.getDate() + window.appointment_settings.advance_booking_days); |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 45 | date_picker.max = today.toISOString().substr(0, 10); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 46 | } |
| 47 | |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 48 | function hide_next_button() { |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 49 | let next_button = document.getElementById('next-button'); |
| 50 | next_button.disabled = true; |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 51 | next_button.onclick = () => frappe.msgprint("Please select a date and time"); |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 52 | } |
| 53 | |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 54 | function show_next_button() { |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 55 | let next_button = document.getElementById('next-button'); |
| 56 | next_button.disabled = false; |
| 57 | next_button.onclick = setup_details_page; |
| 58 | } |
| 59 | |
| 60 | function on_date_or_timezone_select() { |
| 61 | let date_picker = document.getElementById('appointment-date'); |
| 62 | let timezone = document.getElementById('appointment-timezone'); |
| 63 | if (date_picker.value === '') { |
| 64 | clear_time_slots(); |
| 65 | hide_next_button(); |
| 66 | frappe.throw('Please select a date'); |
| 67 | } |
| 68 | window.selected_date = date_picker.value; |
| 69 | window.selected_timezone = timezone.value; |
| 70 | update_time_slots(date_picker.value, timezone.value); |
Pranav Nachanekar | a322b15 | 2019-09-11 14:25:26 +0530 | [diff] [blame] | 71 | let lead_text = document.getElementById('lead-text'); |
| 72 | lead_text.innerHTML = "Select Time" |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 73 | } |
| 74 | |
| 75 | async function get_time_slots(date, timezone) { |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 76 | let slots = (await frappe.call({ |
Pranav Nachnekar | 6aa7632 | 2019-12-10 16:05:28 +0000 | [diff] [blame] | 77 | method: 'erpnext.www.book_appointment.index.get_appointment_slots', |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 78 | args: { |
| 79 | date: date, |
| 80 | timezone: timezone |
| 81 | } |
| 82 | })).message; |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 83 | return slots; |
| 84 | } |
pranav nachnekar | c5b2a58 | 2019-09-03 14:16:47 +0530 | [diff] [blame] | 85 | |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 86 | async function update_time_slots(selected_date, selected_timezone) { |
| 87 | let timeslot_container = document.getElementById('timeslot-container'); |
| 88 | window.slots = await get_time_slots(selected_date, selected_timezone); |
| 89 | clear_time_slots(); |
| 90 | if (window.slots.length <= 0) { |
| 91 | let message_div = document.createElement('p'); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 92 | message_div.innerHTML = "There are no slots available on this date"; |
| 93 | timeslot_container.appendChild(message_div); |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 94 | return |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 95 | } |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 96 | window.slots.forEach((slot, index) => { |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 97 | // Get and append timeslot div |
| 98 | let timeslot_div = get_timeslot_div_layout(slot) |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 99 | timeslot_container.appendChild(timeslot_div); |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 100 | }); |
| 101 | set_default_timeslot(); |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 102 | } |
| 103 | |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 104 | function get_timeslot_div_layout(timeslot) { |
| 105 | let start_time = new Date(timeslot.time) |
| 106 | let timeslot_div = document.createElement('div'); |
| 107 | timeslot_div.classList.add('time-slot'); |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 108 | if (!timeslot.availability) { |
| 109 | timeslot_div.classList.add('unavailable') |
| 110 | } |
| 111 | timeslot_div.innerHTML = get_slot_layout(start_time); |
0Pranav | d3605d2 | 2019-12-11 15:02:23 +0530 | [diff] [blame] | 112 | timeslot_div.id = timeslot.time.substring(11, 19); |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 113 | timeslot_div.addEventListener('click', select_time); |
| 114 | return timeslot_div |
| 115 | } |
| 116 | |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 117 | function clear_time_slots() { |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 118 | // Clear any existing divs in timeslot container |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 119 | let timeslot_container = document.getElementById('timeslot-container'); |
| 120 | while (timeslot_container.firstChild) { |
Pranav Nachanekar | 7323bfd | 2019-09-18 14:33:10 +0530 | [diff] [blame] | 121 | timeslot_container.removeChild(timeslot_container.firstChild); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 122 | } |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 123 | } |
| 124 | |
| 125 | function get_slot_layout(time) { |
0Pranav | 6e6954c | 2019-11-13 16:00:59 +0530 | [diff] [blame] | 126 | let timezone = document.getElementById("appointment-timezone").value; |
Pranav Nachanekar | 7323bfd | 2019-09-18 14:33:10 +0530 | [diff] [blame] | 127 | time = new Date(time); |
0Pranav | 6e6954c | 2019-11-13 16:00:59 +0530 | [diff] [blame] | 128 | let start_time_string = moment(time).tz(timezone).format("LT"); |
| 129 | let end_time = moment(time).tz(timezone).add(window.appointment_settings.appointment_duration, 'minutes'); |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 130 | let end_time_string = end_time.format("LT"); |
| 131 | return `<span style="font-size: 1.2em;">${start_time_string}</span><br><span class="text-muted small">to ${end_time_string}</span>`; |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 132 | } |
| 133 | |
| 134 | function select_time() { |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 135 | if (this.classList.contains('unavailable')) { |
Pranav Nachanekar | 7323bfd | 2019-09-18 14:33:10 +0530 | [diff] [blame] | 136 | return; |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 137 | } |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 138 | let selected_element = document.getElementsByClassName('selected'); |
Pranav Nachanekar | 25148d0 | 2019-10-04 11:32:39 +0530 | [diff] [blame] | 139 | if (!(selected_element.length > 0)) { |
Pranav Nachanekar | 7323bfd | 2019-09-18 14:33:10 +0530 | [diff] [blame] | 140 | this.classList.add('selected'); |
| 141 | show_next_button(); |
| 142 | return; |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 143 | } |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 144 | selected_element = selected_element[0] |
Pranav Nachanekar | 7323bfd | 2019-09-18 14:33:10 +0530 | [diff] [blame] | 145 | window.selected_time = this.id; |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 146 | selected_element.classList.remove('selected'); |
| 147 | this.classList.add('selected'); |
Pranav Nachanekar | a2dbd39 | 2019-09-12 10:48:26 +0530 | [diff] [blame] | 148 | show_next_button(); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 149 | } |
| 150 | |
| 151 | function set_default_timeslot() { |
| 152 | let timeslots = document.getElementsByClassName('time-slot') |
Pranav Nachanekar | 7323bfd | 2019-09-18 14:33:10 +0530 | [diff] [blame] | 153 | // Can't use a forEach here since, we need to break the loop after a timeslot is selected |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 154 | for (let i = 0; i < timeslots.length; i++) { |
| 155 | const timeslot = timeslots[i]; |
| 156 | if (!timeslot.classList.contains('unavailable')) { |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 157 | timeslot.classList.add('selected'); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 158 | break; |
| 159 | } |
| 160 | } |
| 161 | } |
| 162 | |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 163 | function navigate_to_page(page_number) { |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 164 | let page1 = document.getElementById('select-date-time'); |
| 165 | let page2 = document.getElementById('enter-details'); |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 166 | switch (page_number) { |
| 167 | case 1: |
Pranav Nachanekar | a2dbd39 | 2019-09-12 10:48:26 +0530 | [diff] [blame] | 168 | page1.style.display = 'block'; |
| 169 | page2.style.display = 'none'; |
| 170 | break; |
| 171 | case 2: |
| 172 | page1.style.display = 'none'; |
| 173 | page2.style.display = 'block'; |
| 174 | break; |
| 175 | default: |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 176 | break; |
Pranav Nachanekar | a2dbd39 | 2019-09-12 10:48:26 +0530 | [diff] [blame] | 177 | } |
| 178 | } |
| 179 | |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 180 | function setup_details_page() { |
Pranav Nachanekar | a2dbd39 | 2019-09-12 10:48:26 +0530 | [diff] [blame] | 181 | navigate_to_page(2) |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 182 | let date_container = document.getElementsByClassName('date-span')[0]; |
| 183 | let time_container = document.getElementsByClassName('time-span')[0]; |
Pranav Nachnekar | c131270 | 2020-01-28 12:20:47 +0000 | [diff] [blame] | 184 | setup_search_params(); |
Pranav Nachanekar | 5c008ef | 2019-09-10 13:12:07 +0530 | [diff] [blame] | 185 | date_container.innerHTML = moment(window.selected_date).format("MMM Do YYYY"); |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 186 | time_container.innerHTML = moment(window.selected_time, "HH:mm:ss").format("LT"); |
pranav nachnekar | 2791054 | 2019-09-03 12:04:52 +0530 | [diff] [blame] | 187 | } |
| 188 | |
Pranav Nachnekar | c131270 | 2020-01-28 12:20:47 +0000 | [diff] [blame] | 189 | function setup_search_params() { |
| 190 | let search_params = new URLSearchParams(window.location.search); |
| 191 | let customer_name = search_params.get("name") |
| 192 | let customer_email = search_params.get("email") |
| 193 | let detail = search_params.get("details") |
| 194 | if (customer_name) { |
| 195 | let name_input = document.getElementById("customer_name"); |
| 196 | name_input.value = customer_name; |
| 197 | name_input.disabled = true; |
| 198 | } |
| 199 | if(customer_email) { |
| 200 | let email_input = document.getElementById("customer_email"); |
| 201 | email_input.value = customer_email; |
| 202 | email_input.disabled = true; |
| 203 | } |
| 204 | if(detail) { |
| 205 | let detail_input = document.getElementById("customer_notes"); |
| 206 | detail_input.value = detail; |
| 207 | detail_input.disabled = true; |
| 208 | } |
| 209 | } |
pranav nachnekar | c5b2a58 | 2019-09-03 14:16:47 +0530 | [diff] [blame] | 210 | async function submit() { |
0Pranav | a92f060 | 2019-11-13 12:13:42 +0530 | [diff] [blame] | 211 | let button = document.getElementById('submit-button'); |
| 212 | button.disabled = true; |
Pranav Nachanekar | c9cf5ae | 2019-09-24 12:08:37 +0530 | [diff] [blame] | 213 | let form = document.querySelector('#customer-form'); |
Pranav Nachanekar | 25148d0 | 2019-10-04 11:32:39 +0530 | [diff] [blame] | 214 | if (!form.checkValidity()) { |
Pranav Nachanekar | c9cf5ae | 2019-09-24 12:08:37 +0530 | [diff] [blame] | 215 | form.reportValidity(); |
0Pranav | a92f060 | 2019-11-13 12:13:42 +0530 | [diff] [blame] | 216 | button.disabled = false; |
Pranav Nachanekar | c9cf5ae | 2019-09-24 12:08:37 +0530 | [diff] [blame] | 217 | return; |
| 218 | } |
0Pranav | a92f060 | 2019-11-13 12:13:42 +0530 | [diff] [blame] | 219 | let contact = get_form_data(); |
0Pranav | 511780a | 2019-11-14 12:47:08 +0530 | [diff] [blame] | 220 | let appointment = frappe.call({ |
Pranav Nachnekar | 6aa7632 | 2019-12-10 16:05:28 +0000 | [diff] [blame] | 221 | method: 'erpnext.www.book_appointment.index.create_appointment', |
pranav nachnekar | c5b2a58 | 2019-09-03 14:16:47 +0530 | [diff] [blame] | 222 | args: { |
Pranav Nachanekar | db21f86 | 2019-09-09 17:01:40 +0530 | [diff] [blame] | 223 | 'date': window.selected_date, |
| 224 | 'time': window.selected_time, |
0Pranav | a92f060 | 2019-11-13 12:13:42 +0530 | [diff] [blame] | 225 | 'contact': contact, |
0Pranav | d1c530c | 2019-10-31 15:36:33 +0530 | [diff] [blame] | 226 | 'tz':window.selected_timezone |
0Pranav | 511780a | 2019-11-14 12:47:08 +0530 | [diff] [blame] | 227 | }, |
| 228 | callback: (response)=>{ |
| 229 | if (response.message.status == "Unverified") { |
| 230 | frappe.show_alert("Please check your email to confirm the appointment") |
| 231 | } else { |
| 232 | frappe.show_alert("Appointment Created Successfully"); |
| 233 | } |
| 234 | setTimeout(()=>{ |
| 235 | let redirect_url = "/"; |
| 236 | if (window.appointment_settings.success_redirect_url){ |
| 237 | redirect_url += window.appointment_settings.success_redirect_url; |
| 238 | } |
0Pranav | fb1e877 | 2019-11-26 12:14:41 +0530 | [diff] [blame] | 239 | window.location.href = redirect_url;},5000) |
0Pranav | 511780a | 2019-11-14 12:47:08 +0530 | [diff] [blame] | 240 | }, |
| 241 | error: (err)=>{ |
| 242 | frappe.show_alert("Something went wrong please try again"); |
| 243 | button.disabled = false; |
pranav nachnekar | c5b2a58 | 2019-09-03 14:16:47 +0530 | [diff] [blame] | 244 | } |
0Pranav | 511780a | 2019-11-14 12:47:08 +0530 | [diff] [blame] | 245 | }); |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 246 | } |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 247 | |
Pranav Nachanekar | 91a5649 | 2019-09-17 16:58:41 +0530 | [diff] [blame] | 248 | function get_form_data() { |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 249 | contact = {}; |
0Pranav | 6de68c8 | 2019-11-01 09:51:32 +0530 | [diff] [blame] | 250 | let inputs = ['name', 'skype', 'number', 'notes', 'email']; |
| 251 | inputs.forEach((id) => contact[id] = document.getElementById(`customer_${id}`).value) |
0Pranav | a92f060 | 2019-11-13 12:13:42 +0530 | [diff] [blame] | 252 | return contact |
Pranav Nachanekar | 10711dd | 2019-09-09 15:41:20 +0530 | [diff] [blame] | 253 | } |