/* ---------------------------------------------------------------------------- * Easy!Appointments - Open Source Web Scheduler * * @package EasyAppointments * @author A.Tselegidis <alextselegidis@gmail.com> * @copyright Copyright (c) 2013 - 2020, Alex Tselegidis * @license http://opensource.org/licenses/GPL-3.0 - GPLv3 * @link http://easyappointments.org * @since v1.0.0 * ---------------------------------------------------------------------------- */ (function () { 'use strict'; /** * CustomersHelper Class * * This class contains the methods that are used in the backend customers page. * * @class CustomersHelper */ function CustomersHelper() { this.filterResults = {}; this.filterLimit = 20; } /** * Binds the default event handlers of the backend customers page. */ CustomersHelper.prototype.bindEventHandlers = function () { var instance = this; /** * Event: Filter Customers Form "Submit" * * @param {jQuery.Event} event */ $('#customers').on('submit', '#filter-customers form', function (event) { event.preventDefault(); var key = $('#filter-customers .key').val(); $('#filter-customers .selected').removeClass('selected'); instance.filterLimit = 20; instance.resetForm(); instance.filter(key); }); /** * Event: Filter Customers Clear Button "Click" */ $('#customers').on('click', '#filter-customers .clear', function () { $('#filter-customers .key').val(''); instance.filterLimit = 20; instance.filter(''); instance.resetForm(); }); /** * Event: Filter Entry "Click" * * Display the customer data of the selected row. */ $('#customers').on('click', '.customer-row', function () { if ($('#filter-customers .filter').prop('disabled')) { return; // Do nothing when user edits a customer record. } var customerId = $(this).attr('data-id'); var customer = instance.filterResults.find(function (filterResult) { return Number(filterResult.id) === Number(customerId); }); instance.display(customer); $('#filter-customers .selected').removeClass('selected'); $(this).addClass('selected'); $('#edit-customer, #delete-customer').prop('disabled', false); }); /** * Event: Add Customer Button "Click" */ $('#customers').on('click', '#add-customer', function () { instance.resetForm(); $('#add-edit-delete-group').hide(); $('#save-cancel-group').show(); $('.record-details').find('input, select, textarea').prop('disabled', false); $('#filter-customers button').prop('disabled', true); $('#filter-customers .results').css('color', '#AAA'); }); /** * Event: Edit Customer Button "Click" */ $('#customers').on('click', '#edit-customer', function () { $('.record-details').find('input, select, textarea').prop('disabled', false); $('#add-edit-delete-group').hide(); $('#save-cancel-group').show(); $('#filter-customers button').prop('disabled', true); $('#filter-customers .results').css('color', '#AAA'); }); /** * Event: Cancel Customer Add/Edit Operation Button "Click" */ $('#customers').on('click', '#cancel-customer', function () { var id = $('#customer-id').val(); instance.resetForm(); if (id) { instance.select(id, true); } }); /** * Event: Save Add/Edit Customer Operation "Click" */ $('#customers').on('click', '#save-customer', function () { var customer = { first_name: $('#first-name').val(), last_name: $('#last-name').val(), email: $('#email').val(), phone_number: $('#phone-number').val(), address: $('#address').val(), city: $('#city').val(), zip_code: $('#zip-code').val(), notes: $('#notes').val(), timezone: $('#timezone').val(), language: $('#language').val() || 'english' }; if ($('#customer-id').val()) { customer.id = $('#customer-id').val(); } if (!instance.validate()) { return; } instance.save(customer); }); /** * Event: Delete Customer Button "Click" */ $('#customers').on('click', '#delete-customer', function () { var customerId = $('#customer-id').val(); var buttons = [ { text: App.Lang.cancel, click: function () { $('#message-box').dialog('close'); } }, { text: App.Lang.delete, click: function () { instance.delete(customerId); $('#message-box').dialog('close'); } } ]; GeneralFunctions.displayMessageBox(App.Lang.delete_customer, App.Lang.delete_record_prompt, buttons); }); }; /** * Save a customer record to the database (via ajax post). * * @param {Object} customer Contains the customer data. */ CustomersHelper.prototype.save = function (customer) { var url = GlobalVariables.baseUrl + '/index.php/customers/' + (customer.id ? 'update' : 'create'); var data = { csrf_token: GlobalVariables.csrfToken, customer: JSON.stringify(customer) }; $.post(url, data).done( function (response) { Backend.displayNotification(App.Lang.customer_saved); this.resetForm(); $('#filter-customers .key').val(''); this.filter('', response.id, true); }.bind(this) ); }; /** * Delete a customer record from database. * * @param {Number} id Record id to be deleted. */ CustomersHelper.prototype.delete = function (id) { var url = GlobalVariables.baseUrl + '/index.php/customers/destroy'; var data = { csrf_token: GlobalVariables.csrfToken, customer_id: id }; $.post(url, data).done( function () { Backend.displayNotification(App.Lang.customer_deleted); this.resetForm(); this.filter($('#filter-customers .key').val()); }.bind(this) ); }; /** * Validate customer data before save (insert or update). */ CustomersHelper.prototype.validate = function () { $('#form-message').removeClass('alert-danger').hide(); $('.is-invalid').removeClass('is-invalid'); try { // Validate required fields. var missingRequired = false; $('.required').each(function (index, requiredField) { if ($(requiredField).val() === '') { $(requiredField).addClass('is-invalid'); missingRequired = true; } }); if (missingRequired) { throw new Error(App.Lang.fields_are_required); } // Validate email address. if (!GeneralFunctions.validateEmail($('#email').val())) { $('#email').addClass('is-invalid'); throw new Error(App.Lang.invalid_email); } return true; } catch (error) { $('#form-message').addClass('alert-danger').text(error.message).show(); return false; } }; /** * Bring the customer form back to its initial state. */ CustomersHelper.prototype.resetForm = function () { $('.record-details').find('input, select, textarea').val('').prop('disabled', true); $('.record-details #timezone').val('UTC'); $('#language').val('english'); $('#customer-appointments').empty(); $('#edit-customer, #delete-customer').prop('disabled', true); $('#add-edit-delete-group').show(); $('#save-cancel-group').hide(); $('.record-details .is-invalid').removeClass('is-invalid'); $('.record-details #form-message').hide(); $('#filter-customers button').prop('disabled', false); $('#filter-customers .selected').removeClass('selected'); $('#filter-customers .results').css('color', ''); }; /** * Display a customer record into the form. * * @param {Object} customer Contains the customer record data. */ CustomersHelper.prototype.display = function (customer) { $('#customer-id').val(customer.id); $('#first-name').val(customer.first_name); $('#last-name').val(customer.last_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); $('#notes').val(customer.notes); $('#timezone').val(customer.timezone); $('#language').val(customer.language || 'english'); $('#customer-appointments').empty(); if (!customer.appointments.length) { $('<p/>', { 'text': App.Lang.no_records_found }).appendTo('#customer-appointments'); } customer.appointments.forEach(function (appointment) { if ( GlobalVariables.user.role_slug === Backend.DB_SLUG_PROVIDER && parseInt(appointment.id_users_provider) !== GlobalVariables.user.id ) { return; } if ( GlobalVariables.user.role_slug === Backend.DB_SLUG_SECRETARY && GlobalVariables.secretaryProviders.indexOf(appointment.id_users_provider) === -1 ) { return; } var start = GeneralFunctions.formatDate( moment(appointment.start_datetime).toDate(), GlobalVariables.dateFormat, true ); var end = GeneralFunctions.formatDate( moment(appointment.end_datetime).toDate(), GlobalVariables.dateFormat, true ); $('<div/>', { 'class': 'appointment-row', 'data-id': appointment.id, 'html': [ // Service - Provider $('<a/>', { 'href': GlobalVariables.baseUrl + '/index.php/backend/index/' + appointment.hash, 'html': [ $('<i/>', { 'class': 'fas fa-edit me-1' }), $('<strong/>', { 'text': appointment.service.name + ' - ' + appointment.provider.first_name + ' ' + appointment.provider.last_name }), $('<br/>') ] }), // Start $('<small/>', { 'text': start }), $('<br/>'), // End $('<small/>', { 'text': end }), $('<br/>'), // Timezone $('<small/>', { 'text': GlobalVariables.timezones[appointment.provider.timezone] }) ] }).appendTo('#customer-appointments'); }); }; /** * Filter customer records. * * @param {String} keyword This keyword string is used to filter the customer records. * @param {Number} selectId Optional, if set then after the filter operation the record with the given * ID will be selected (but not displayed). * @param {Boolean} display Optional (false), if true then the selected record will be displayed on the form. */ CustomersHelper.prototype.filter = function (keyword, selectId, display) { display = display || false; var url = GlobalVariables.baseUrl + '/index.php/customers/search'; var data = { csrf_token: GlobalVariables.csrfToken, keyword: keyword, limit: this.filterLimit }; $.post(url, data).done( function (response) { this.filterResults = response; $('#filter-customers .results').empty(); response.forEach( function (customer) { $('#filter-customers .results').append(this.getFilterHtml(customer)).append($('<hr/>')); }.bind(this) ); if (!response.length) { $('#filter-customers .results').append( $('<em/>', { 'text': App.Lang.no_records_found }) ); } else if (response.length === this.filterLimit) { $('<button/>', { 'type': 'button', 'class': 'btn btn-outline-secondary w-100 load-more text-center', 'text': App.Lang.load_more, 'click': function () { this.filterLimit += 20; this.filter(keyword, selectId, display); }.bind(this) }).appendTo('#filter-customers .results'); } if (selectId) { this.select(selectId, display); } }.bind(this) ); }; /** * Get the filter results row HTML code. * * @param {Object} customer Contains the customer data. * * @return {String} Returns the record HTML code. */ CustomersHelper.prototype.getFilterHtml = function (customer) { var name = customer.first_name + ' ' + customer.last_name; var info = customer.email; info = customer.phone_number ? info + ', ' + customer.phone_number : info; return $('<div/>', { 'class': 'customer-row entry', 'data-id': customer.id, 'html': [ $('<strong/>', { 'text': name }), $('<br/>'), $('<span/>', { 'text': info }), $('<br/>') ] }); }; /** * Select a specific record from the current filter results. * * If the customer id does not exist in the list then no record will be selected. * * @param {Number} id The record id to be selected from the filter results. * @param {Boolean} display Optional (false), if true then the method will display the record * on the form. */ CustomersHelper.prototype.select = function (id, display) { display = display || false; $('#filter-customers .selected').removeClass('selected'); $('#filter-customers .entry[data-id="' + id + '"]').addClass('selected'); if (display) { var customer = this.filterResults.find(function (filterResult) { return Number(filterResult.id) === Number(id); }); this.display(customer); $('#edit-customer, #delete-customer').prop('disabled', false); } }; window.CustomersHelper = CustomersHelper; })();