easyappointments/assets/js/frontend_book.js

824 lines
31 KiB
JavaScript
Raw Normal View History

2015-07-20 22:41:24 +03:00
/* ----------------------------------------------------------------------------
* Easy!Appointments - Open Source Web Scheduler
*
2015-07-20 22:41:24 +03:00
* @package EasyAppointments
* @author A.Tselegidis <alextselegidis@gmail.com>
* @copyright Copyright (c) 2013 - 2020, Alex Tselegidis
* @license http://opensource.org/licenses/GPL-3.0 - GPLv3
2015-07-20 22:41:24 +03:00
* @link http://easyappointments.org
* @since v1.0.0
* ---------------------------------------------------------------------------- */
2016-04-26 22:51:16 +03:00
window.FrontendBook = window.FrontendBook || {};
/**
2016-04-26 22:51:16 +03:00
* Frontend Book
*
* This module contains functions that implement the book appointment page functionality. Once the
* initialize() method is called the page is fully functional and can serve the appointment booking
* process.
*
2016-04-26 22:51:16 +03:00
* @module FrontendBook
*/
2018-01-23 12:08:37 +03:00
(function (exports) {
2016-04-26 22:51:16 +03:00
'use strict';
/**
* Contains terms and conditions consent.
*
* @type {Object}
*/
var termsAndConditionsConsent;
/**
* Contains privacy policy consent.
*
* @type {Object}
*/
var privacyPolicyConsent;
/**
* Determines the functionality of the page.
*
* @type {Boolean}
*/
2016-04-26 22:51:16 +03:00
exports.manageMode = false;
/**
* This method initializes the book appointment page.
*
* @param {Boolean} defaultEventHandlers (OPTIONAL) Determines whether the default
2016-10-10 19:29:48 +03:00
* event handlers will be bound to the dom elements.
* @param {Boolean} manageMode (OPTIONAL) Determines whether the customer is going
* to make changes to an existing appointment rather than booking a new one.
*/
exports.initialize = function (defaultEventHandlers, manageMode) {
defaultEventHandlers = defaultEventHandlers || true;
2016-04-26 22:51:16 +03:00
manageMode = manageMode || false;
if (GlobalVariables.displayCookieNotice) {
cookieconsent.initialise({
palette: {
popup: {
background: '#ffffffbd',
text: '#666666'
},
button: {
background: '#429a82',
text: '#ffffff'
}
},
content: {
message: EALang.website_using_cookies_to_ensure_best_experience,
dismiss: 'OK'
}
});
$('.cc-link').replaceWith(
$('<a/>', {
'data-toggle': 'modal',
'data-target': '#cookie-notice-modal',
'href': '#',
'class': 'cc-link',
'text': $('.cc-link').text()
})
);
}
2016-01-01 22:57:44 +02:00
FrontendBook.manageMode = manageMode;
// Initialize page's components (tooltips, datepickers etc).
2020-10-22 12:29:09 +03:00
tippy('[data-tippy-content]');
var weekDayId = GeneralFunctions.getWeekDayId(GlobalVariables.firstWeekday);
$('#select-date').datepicker({
dateFormat: 'dd-mm-yy',
firstDay: weekDayId,
minDate: 0,
defaultDate: moment().toDate(),
dayNames: [
EALang.sunday,
EALang.monday,
EALang.tuesday,
EALang.wednesday,
EALang.thursday,
EALang.friday,
EALang.saturday
],
dayNamesShort: [
EALang.sunday.substr(0, 3),
EALang.monday.substr(0, 3),
EALang.tuesday.substr(0, 3),
EALang.wednesday.substr(0, 3),
EALang.thursday.substr(0, 3),
EALang.friday.substr(0, 3),
EALang.saturday.substr(0, 3)
],
dayNamesMin: [
EALang.sunday.substr(0, 2),
EALang.monday.substr(0, 2),
EALang.tuesday.substr(0, 2),
EALang.wednesday.substr(0, 2),
EALang.thursday.substr(0, 2),
EALang.friday.substr(0, 2),
EALang.saturday.substr(0, 2)
],
monthNames: [
EALang.january,
EALang.february,
EALang.march,
EALang.april,
EALang.may,
EALang.june,
EALang.july,
EALang.august,
EALang.september,
EALang.october,
EALang.november,
EALang.december
],
prevText: EALang.previous,
nextText: EALang.next,
currentText: EALang.now,
closeText: EALang.close,
2018-01-23 12:08:37 +03:00
onSelect: function (dateText, instance) {
2021-11-25 10:40:48 +03:00
FrontendBookApi.getAvailableHours(moment($(this).datepicker('getDate')).format('YYYY-MM-DD'));
FrontendBook.updateConfirmFrame();
},
2018-01-23 12:08:37 +03:00
onChangeMonthYear: function (year, month, instance) {
var currentDate = new Date(year, month - 1, 1);
2021-11-25 10:40:48 +03:00
FrontendBookApi.getUnavailableDates(
$('#select-provider').val(),
$('#select-service').val(),
2021-11-25 10:40:48 +03:00
moment(currentDate).format('YYYY-MM-DD')
);
}
});
2020-03-29 17:20:30 +03:00
$('#select-timezone').val(Intl.DateTimeFormat().resolvedOptions().timeZone);
// Bind the event handlers (might not be necessary every time we use this class).
if (defaultEventHandlers) {
bindEventHandlers();
}
// If the manage mode is true, the appointments data should be loaded by default.
if (FrontendBook.manageMode) {
applyAppointmentData(
GlobalVariables.appointmentData,
GlobalVariables.providerData,
GlobalVariables.customerData
);
} else {
var $selectProvider = $('#select-provider');
2018-01-23 12:08:37 +03:00
var $selectService = $('#select-service');
// Check if a specific service was selected (via URL parameter).
var selectedServiceId = GeneralFunctions.getUrlParameter(location.href, 'service');
if (selectedServiceId && $selectService.find('option[value="' + selectedServiceId + '"]').length > 0) {
$selectService.val(selectedServiceId);
}
$selectService.trigger('change'); // Load the available hours.
2020-03-29 17:20:30 +03:00
// Check if a specific provider was selected.
var selectedProviderId = GeneralFunctions.getUrlParameter(location.href, 'provider');
2018-01-23 12:08:37 +03:00
if (selectedProviderId && $selectProvider.find('option[value="' + selectedProviderId + '"]').length === 0) {
2020-03-29 17:20:30 +03:00
// Select a service of this provider in order to make the provider available in the select box.
for (var index in GlobalVariables.availableProviders) {
2018-01-23 12:08:37 +03:00
var provider = GlobalVariables.availableProviders[index];
if (provider.id === selectedProviderId && provider.services.length > 0) {
$selectService.val(provider.services[0]).trigger('change');
}
}
}
if (selectedProviderId && $selectProvider.find('option[value="' + selectedProviderId + '"]').length > 0) {
$selectProvider.val(selectedProviderId).trigger('change');
}
}
2016-04-26 22:51:16 +03:00
};
/**
2016-04-26 22:51:16 +03:00
* This method binds the necessary event handlers for the book appointments page.
*/
function bindEventHandlers() {
2020-03-29 17:20:30 +03:00
/**
* Event: Timezone "Changed"
*/
$('#select-timezone').on('change', function () {
var date = $('#select-date').datepicker('getDate');
if (!date) {
return;
}
2021-11-25 10:40:48 +03:00
FrontendBookApi.getAvailableHours(moment(date).format('YYYY-MM-DD'));
2020-03-29 17:20:30 +03:00
FrontendBook.updateConfirmFrame();
});
/**
* Event: Selected Provider "Changed"
*
* Whenever the provider changes the available appointment date - time periods must be updated.
*/
$('#select-provider').on('change', function () {
FrontendBookApi.getUnavailableDates(
$(this).val(),
$('#select-service').val(),
2021-11-25 10:40:48 +03:00
moment($('#select-date').datepicker('getDate')).format('YYYY-MM-DD')
);
FrontendBook.updateConfirmFrame();
});
/**
* Event: Selected Service "Changed"
*
* When the user clicks on a service, its available providers should
* become visible.
*/
$('#select-service').on('change', function () {
var serviceId = $('#select-service').val();
$('#select-provider').empty();
GlobalVariables.availableProviders.forEach(function (provider) {
// If the current provider is able to provide the selected service, add him to the list box.
var canServeService =
provider.services.filter(function (providerServiceId) {
return Number(providerServiceId) === Number(serviceId);
}).length > 0;
if (canServeService) {
$('#select-provider').append(
new Option(provider.first_name + ' ' + provider.last_name, provider.id)
);
}
});
// Add the "Any Provider" entry.
if ($('#select-provider option').length >= 1 && GlobalVariables.displayAnyProvider === '1') {
$('#select-provider').prepend(
new Option('- ' + EALang.any_provider + ' -', 'any-provider', true, true)
);
}
FrontendBookApi.getUnavailableDates(
$('#select-provider').val(),
$(this).val(),
2021-11-25 10:40:48 +03:00
moment($('#select-date').datepicker('getDate')).format('YYYY-MM-DD')
);
FrontendBook.updateConfirmFrame();
updateServiceDescription(serviceId);
});
/**
* Event: Next Step Button "Clicked"
*
* This handler is triggered every time the user pressed the "next" button on the book wizard.
2016-10-10 19:29:48 +03:00
* Some special tasks might be performed, depending the current wizard step.
*/
$('.button-next').on('click', function () {
// If we are on the first step and there is not provider selected do not continue with the next step.
if ($(this).attr('data-step_index') === '1' && !$('#select-provider').val()) {
return;
}
// If we are on the 2nd tab then the user should have an appointment hour selected.
if ($(this).attr('data-step_index') === '2') {
if (!$('.selected-hour').length) {
if (!$('#select-hour-prompt').length) {
$('<div/>', {
'id': 'select-hour-prompt',
'class': 'text-danger mb-4',
'text': EALang.appointment_hour_missing
}).prependTo('#available-hours');
}
return;
}
}
// If we are on the 3rd tab then we will need to validate the user's input before proceeding to the next
// step.
if ($(this).attr('data-step_index') === '3') {
if (!validateCustomerForm()) {
return; // Validation failed, do not continue.
} else {
FrontendBook.updateConfirmFrame();
var $acceptToTermsAndConditions = $('#accept-to-terms-and-conditions');
if ($acceptToTermsAndConditions.length && $acceptToTermsAndConditions.prop('checked') === true) {
var newTermsAndConditionsConsent = {
first_name: $('#first-name').val(),
last_name: $('#last-name').val(),
email: $('#email').val(),
type: 'terms-and-conditions'
};
if (
JSON.stringify(newTermsAndConditionsConsent) !== JSON.stringify(termsAndConditionsConsent)
) {
termsAndConditionsConsent = newTermsAndConditionsConsent;
FrontendBookApi.saveConsent(termsAndConditionsConsent);
}
}
var $acceptToPrivacyPolicy = $('#accept-to-privacy-policy');
if ($acceptToPrivacyPolicy.length && $acceptToPrivacyPolicy.prop('checked') === true) {
var newPrivacyPolicyConsent = {
first_name: $('#first-name').val(),
last_name: $('#last-name').val(),
email: $('#email').val(),
type: 'privacy-policy'
};
if (JSON.stringify(newPrivacyPolicyConsent) !== JSON.stringify(privacyPolicyConsent)) {
privacyPolicyConsent = newPrivacyPolicyConsent;
FrontendBookApi.saveConsent(privacyPolicyConsent);
}
}
}
}
// Display the next step tab (uses jquery animation effect).
var nextTabIndex = parseInt($(this).attr('data-step_index')) + 1;
$(this)
.parents()
.eq(1)
.hide('fade', function () {
$('.active-step').removeClass('active-step');
$('#step-' + nextTabIndex).addClass('active-step');
$('#wizard-frame-' + nextTabIndex).show('fade');
});
});
/**
* Event: Back Step Button "Clicked"
*
* This handler is triggered every time the user pressed the "back" button on the
* book wizard.
*/
$('.button-back').on('click', function () {
var prevTabIndex = parseInt($(this).attr('data-step_index')) - 1;
$(this)
.parents()
.eq(1)
.hide('fade', function () {
$('.active-step').removeClass('active-step');
$('#step-' + prevTabIndex).addClass('active-step');
$('#wizard-frame-' + prevTabIndex).show('fade');
});
});
/**
* Event: Available Hour "Click"
*
* Triggered whenever the user clicks on an available hour
* for his appointment.
*/
2018-01-23 12:08:37 +03:00
$('#available-hours').on('click', '.available-hour', function () {
$('.selected-hour').removeClass('selected-hour');
$(this).addClass('selected-hour');
FrontendBook.updateConfirmFrame();
});
if (FrontendBook.manageMode) {
/**
* Event: Cancel Appointment Button "Click"
*
* When the user clicks the "Cancel" button this form is going to be submitted. We need
* the user to confirm this action because once the appointment is cancelled, it will be
* delete from the database.
*
* @param {jQuery.Event} event
*/
$('#cancel-appointment').on('click', function (event) {
var buttons = [
{
text: EALang.cancel,
click: function () {
$('#message-box').dialog('close');
}
},
{
text: 'OK',
2018-01-23 12:08:37 +03:00
click: function () {
if ($('#cancel-reason').val() === '') {
$('#cancel-reason').css('border', '2px solid #DC3545');
return;
}
$('#cancel-appointment-form textarea').val($('#cancel-reason').val());
$('#cancel-appointment-form').submit();
}
}
];
GeneralFunctions.displayMessageBox(
EALang.cancel_appointment_title,
EALang.write_appointment_removal_reason,
buttons
);
$('<textarea/>', {
'class': 'form-control',
'id': 'cancel-reason',
'rows': '3',
'css': {
'width': '100%'
}
}).appendTo('#message-box');
2014-01-04 00:26:10 +02:00
return false;
});
$('#delete-personal-information').on('click', function () {
var buttons = [
{
text: EALang.cancel,
click: function () {
$('#message-box').dialog('close');
}
},
{
text: EALang.delete,
click: function () {
FrontendBookApi.deletePersonalInformation(GlobalVariables.customerToken);
}
}
];
GeneralFunctions.displayMessageBox(
EALang.delete_personal_information,
EALang.delete_personal_information_prompt,
buttons
);
});
}
/**
* Event: Book Appointment Form "Submit"
*
* Before the form is submitted to the server we need to make sure that
* in the meantime the selected appointment date/time wasn't reserved by
* another customer or event.
*
* @param {jQuery.Event} event
*/
$('#book-appointment-submit').on('click', function () {
FrontendBookApi.registerAppointment();
});
/**
* Event: Refresh captcha image.
*
* @param {jQuery.Event} event
*/
$('.captcha-title button').on('click', function (event) {
$('.captcha-image').attr('src', GlobalVariables.baseUrl + '/index.php/captcha?' + Date.now());
});
2018-01-23 12:08:37 +03:00
$('#select-date').on('mousedown', '.ui-datepicker-calendar td', function (event) {
setTimeout(function () {
FrontendBookApi.applyPreviousUnavailableDates(); // New jQuery UI version will replace the td elements.
}, 300); // There is no draw event unfortunately.
});
}
/**
2016-10-10 19:29:48 +03:00
* This function validates the customer's data input. The user cannot continue
* without passing all the validation checks.
*
* @return {Boolean} Returns the validation result.
*/
function validateCustomerForm() {
$('#wizard-frame-3 .is-invalid').removeClass('is-invalid');
$('#wizard-frame-3 label.text-danger').removeClass('text-danger');
try {
// Validate required fields.
var missingRequiredField = false;
$('.required').each(function (index, requiredField) {
if (!$(requiredField).val()) {
$(requiredField).parents('.form-group').addClass('is-invalid');
missingRequiredField = true;
}
});
if (missingRequiredField) {
throw new Error(EALang.fields_are_required);
}
var $acceptToTermsAndConditions = $('#accept-to-terms-and-conditions');
if ($acceptToTermsAndConditions.length && !$acceptToTermsAndConditions.prop('checked')) {
$acceptToTermsAndConditions.parents('.form-check').addClass('text-danger');
throw new Error(EALang.fields_are_required);
}
var $acceptToPrivacyPolicy = $('#accept-to-privacy-policy');
if ($acceptToPrivacyPolicy.length && !$acceptToPrivacyPolicy.prop('checked')) {
$acceptToPrivacyPolicy.parents('.form-check').addClass('text-danger');
throw new Error(EALang.fields_are_required);
}
// Validate email address.
if (!GeneralFunctions.validateEmail($('#email').val())) {
$('#email').parents('.form-group').addClass('is-invalid');
throw new Error(EALang.invalid_email);
}
return true;
} catch (error) {
$('#form-message').text(error.message);
return false;
}
2016-04-26 22:51:16 +03:00
}
/**
* Every time this function is executed, it updates the confirmation page with the latest
2016-10-10 19:29:48 +03:00
* customer settings and input for the appointment booking.
*/
2018-01-23 12:08:37 +03:00
exports.updateConfirmFrame = function () {
2017-10-31 14:56:29 +03:00
if ($('.selected-hour').text() === '') {
return;
}
// Appointment Details
var selectedDate = $('#select-date').datepicker('getDate');
if (selectedDate !== null) {
selectedDate = GeneralFunctions.formatDate(selectedDate, GlobalVariables.dateFormat);
}
var serviceId = $('#select-service').val();
var servicePrice = '';
var serviceCurrency = '';
GlobalVariables.availableServices.forEach(function (service, index) {
2020-12-09 15:17:45 +03:00
if (Number(service.id) === Number(serviceId) && Number(service.price) > 0) {
servicePrice = service.price;
serviceCurrency = service.currency;
return false; // break loop
}
});
$('#appointment-details').empty();
$('<div/>', {
'html': [
$('<h4/>', {
'text': EALang.appointment
}),
$('<p/>', {
'html': [
$('<span/>', {
'text': EALang.service + ': ' + $('#select-service option:selected').text()
}),
$('<br/>'),
$('<span/>', {
'text': EALang.provider + ': ' + $('#select-provider option:selected').text()
}),
$('<br/>'),
$('<span/>', {
'text': EALang.start + ': ' + selectedDate + ' ' + $('.selected-hour').text()
}),
$('<br/>'),
$('<span/>', {
2020-12-09 15:17:45 +03:00
'text': EALang.timezone + ': ' + $('#select-timezone option:selected').text()
}),
$('<br/>'),
$('<span/>', {
'text': EALang.price + ': ' + servicePrice + ' ' + serviceCurrency,
'prop': {
'hidden': !servicePrice
}
})
]
})
]
}).appendTo('#appointment-details');
// Customer Details
2016-10-10 19:29:48 +03:00
var firstName = GeneralFunctions.escapeHtml($('#first-name').val());
var lastName = GeneralFunctions.escapeHtml($('#last-name').val());
2016-07-15 21:52:21 +03:00
var phoneNumber = GeneralFunctions.escapeHtml($('#phone-number').val());
var email = GeneralFunctions.escapeHtml($('#email').val());
var address = GeneralFunctions.escapeHtml($('#address').val());
var city = GeneralFunctions.escapeHtml($('#city').val());
var zipCode = GeneralFunctions.escapeHtml($('#zip-code').val());
$('#customer-details').empty();
$('<div/>', {
'html': [
$('<h4/>)', {
'text': EALang.customer
}),
$('<p/>', {
'html': [
$('<span/>', {
'text': EALang.customer + ': ' + firstName + ' ' + lastName
}),
$('<br/>'),
$('<span/>', {
'text': EALang.phone_number + ': ' + phoneNumber
}),
$('<br/>'),
$('<span/>', {
'text': EALang.email + ': ' + email
}),
$('<br/>'),
$('<span/>', {
'text': address ? EALang.address + ': ' + address : ''
}),
$('<br/>'),
$('<span/>', {
'text': city ? EALang.city + ': ' + city : ''
}),
$('<br/>'),
$('<span/>', {
'text': zipCode ? EALang.zip_code + ': ' + zipCode : ''
}),
$('<br/>')
]
})
]
}).appendTo('#customer-details');
// Update appointment form data for submission to server when the user confirms the appointment.
var data = {};
data.customer = {
2016-04-26 22:51:16 +03:00
last_name: $('#last-name').val(),
first_name: $('#first-name').val(),
email: $('#email').val(),
phone_number: $('#phone-number').val(),
address: $('#address').val(),
city: $('#city').val(),
2020-03-29 17:20:30 +03:00
zip_code: $('#zip-code').val(),
timezone: $('#select-timezone').val()
};
data.appointment = {
start_datetime:
moment($('#select-date').datepicker('getDate')).format('YYYY-MM-DD') +
' ' +
2021-11-25 10:40:48 +03:00
moment($('.selected-hour').data('value'), 'HH:mm').format('HH:mm') +
':00',
end_datetime: calculateEndDatetime(),
2016-04-26 22:51:16 +03:00
notes: $('#notes').val(),
is_unavailable: false,
id_users_provider: $('#select-provider').val(),
id_services: $('#select-service').val()
};
data.manage_mode = FrontendBook.manageMode;
if (FrontendBook.manageMode) {
data.appointment.id = GlobalVariables.appointmentData.id;
data.customer.id = GlobalVariables.customerData.id;
}
2015-05-28 00:26:36 +03:00
$('input[name="csrfToken"]').val(GlobalVariables.csrfToken);
$('input[name="post_data"]').val(JSON.stringify(data));
2016-10-10 19:29:48 +03:00
};
/**
* This method calculates the end datetime of the current appointment.
2016-10-10 19:29:48 +03:00
* End datetime is depending on the service and start datetime fields.
*
* @return {String} Returns the end datetime in string format.
*/
function calculateEndDatetime() {
// Find selected service duration.
var serviceId = $('#select-service').val();
var service = GlobalVariables.availableServices.find(function (availableService) {
return Number(availableService.id) === Number(serviceId);
});
// Add the duration to the start datetime.
var selectedDate = moment($('#select-date').datepicker('getDate')).format('YYYY-MM-DD');
var selectedHour = $('.selected-hour').data('value'); // HH:mm
var startMoment = moment(selectedDate + ' ' + selectedHour);
var endMoment;
if (service.duration && startMoment) {
endMoment = startMoment.clone().add({'minutes': parseInt(service.duration)});
} else {
endMoment = moment();
}
return endMoment.format('YYYY-MM-DD HH:mm:ss');
2016-04-26 22:51:16 +03:00
}
/**
* This method applies the appointment's data to the wizard so
* that the user can start making changes on an existing record.
*
* @param {Object} appointment Selected appointment's data.
* @param {Object} provider Selected provider's data.
* @param {Object} customer Selected customer's data.
*
* @return {Boolean} Returns the operation result.
*/
function applyAppointmentData(appointment, provider, customer) {
try {
// Select Service & Provider
$('#select-service').val(appointment.id_services).trigger('change');
$('#select-provider').val(appointment.id_users_provider);
// Set Appointment Date
var startMoment = moment(appointment.start_datetime);
$('#select-date').datepicker('setDate', startMoment.toDate());
FrontendBookApi.getAvailableHours(startMoment.format('YYYY-MM-DD'));
// Apply Customer's Data
$('#last-name').val(customer.last_name);
$('#first-name').val(customer.first_name);
$('#email').val(customer.email);
$('#phone-number').val(customer.phone_number);
$('#address').val(customer.address);
$('#city').val(customer.city);
$('#zip-code').val(customer.zip_code);
if (customer.timezone) {
$('#select-timezone').val(customer.timezone);
}
var appointmentNotes = appointment.notes !== null ? appointment.notes : '';
$('#notes').val(appointmentNotes);
FrontendBook.updateConfirmFrame();
return true;
2018-01-23 12:08:37 +03:00
} catch (exc) {
return false;
}
2016-04-26 22:51:16 +03:00
}
/**
* This method updates a div's html content with a brief description of the
2016-10-10 19:29:48 +03:00
* user selected service (only if available in db). This is useful for the
* customers upon selecting the correct service.
*
* @param {Number} serviceId The selected service record id.
*/
function updateServiceDescription(serviceId) {
var $serviceDescription = $('#service-description');
$serviceDescription.empty();
2020-12-09 15:17:45 +03:00
var service = GlobalVariables.availableServices.find(function (availableService) {
return Number(availableService.id) === Number(serviceId);
2020-05-07 20:00:33 +03:00
});
if (!service) {
return;
}
$('<strong/>', {
'text': service.name
}).appendTo($serviceDescription);
if (service.description) {
$('<br/>').appendTo($serviceDescription);
$('<span/>', {
'html': GeneralFunctions.escapeHtml(service.description).replaceAll('\n', '<br/>')
}).appendTo($serviceDescription);
}
if (service.duration || Number(service.price) > 0 || service.location) {
$('<br/>').appendTo($serviceDescription);
}
if (service.duration) {
$('<span/>', {
'text': '[' + EALang.duration + ' ' + service.duration + ' ' + EALang.minutes + ']'
}).appendTo($serviceDescription);
}
if (Number(service.price) > 0) {
$('<span/>', {
'text': '[' + EALang.price + ' ' + service.price + ' ' + service.currency + ']'
}).appendTo($serviceDescription);
}
if (service.location) {
$('<span/>', {
'text': '[' + EALang.location + ' ' + service.location + ']'
}).appendTo($serviceDescription);
}
2016-04-26 22:51:16 +03:00
}
})(window.FrontendBook);