2016-04-27 22:57:11 +03:00
|
|
|
/* ----------------------------------------------------------------------------
|
|
|
|
* Easy!Appointments - Open Source Web Scheduler
|
|
|
|
*
|
|
|
|
* @package EasyAppointments
|
|
|
|
* @author A.Tselegidis <alextselegidis@gmail.com>
|
2020-03-11 12:10:59 +03:00
|
|
|
* @copyright Copyright (c) 2013 - 2020, Alex Tselegidis
|
2016-04-27 22:57:11 +03:00
|
|
|
* @license http://opensource.org/licenses/GPL-3.0 - GPLv3
|
|
|
|
* @link http://easyappointments.org
|
|
|
|
* @since v1.0.0
|
|
|
|
* ---------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
window.FrontendBookApi = window.FrontendBookApi || {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Frontend Book API
|
|
|
|
*
|
|
|
|
* This module serves as the API consumer for the booking wizard of the app.
|
|
|
|
*
|
|
|
|
* @module FrontendBookApi
|
|
|
|
*/
|
2018-01-23 12:08:37 +03:00
|
|
|
(function (exports) {
|
2016-04-27 22:57:11 +03:00
|
|
|
'use strict';
|
|
|
|
|
2016-07-19 00:23:53 +03:00
|
|
|
var unavailableDatesBackup;
|
|
|
|
var selectedDateStringBackup;
|
2016-10-10 17:45:19 +03:00
|
|
|
var processingUnavailabilities = false;
|
2016-07-19 00:23:53 +03:00
|
|
|
|
2016-04-27 22:57:11 +03:00
|
|
|
/**
|
2016-05-14 13:09:21 +03:00
|
|
|
* Get Available Hours
|
2016-04-27 22:57:11 +03:00
|
|
|
*
|
2016-05-14 13:09:21 +03:00
|
|
|
* This function makes an AJAX call and returns the available hours for the selected service,
|
|
|
|
* provider and date.
|
|
|
|
*
|
2020-08-31 17:35:27 +03:00
|
|
|
* @param {String} selectedDate The selected date of the available hours we need.
|
2016-04-27 22:57:11 +03:00
|
|
|
*/
|
2020-08-31 17:35:27 +03:00
|
|
|
exports.getAvailableHours = function (selectedDate) {
|
2016-04-27 22:57:11 +03:00
|
|
|
$('#available-hours').empty();
|
|
|
|
|
2020-05-06 20:15:11 +03:00
|
|
|
// Find the selected service duration (it is going to be send within the "data" object).
|
|
|
|
var serviceId = $('#select-service').val();
|
|
|
|
|
|
|
|
// Default value of duration (in minutes).
|
|
|
|
var serviceDuration = 15;
|
|
|
|
|
2020-12-09 15:17:45 +03:00
|
|
|
var service = GlobalVariables.availableServices.find(function (availableService) {
|
2020-05-06 20:15:11 +03:00
|
|
|
return Number(availableService.id) === Number(serviceId);
|
2020-05-07 20:00:33 +03:00
|
|
|
});
|
2020-05-06 20:15:11 +03:00
|
|
|
|
|
|
|
if (service) {
|
|
|
|
serviceDuration = service.duration;
|
|
|
|
}
|
2016-04-27 22:57:11 +03:00
|
|
|
|
2016-05-14 13:09:21 +03:00
|
|
|
// If the manage mode is true then the appointment's start date should return as available too.
|
2020-05-06 20:15:11 +03:00
|
|
|
var appointmentId = FrontendBook.manageMode ? GlobalVariables.appointmentData.id : null;
|
2016-04-27 22:57:11 +03:00
|
|
|
|
|
|
|
// Make ajax post request and get the available hours.
|
2021-11-18 09:35:14 +03:00
|
|
|
var url = GlobalVariables.baseUrl + '/index.php/booking/ajax_get_available_hours';
|
2020-05-06 20:15:11 +03:00
|
|
|
|
|
|
|
var data = {
|
2016-07-15 21:52:21 +03:00
|
|
|
csrfToken: GlobalVariables.csrfToken,
|
|
|
|
service_id: $('#select-service').val(),
|
|
|
|
provider_id: $('#select-provider').val(),
|
2020-08-31 17:35:27 +03:00
|
|
|
selected_date: selectedDate,
|
2020-05-06 20:15:11 +03:00
|
|
|
service_duration: serviceDuration,
|
2016-07-15 21:52:21 +03:00
|
|
|
manage_mode: FrontendBook.manageMode,
|
|
|
|
appointment_id: appointmentId
|
|
|
|
};
|
2016-04-27 22:57:11 +03:00
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
$.post(url, data).done(function (response) {
|
|
|
|
// The response contains the available hours for the selected provider and service. Fill the available
|
|
|
|
// hours div with response data.
|
|
|
|
if (response.length > 0) {
|
|
|
|
var providerId = $('#select-provider').val();
|
|
|
|
|
|
|
|
if (providerId === 'any-provider') {
|
|
|
|
for (var availableProvider of GlobalVariables.availableProviders) {
|
|
|
|
if (availableProvider.services.indexOf(Number(serviceId)) !== -1) {
|
|
|
|
providerId = availableProvider.id; // Use first available provider.
|
|
|
|
break;
|
2021-01-27 15:39:12 +03:00
|
|
|
}
|
2020-05-06 20:15:11 +03:00
|
|
|
}
|
2021-11-06 19:38:37 +03:00
|
|
|
}
|
2020-04-08 11:57:18 +03:00
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
var provider = GlobalVariables.availableProviders.find(function (availableProvider) {
|
|
|
|
return Number(providerId) === Number(availableProvider.id);
|
|
|
|
});
|
2020-03-29 17:20:30 +03:00
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
if (!provider) {
|
|
|
|
throw new Error('Could not find provider.');
|
|
|
|
}
|
2020-05-06 20:15:11 +03:00
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
var providerTimezone = provider.timezone;
|
|
|
|
var selectedTimezone = $('#select-timezone').val();
|
|
|
|
var timeFormat = GlobalVariables.timeFormat === 'regular' ? 'h:mm a' : 'HH:mm';
|
2020-03-29 17:20:30 +03:00
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
response.forEach(function (availableHour) {
|
|
|
|
var availableHourMoment = moment
|
|
|
|
.tz(selectedDate + ' ' + availableHour + ':00', providerTimezone)
|
|
|
|
.tz(selectedTimezone);
|
2021-10-28 14:30:39 +03:00
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
if (availableHourMoment.format('YYYY-MM-DD') !== selectedDate) {
|
|
|
|
return; // Due to the selected timezone the available hour belongs to another date.
|
2016-04-27 22:57:11 +03:00
|
|
|
}
|
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
$('#available-hours').append(
|
|
|
|
$('<button/>', {
|
2021-11-23 10:33:43 +03:00
|
|
|
'class': 'btn btn-outline-secondary w-100 shadow-none available-hour',
|
2021-11-06 19:38:37 +03:00
|
|
|
'data': {
|
|
|
|
'value': availableHour
|
|
|
|
},
|
|
|
|
'text': availableHourMoment.format(timeFormat)
|
|
|
|
})
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (FrontendBook.manageMode) {
|
|
|
|
// Set the appointment's start time as the default selection.
|
|
|
|
$('.available-hour')
|
|
|
|
.removeClass('selected-hour')
|
|
|
|
.filter(function () {
|
|
|
|
return (
|
|
|
|
$(this).text() ===
|
2021-11-24 10:34:26 +03:00
|
|
|
moment(GlobalVariables.appointmentData.start_datetime).format(timeFormat)
|
2021-11-06 19:38:37 +03:00
|
|
|
);
|
|
|
|
})
|
|
|
|
.addClass('selected-hour');
|
|
|
|
} else {
|
|
|
|
// Set the first available hour as the default selection.
|
|
|
|
$('.available-hour:eq(0)').addClass('selected-hour');
|
2021-07-19 16:02:45 +03:00
|
|
|
}
|
2020-05-06 20:15:11 +03:00
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
FrontendBook.updateConfirmFrame();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$('.available-hour').length) {
|
2021-12-13 09:52:09 +03:00
|
|
|
$('#available-hours').text(App.Lang.no_available_hours);
|
2021-11-06 19:38:37 +03:00
|
|
|
}
|
|
|
|
});
|
2016-04-27 22:57:11 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Register an appointment to the database.
|
|
|
|
*
|
|
|
|
* This method will make an ajax call to the appointments controller that will register
|
|
|
|
* the appointment to the database.
|
|
|
|
*/
|
2018-01-23 12:08:37 +03:00
|
|
|
exports.registerAppointment = function () {
|
2016-04-27 22:57:11 +03:00
|
|
|
var $captchaText = $('.captcha-text');
|
|
|
|
|
|
|
|
if ($captchaText.length > 0) {
|
2021-11-23 12:10:09 +03:00
|
|
|
$captchaText.removeClass('is-invalid');
|
2016-04-27 22:57:11 +03:00
|
|
|
if ($captchaText.val() === '') {
|
2021-11-23 12:10:09 +03:00
|
|
|
$captchaText.addClass('is-invalid');
|
2016-04-27 22:57:11 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-09 22:28:26 +03:00
|
|
|
var formData = JSON.parse($('input[name="post_data"]').val());
|
|
|
|
|
2020-05-06 20:15:11 +03:00
|
|
|
var data = {
|
2016-07-15 21:52:21 +03:00
|
|
|
csrfToken: GlobalVariables.csrfToken,
|
|
|
|
post_data: formData
|
|
|
|
};
|
2016-04-27 22:57:11 +03:00
|
|
|
|
|
|
|
if ($captchaText.length > 0) {
|
2020-05-06 20:15:11 +03:00
|
|
|
data.captcha = $captchaText.val();
|
2016-04-27 22:57:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (GlobalVariables.manageMode) {
|
2020-05-06 20:15:11 +03:00
|
|
|
data.exclude_appointment_id = GlobalVariables.appointmentData.id;
|
2016-04-27 22:57:11 +03:00
|
|
|
}
|
|
|
|
|
2021-11-18 09:35:14 +03:00
|
|
|
var url = GlobalVariables.baseUrl + '/index.php/booking/ajax_register_appointment';
|
2020-05-06 20:15:11 +03:00
|
|
|
|
2016-07-15 21:52:21 +03:00
|
|
|
var $layer = $('<div/>');
|
2016-04-27 22:57:11 +03:00
|
|
|
|
|
|
|
$.ajax({
|
2020-05-06 20:15:11 +03:00
|
|
|
url: url,
|
2016-04-27 22:57:11 +03:00
|
|
|
method: 'post',
|
2020-05-06 20:15:11 +03:00
|
|
|
data: data,
|
2016-04-27 22:57:11 +03:00
|
|
|
dataType: 'json',
|
2018-01-23 12:08:37 +03:00
|
|
|
beforeSend: function (jqxhr, settings) {
|
2021-11-06 19:38:37 +03:00
|
|
|
$layer.appendTo('body').css({
|
|
|
|
background: 'white',
|
|
|
|
position: 'fixed',
|
|
|
|
top: '0',
|
|
|
|
left: '0',
|
|
|
|
height: '100vh',
|
|
|
|
width: '100vw',
|
|
|
|
opacity: '0.5'
|
|
|
|
});
|
2016-04-27 22:57:11 +03:00
|
|
|
}
|
|
|
|
})
|
2018-01-23 12:08:37 +03:00
|
|
|
.done(function (response) {
|
2016-04-27 22:57:11 +03:00
|
|
|
if (response.captcha_verification === false) {
|
2021-12-13 09:52:09 +03:00
|
|
|
$('#captcha-hint').text(App.Lang.captcha_is_wrong).fadeTo(400, 1);
|
2016-04-27 22:57:11 +03:00
|
|
|
|
2018-01-23 12:08:37 +03:00
|
|
|
setTimeout(function () {
|
2016-04-27 22:57:11 +03:00
|
|
|
$('#captcha-hint').fadeTo(400, 0);
|
|
|
|
}, 3000);
|
|
|
|
|
2020-12-14 11:48:36 +03:00
|
|
|
$('.captcha-title button').trigger('click');
|
2016-04-27 22:57:11 +03:00
|
|
|
|
2021-11-23 12:10:09 +03:00
|
|
|
$captchaText.addClass('is-invalid');
|
2016-04-27 22:57:11 +03:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
window.location.href =
|
2021-12-06 11:00:02 +03:00
|
|
|
GlobalVariables.baseUrl + '/index.php/booking/book_success/' + response.appointment_hash;
|
2016-04-27 22:57:11 +03:00
|
|
|
})
|
2018-01-23 12:08:37 +03:00
|
|
|
.fail(function (jqxhr, textStatus, errorThrown) {
|
2020-12-14 11:48:36 +03:00
|
|
|
$('.captcha-title button').trigger('click');
|
2016-04-27 22:57:11 +03:00
|
|
|
})
|
2018-01-23 12:08:37 +03:00
|
|
|
.always(function () {
|
2016-04-27 22:57:11 +03:00
|
|
|
$layer.remove();
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the unavailable dates of a provider.
|
|
|
|
*
|
|
|
|
* This method will fetch the unavailable dates of the selected provider and service and then it will
|
2016-05-14 13:09:21 +03:00
|
|
|
* select the first available date (if any). It uses the "FrontendBookApi.getAvailableHours" method to
|
|
|
|
* fetch the appointment* hours of the selected date.
|
2016-04-27 22:57:11 +03:00
|
|
|
*
|
2016-05-14 13:09:21 +03:00
|
|
|
* @param {Number} providerId The selected provider ID.
|
|
|
|
* @param {Number} serviceId The selected service ID.
|
|
|
|
* @param {String} selectedDateString Y-m-d value of the selected date.
|
2016-04-27 22:57:11 +03:00
|
|
|
*/
|
2018-01-23 12:08:37 +03:00
|
|
|
exports.getUnavailableDates = function (providerId, serviceId, selectedDateString) {
|
2016-10-10 17:45:19 +03:00
|
|
|
if (processingUnavailabilities) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-11-16 11:16:48 +03:00
|
|
|
if (!providerId || !serviceId) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-06 20:15:11 +03:00
|
|
|
var appointmentId = FrontendBook.manageMode ? GlobalVariables.appointmentData.id : null;
|
2018-04-23 16:58:04 +03:00
|
|
|
|
2021-11-18 09:35:14 +03:00
|
|
|
var url = GlobalVariables.baseUrl + '/index.php/booking/ajax_get_unavailable_dates';
|
2020-05-06 20:15:11 +03:00
|
|
|
|
2016-07-15 21:52:21 +03:00
|
|
|
var data = {
|
|
|
|
provider_id: providerId,
|
|
|
|
service_id: serviceId,
|
|
|
|
selected_date: encodeURIComponent(selectedDateString),
|
2018-04-23 16:58:04 +03:00
|
|
|
csrfToken: GlobalVariables.csrfToken,
|
|
|
|
manage_mode: FrontendBook.manageMode,
|
|
|
|
appointment_id: appointmentId
|
2016-07-15 21:52:21 +03:00
|
|
|
};
|
2016-04-27 22:57:11 +03:00
|
|
|
|
|
|
|
$.ajax({
|
|
|
|
url: url,
|
|
|
|
type: 'GET',
|
|
|
|
data: data,
|
|
|
|
dataType: 'json'
|
2021-11-06 19:38:37 +03:00
|
|
|
}).done(function (response) {
|
|
|
|
unavailableDatesBackup = response;
|
|
|
|
selectedDateStringBackup = selectedDateString;
|
|
|
|
applyUnavailableDates(response, selectedDateString, true);
|
|
|
|
});
|
2016-04-27 22:57:11 +03:00
|
|
|
};
|
|
|
|
|
2018-01-23 12:08:37 +03:00
|
|
|
exports.applyPreviousUnavailableDates = function () {
|
2020-05-06 20:23:49 +03:00
|
|
|
applyUnavailableDates(unavailableDatesBackup, selectedDateStringBackup);
|
2016-07-19 00:23:53 +03:00
|
|
|
};
|
|
|
|
|
2020-05-06 20:23:49 +03:00
|
|
|
function applyUnavailableDates(unavailableDates, selectedDateString, setDate) {
|
2016-07-19 00:23:53 +03:00
|
|
|
setDate = setDate || false;
|
|
|
|
|
2016-10-10 17:45:19 +03:00
|
|
|
processingUnavailabilities = true;
|
|
|
|
|
2016-07-19 00:23:53 +03:00
|
|
|
// Select first enabled date.
|
2021-11-25 10:40:48 +03:00
|
|
|
var selectedDateMoment = moment(selectedDateString);
|
|
|
|
var selectedDate = selectedDateMoment.toDate();
|
|
|
|
var numberOfDays = selectedDateMoment.daysInMonth();
|
2016-07-19 00:23:53 +03:00
|
|
|
|
2018-04-15 09:34:01 +03:00
|
|
|
if (setDate && !GlobalVariables.manageMode) {
|
2018-01-23 12:08:37 +03:00
|
|
|
for (var i = 1; i <= numberOfDays; i++) {
|
2016-07-19 00:23:53 +03:00
|
|
|
var currentDate = new Date(selectedDate.getFullYear(), selectedDate.getMonth(), i);
|
2021-11-25 10:40:48 +03:00
|
|
|
if (unavailableDates.indexOf(moment(currentDate).format('YYYY-MM-DD')) === -1) {
|
2016-07-19 00:23:53 +03:00
|
|
|
$('#select-date').datepicker('setDate', currentDate);
|
2021-11-25 10:40:48 +03:00
|
|
|
FrontendBookApi.getAvailableHours(moment(currentDate).format('YYYY-MM-DD'));
|
2016-07-19 00:23:53 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If all the days are unavailable then hide the appointments hours.
|
|
|
|
if (unavailableDates.length === numberOfDays) {
|
2021-12-13 09:52:09 +03:00
|
|
|
$('#available-hours').text(App.Lang.no_available_hours);
|
2016-07-19 00:23:53 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Grey out unavailable dates.
|
2018-01-23 12:08:37 +03:00
|
|
|
$('#select-date .ui-datepicker-calendar td:not(.ui-datepicker-other-month)').each(function (index, td) {
|
2021-11-25 10:40:48 +03:00
|
|
|
selectedDateMoment.set({day: index + 1});
|
|
|
|
if (unavailableDates.indexOf(selectedDateMoment.format('YYYY-MM-DD')) !== -1) {
|
2016-07-19 00:23:53 +03:00
|
|
|
$(td).addClass('ui-datepicker-unselectable ui-state-disabled');
|
|
|
|
}
|
|
|
|
});
|
2016-10-10 17:45:19 +03:00
|
|
|
|
|
|
|
processingUnavailabilities = false;
|
2016-07-19 00:23:53 +03:00
|
|
|
}
|
|
|
|
|
2018-06-24 18:27:16 +03:00
|
|
|
/**
|
|
|
|
* Save the user's consent.
|
|
|
|
*
|
|
|
|
* @param {Object} consent Contains user's consents.
|
|
|
|
*/
|
|
|
|
exports.saveConsent = function (consent) {
|
|
|
|
var url = GlobalVariables.baseUrl + '/index.php/consents/ajax_save_consent';
|
2020-05-06 20:15:11 +03:00
|
|
|
|
2018-06-24 18:27:16 +03:00
|
|
|
var data = {
|
|
|
|
csrfToken: GlobalVariables.csrfToken,
|
|
|
|
consent: consent
|
|
|
|
};
|
|
|
|
|
2020-12-02 20:57:49 +03:00
|
|
|
$.post(url, data);
|
2018-06-24 18:27:16 +03:00
|
|
|
};
|
|
|
|
|
2018-06-24 20:08:45 +03:00
|
|
|
/**
|
|
|
|
* Delete personal information.
|
|
|
|
*
|
|
|
|
* @param {Number} customerToken Customer unique token.
|
|
|
|
*/
|
|
|
|
exports.deletePersonalInformation = function (customerToken) {
|
|
|
|
var url = GlobalVariables.baseUrl + '/index.php/privacy/ajax_delete_personal_information';
|
2020-05-06 20:15:11 +03:00
|
|
|
|
2018-06-24 20:08:45 +03:00
|
|
|
var data = {
|
|
|
|
csrfToken: GlobalVariables.csrfToken,
|
|
|
|
customer_token: customerToken
|
|
|
|
};
|
|
|
|
|
2021-11-06 19:38:37 +03:00
|
|
|
$.post(url, data).done(function () {
|
|
|
|
window.location.href = GlobalVariables.baseUrl;
|
|
|
|
});
|
2018-06-24 20:08:45 +03:00
|
|
|
};
|
2016-04-27 22:57:11 +03:00
|
|
|
})(window.FrontendBookApi);
|