Merge pull request #215 from SchrodingersGat/modal-improvements

Modal improvements
This commit is contained in:
Oliver 2019-05-03 23:52:48 +10:00 committed by GitHub
commit f286effd62
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 200 additions and 22 deletions

View File

@ -1,20 +1,45 @@
function makeOption(id, title) { function makeOption(id, title) {
/* Format an option for a select element
*/
return "<option value='" + id + "'>" + title + "</option>"; return "<option value='" + id + "'>" + title + "</option>";
} }
function attachSelect(modal) {
// Attach to any 'select' inputs on the modal function attachSelect(modal) {
// Provide search filtering of dropdown items /* Attach 'select2' functionality to any drop-down list in the modal.
$(modal + ' .select').select2({ * Provides search filtering for dropdown items
*/
$(modal + ' .select').select2({
dropdownParent: $(modal), dropdownParent: $(modal),
// dropdownAutoWidth parameter is required to work properly with modal forms
dropdownAutoWidth: true, dropdownAutoWidth: true,
}); });
} }
function afterForm(response, options) {
// Should we show alerts immediately or cache them? function loadingMessageContent() {
/* Render a 'loading' message to display in a form
* when waiting for a response from the server
*/
// TODO - This can be made a lot better
return '<b>Loading...</b>';
}
function afterForm(response, options) {
/* afterForm is called after a form is successfully submitted,
* and the form is dismissed.
* Used for general purpose functionality after form submission:
*
* - Display a bootstrap alert (success / info / warning / danger)
* - Run a supplied success callback function
* - Redirect the browser to a different URL
* - Reload the page
*/
// Should we show alerts immediately or cache them?
var cache = (options.follow && response.url) || var cache = (options.follow && response.url) ||
options.redirect || options.redirect ||
options.reload; options.reload;
@ -46,27 +71,53 @@ function afterForm(response, options) {
else if (options.reload) { else if (options.reload) {
location.reload(); location.reload();
} }
} }
function modalEnable(modal, enable=true) {
/* Enable (or disable) modal form elements to prevent user input
*/
// Enable or disable the submit button
$(modal).find('#modal-form-submit').prop('disabled', !enable);
}
function modalSetTitle(modal, title='') { function modalSetTitle(modal, title='') {
/* Update the title of a modal form
*/
$(modal + ' #modal-title').html(title); $(modal + ' #modal-title').html(title);
} }
function modalSetContent(modal, content='') { function modalSetContent(modal, content='') {
/* Update the content panel of a modal form
*/
$(modal).find('.modal-form-content').html(content); $(modal).find('.modal-form-content').html(content);
} }
function modalSetButtonText(modal, submit_text, close_text) { function modalSetButtonText(modal, submit_text, close_text) {
/* Set the button text for a modal form
*
* submit_text - text for the form submit button
* close_text - text for the form dismiss button
*/
$(modal).find("#modal-form-submit").html(submit_text); $(modal).find("#modal-form-submit").html(submit_text);
$(modal).find("#modal-form-close").html(close_text); $(modal).find("#modal-form-close").html(close_text);
} }
function closeModal(modal='#modal-form') { function closeModal(modal='#modal-form') {
/* Dismiss (hide) a modal form
*/
$(modal).modal('hide'); $(modal).modal('hide');
} }
function modalSubmit(modal, callback) { function modalSubmit(modal, callback) {
/* Perform the submission action for the modal form
*/
$(modal).off('click', '#modal-form-submit'); $(modal).off('click', '#modal-form-submit');
$(modal).on('click', '#modal-form-submit', function() { $(modal).on('click', '#modal-form-submit', function() {
@ -75,7 +126,72 @@ function modalSubmit(modal, callback) {
} }
function renderErrorMessage(xhr) {
var html = '<b>' + xhr.statusText + '</b><br>';
html += '<b>Status Code - ' + xhr.status + '</b><br><hr>';
html += `
<div class='panel-group'>
<div class='panel panel-default'>
<div class='panel panel-heading'>
<div class='panel-title'>
<a data-toggle='collapse' href="#collapse-error-info">Show Error Information</a>
</div>
</div>
<div class='panel-collapse collapse' id='collapse-error-info'>
<div class='panel-body'>`;
html += xhr.responseText;
html += `
</div>
</div>
</div>
</div>`;
return html;
}
function showDialog(title, content, options={}) {
/* Display a modal dialog message box.
*
* title - Title text
* content - HTML content of the dialog window
* options:
* modal - modal form to use (default = '#modal-dialog')
*/
var modal = options.modal || '#modal-dialog';
$(modal).on('shown.bs.modal', function() {
$(modal + ' .modal-form-content').scrollTop(0);
});
modalSetTitle(modal, title);
modalSetContent(modal, content);
$(modal).modal({
backdrop: 'static',
keyboard: false,
});
$(modal).modal('show');
}
function openModal(options) { function openModal(options) {
/* Open a modal form, and perform some action based on the provided options object:
*
* options can contain:
*
* modal - ID of the modal form element (default = '#modal-form')
* title - Custom title for the form
* content - Default content for the form panel
* submit_text - Label for the submit button (default = 'Submit')
* close_text - Label for the close button (default = 'Close')
*/
var modal = options.modal || '#modal-form'; var modal = options.modal || '#modal-form';
@ -95,12 +211,18 @@ function openModal(options) {
} }
}); });
// Unless the title is explicitly set, display loading message
if (options.title) { if (options.title) {
modalSetTitle(modal, options.title); modalSetTitle(modal, options.title);
} else {
modalSetTitle(modal, 'Loading Form Data...');
} }
// Unless the content is explicitly set, display loading message
if (options.content) { if (options.content) {
modalSetContent(modal, options.content); modalSetContent(modal, options.content);
} else {
modalSetContent(modal, loadingMessageContent());
} }
// Default labels for 'Submit' and 'Close' buttons in the form // Default labels for 'Submit' and 'Close' buttons in the form
@ -113,11 +235,18 @@ function openModal(options) {
backdrop: 'static', backdrop: 'static',
keyboard: false, keyboard: false,
}); });
// Disable the form
modalEnable(modal, false);
// Finally, display the modal window
$(modal).modal('show'); $(modal).modal('show');
} }
function launchDeleteForm(url, options = {}) { function launchDeleteForm(url, options = {}) {
/* Launch a modal form to delete an object
*/
var modal = options.modal || '#modal-delete'; var modal = options.modal || '#modal-delete';
@ -144,13 +273,14 @@ function launchDeleteForm(url, options = {}) {
modalSetContent(modal, response.html_data); modalSetContent(modal, response.html_data);
} }
else { else {
alert('JSON response missing HTML data');
$(modal).modal('hide'); $(modal).modal('hide');
showDialog('Invalid form response', 'JSON response missing HTML data');
} }
}, },
error: function (xhr, ajaxOptions, thrownError) { error: function (xhr, ajaxOptions, thrownError) {
alert('Error requesting JSON data:\n' + thrownError);
$(modal).modal('hide'); $(modal).modal('hide');
showDialog('Error requesting form data', renderErrorMessage(xhr));
} }
}); });
@ -168,20 +298,30 @@ function launchDeleteForm(url, options = {}) {
afterForm(response, options); afterForm(response, options);
}, },
error: function (xhr, ajaxOptions, thrownError) { error: function (xhr, ajaxOptions, thrownError) {
alert('Error deleting item:\n' + thrownError);
$(modal).modal('hide'); $(modal).modal('hide');
showDialog('Error deleting item', renderErrorMessage(xhr));
} }
}); });
}); });
} }
function injectModalForm(modal, form_html) { function injectModalForm(modal, form_html) {
// Inject the form data into the modal window /* Inject form content into the modal.
* Updates the HTML of the form content, and then applies some other updates
*/
$(modal).find('.modal-form-content').html(form_html); $(modal).find('.modal-form-content').html(form_html);
attachSelect(modal); attachSelect(modal);
} }
function handleModalForm(url, options) { function handleModalForm(url, options) {
/* Update a modal form after data are received from the server.
* Manages POST requests until the form is successfully submitted.
*
* The server should respond with a JSON object containing a boolean value 'form_valid'
* Form submission repeats (after user interaction) until 'form_valid' = true
*/
var modal = options.modal || '#modal-form'; var modal = options.modal || '#modal-form';
@ -196,6 +336,7 @@ function handleModalForm(url, options) {
}); });
form.submit(function() { form.submit(function() {
// We should never get here (form submission is overridden)
alert('form submit'); alert('form submit');
return false; return false;
}); });
@ -203,8 +344,14 @@ function handleModalForm(url, options) {
modalSubmit(modal, function() { modalSubmit(modal, function() {
$(modal).find('.js-modal-form').ajaxSubmit({ $(modal).find('.js-modal-form').ajaxSubmit({
url: url, url: url,
beforeSend: function() {
// Disable modal until the server returns a response
modalEnable(modal, false);
},
// POST was successful // POST was successful
success: function(response, status, xhr, f) { success: function(response, status, xhr, f) {
// Re-enable the modal
modalEnable(modal, true);
if ('form_valid' in response) { if ('form_valid' in response) {
// Form data was validated correctly // Form data was validated correctly
if (response.form_valid) { if (response.form_valid) {
@ -217,7 +364,8 @@ function handleModalForm(url, options) {
injectModalForm(modal, response.html_form); injectModalForm(modal, response.html_form);
} }
else { else {
alert('HTML form data missing from server response'); $(modal).modal('hide');
showDialog('Invalid response from server', 'Form data missing from server response');
} }
} }
} }
@ -227,8 +375,10 @@ function handleModalForm(url, options) {
} }
}, },
error: function(xhr, ajaxOptions, thrownError) { error: function(xhr, ajaxOptions, thrownError) {
alert('Error posting form data:\n' + thrownError); // There was an error submitting form data via POST
$(modal).modal('hide');
$(modal).modal('hide');
showDialog('Error posting form data', renderErrorMessage(xhr));
}, },
complete: function(xhr) { complete: function(xhr) {
//TODO //TODO
@ -237,12 +387,16 @@ function handleModalForm(url, options) {
}); });
} }
/*
* launchModalForm
* Opens a model window and fills it with a requested form
* If the form is loaded successfully, calls handleModalForm
*/
function launchModalForm(url, options = {}) { function launchModalForm(url, options = {}) {
/* Launch a modal form, and request data from the server to fill the form
* If the form data is returned from the server, calls handleModalForm()
*
* A successful request will return a JSON object with, at minimum,
* an object called 'html_form'
*
* If the request is NOT successful, displays an appropriate error message.
*/
var modal = options.modal || '#modal-form'; var modal = options.modal || '#modal-form';
@ -263,6 +417,10 @@ function launchModalForm(url, options = {}) {
}); });
}, },
success: function(response) { success: function(response) {
// Enable the form
modalEnable(modal, true);
if (response.title) { if (response.title) {
modalSetTitle(modal, response.title); modalSetTitle(modal, response.title);
} }
@ -272,13 +430,13 @@ function launchModalForm(url, options = {}) {
handleModalForm(url, options); handleModalForm(url, options);
} else { } else {
alert('JSON response missing form data');
$(modal).modal('hide'); $(modal).modal('hide');
showDialog('Invalid server response', 'JSON response missing form data');
} }
}, },
error: function (xhr, ajaxOptions, thrownError) { error: function (xhr, ajaxOptions, thrownError) {
alert('Error requesting form data:\n' + thrownError);
$(modal).modal('hide'); $(modal).modal('hide');
showDialog('Error requesting form data', renderErrorMessage(xhr));
} }
}; };
@ -289,4 +447,4 @@ function launchModalForm(url, options = {}) {
// Send the AJAX request // Send the AJAX request
$.ajax(ajax_data); $.ajax(ajax_data);
} }

View File

@ -110,6 +110,8 @@ function updateStock(items, options={}) {
$(modal).find('#note-warning').hide(); $(modal).find('#note-warning').hide();
modalEnable(modal, true);
modalSubmit(modal, function() { modalSubmit(modal, function() {
var stocktake = []; var stocktake = [];

View File

@ -37,4 +37,22 @@
</div> </div>
</div> </div>
</div> </div>
</div>
<div class='modal fade modal-fixed-footer' tabindex='-1' role='dialog' id='modal-dialog'>
<div class='modal-dialog'>
<div class='modal-content'>
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">&times;</span>
</button>
<h3 id='modal-title'>Confirm Item Deletion</h3>
</div>
<div class='modal-form-content'>
</div>
<div class='modal-footer'>
<button type='button' class='btn btn-default' data-dismiss='modal'>Close</button>
</div>
</div>
</div>
</div> </div>