|
1 | 1 | (function ($) { |
2 | 2 | $(document).ready(function () { |
3 | 3 | var $modal = $( |
4 | | - '<div id="dma-modal-action" class="dma-modal"><div class="dma-modal-content"></div></div>', |
| 4 | + '<div id="dma-modal-action" class="dma-modal" role="dialog" aria-modal="true"><div class="dma-modal-content"></div></div>', |
5 | 5 | ).appendTo("body"); |
6 | 6 | var $modalContent = $modal.find(".dma-modal-content"); |
| 7 | + var lastFocusedElement = null; |
7 | 8 |
|
8 | 9 | // Function to handle conditional fields visibility |
9 | 10 | function handleConditionalFields() { |
|
31 | 32 | // For radio buttons and checkboxes |
32 | 33 | if ($dependentField.is(':radio') || $dependentField.is(':checkbox')) { |
33 | 34 | var currentValue = $form.find('[name="' + config.dependent_field + '"]:checked').val(); |
34 | | - toggleFieldVisibility($fieldContainer, currentValue, config.show_on_values); |
| 35 | + _toggleFieldVisibility($fieldContainer, currentValue, config.show_on_values); |
35 | 36 | } |
36 | 37 | // For select elements |
37 | 38 | else if ($dependentField.is('select')) { |
38 | 39 | var currentValue = $dependentField.val(); |
39 | | - toggleFieldVisibility($fieldContainer, currentValue, config.show_on_values); |
| 40 | + _toggleFieldVisibility($fieldContainer, currentValue, config.show_on_values); |
40 | 41 | } |
41 | 42 | // For other input types |
42 | 43 | else { |
43 | 44 | var currentValue = $dependentField.val(); |
44 | | - toggleFieldVisibility($fieldContainer, currentValue, config.show_on_values); |
| 45 | + _toggleFieldVisibility($fieldContainer, currentValue, config.show_on_values); |
45 | 46 | } |
46 | 47 |
|
47 | 48 | // Add event listener to the dependent field |
|
54 | 55 | newValue = $(this).val(); |
55 | 56 | } |
56 | 57 |
|
57 | | - toggleFieldVisibility($fieldContainer, newValue, config.show_on_values); |
| 58 | + _toggleFieldVisibility($fieldContainer, newValue, config.show_on_values); |
58 | 59 | }); |
59 | 60 | } |
60 | 61 | } |
61 | 62 | }); |
| 63 | + } |
62 | 64 |
|
63 | | - // Helper function to toggle field visibility |
64 | | - function toggleFieldVisibility($field, currentValue, showOnValues) { |
65 | | - if (showOnValues.includes(currentValue)) { |
66 | | - $field.show(); |
67 | | - } else { |
68 | | - $field.hide(); |
| 65 | + function _toggleFieldVisibility($field, currentValue, showOnValues) { |
| 66 | + if (showOnValues.includes(currentValue)) { |
| 67 | + $field.show(); |
| 68 | + } else { |
| 69 | + $field.hide(); |
| 70 | + } |
| 71 | + } |
| 72 | + |
| 73 | + function _getFocusableElements($container) { |
| 74 | + return $container.find( |
| 75 | + 'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])' |
| 76 | + ).filter(':visible'); |
| 77 | + } |
| 78 | + |
| 79 | + function _trapFocus() { |
| 80 | + var focusableElements = _getFocusableElements($modal); |
| 81 | + if (focusableElements.length === 0) { |
| 82 | + $modal.off('keydown.focustrap'); |
| 83 | + return; |
| 84 | + } |
| 85 | + |
| 86 | + if (focusableElements.length === 1) { |
| 87 | + $modal.off('keydown.focustrap').on('keydown.focustrap', function(e) { |
| 88 | + if (e.key === 'Tab' || e.keyCode === 9) { |
| 89 | + e.preventDefault(); |
| 90 | + } |
| 91 | + }); |
| 92 | + return; |
| 93 | + } |
| 94 | + |
| 95 | + var firstFocusable = focusableElements.first(); |
| 96 | + var lastFocusable = focusableElements.last(); |
| 97 | + |
| 98 | + $modal.off('keydown.focustrap').on('keydown.focustrap', function(e) { |
| 99 | + if (e.key === 'Tab' || e.keyCode === 9) { |
| 100 | + if (e.shiftKey) { |
| 101 | + if (document.activeElement === firstFocusable[0]) { |
| 102 | + e.preventDefault(); |
| 103 | + lastFocusable.focus(); |
| 104 | + } |
| 105 | + } else { |
| 106 | + if (document.activeElement === lastFocusable[0]) { |
| 107 | + e.preventDefault(); |
| 108 | + firstFocusable.focus(); |
| 109 | + } |
| 110 | + } |
69 | 111 | } |
| 112 | + }); |
| 113 | + } |
| 114 | + |
| 115 | + function _closeModal() { |
| 116 | + $modal.hide(); |
| 117 | + $modal.off('keydown.focustrap'); |
| 118 | + $(document).off('keydown.dmaModal'); |
| 119 | + if (lastFocusedElement) { |
| 120 | + lastFocusedElement.focus(); |
| 121 | + lastFocusedElement = null; |
70 | 122 | } |
71 | 123 | } |
72 | 124 |
|
73 | 125 | $(document).on("click", ".dma-modal-action-button", function (e) { |
74 | 126 | e.preventDefault(); |
| 127 | + lastFocusedElement = this; |
75 | 128 | var url = $(this).attr("href"); |
76 | 129 | var isListAction = url.includes("list-modal-action"); |
77 | 130 |
|
|
84 | 137 | } |
85 | 138 | $.get(url, function (data) { |
86 | 139 | if (data.success !== undefined) { |
87 | | - // Skip confirmation case - action was executed directly |
88 | 140 | if (data.success) { |
89 | 141 | location.reload(); |
90 | 142 | } else if (data.errors) { |
91 | | - displayErrors(data.errors); |
| 143 | + _displayErrors(data.errors); |
92 | 144 | } |
93 | 145 | } else if (data.content) { |
94 | | - // Normal case - show modal with confirmation |
95 | 146 | $modalContent.html(data.content); |
| 147 | + |
| 148 | + var $heading = $modalContent.find("h2"); |
| 149 | + if ($heading.length) { |
| 150 | + var headingId = "dma-modal-heading-" + Date.now(); |
| 151 | + $heading.attr("id", headingId); |
| 152 | + $modal.attr("aria-labelledby", headingId); |
| 153 | + } else { |
| 154 | + $modal.removeAttr("aria-labelledby"); |
| 155 | + var modalTitle = $modalContent.find("h1, h2, h3").first().text() || "Dialog"; |
| 156 | + $modal.attr("aria-label", modalTitle); |
| 157 | + } |
| 158 | + |
96 | 159 | $modal.show(); |
97 | | - // Initialize conditional fields after modal content is loaded |
98 | 160 | handleConditionalFields(); |
| 161 | + |
| 162 | + var $firstFocusable = _getFocusableElements($modal).first(); |
| 163 | + if ($firstFocusable.length) { |
| 164 | + $firstFocusable.focus(); |
| 165 | + } else { |
| 166 | + $modal.find('h2').attr('tabindex', '-1').focus(); |
| 167 | + } |
| 168 | + |
| 169 | + _trapFocus(); |
| 170 | + |
| 171 | + $(document).off('keydown.dmaModal').on('keydown.dmaModal', function(e) { |
| 172 | + if (e.key === 'Escape' || e.keyCode === 27) { |
| 173 | + if ($modal.is(':visible')) { |
| 174 | + e.preventDefault(); |
| 175 | + _closeModal(); |
| 176 | + } |
| 177 | + } |
| 178 | + }); |
99 | 179 | } |
100 | 180 | }); |
101 | 181 | }); |
|
105 | 185 | "#dma-modal-action .cancel, #dma-modal-action .dma-modal-close", |
106 | 186 | function (e) { |
107 | 187 | e.preventDefault(); |
108 | | - $modal.hide(); |
| 188 | + _closeModal(); |
109 | 189 | }, |
110 | 190 | ); |
111 | 191 |
|
112 | | - function displayErrors(errors) { |
| 192 | + function _displayErrors(errors) { |
113 | 193 | $(".dma-errorlist, .dma-alert-danger").remove(); |
114 | 194 |
|
| 195 | + var errorCount = Object.keys(errors).length; |
| 196 | + var $statusMessage = $(".dma-status-message"); |
| 197 | + if ($statusMessage.length) { |
| 198 | + $statusMessage.text(""); |
| 199 | + setTimeout(function() { |
| 200 | + $statusMessage.text("Form submission failed. " + errorCount + " error" + (errorCount > 1 ? "s" : "") + " found. Please review and correct."); |
| 201 | + }, 100); |
| 202 | + } |
| 203 | + |
115 | 204 | $.each(errors, function (field, messages) { |
116 | 205 | if (field === "__all__") { |
117 | 206 | var $generalError = $( |
|
121 | 210 | $("#dma-modal-action form").prepend($generalError); |
122 | 211 | } else { |
123 | 212 | var $field = $("#id_" + field); |
124 | | - var $errorList = $('<ul class="dma-errorlist"></ul>'); |
| 213 | + var $errorList = $('<ul class="dma-errorlist" role="alert"></ul>'); |
125 | 214 | $.each(messages, function (index, message) { |
126 | 215 | $errorList.append($("<li></li>").text(message)); |
127 | 216 | }); |
|
135 | 224 | ); |
136 | 225 | $("#dma-modal-action form").prepend($generalError); |
137 | 226 | } |
| 227 | + |
| 228 | + if ($modal.is(':visible')) { |
| 229 | + _trapFocus(); |
| 230 | + } |
138 | 231 | } |
139 | 232 |
|
140 | 233 | $(document).on("submit", "#dma-modal-action form", function (e) { |
|
143 | 236 | var url = form.attr("action"); |
144 | 237 | var formData = new FormData(form[0]); |
145 | 238 |
|
| 239 | + var $confirmBtn = form.closest("#dma-modal-action").find(".dma-confirm-btn"); |
| 240 | + $confirmBtn.prop("disabled", true).addClass("dma-loading").attr("aria-busy", "true"); |
| 241 | + $(".dma-status-message").text("Processing your request, please wait..."); |
| 242 | + |
146 | 243 | var selectedIds = form.find('input[name="selected_ids"]').val(); |
147 | 244 | if (selectedIds) { |
148 | 245 | formData.append("selected_ids", selectedIds); |
|
154 | 251 | processData: false, |
155 | 252 | contentType: false, |
156 | 253 | dataType: "json", |
| 254 | + timeout: 30000, |
157 | 255 | success: function (data) { |
158 | 256 | if (data.success) { |
159 | | - $modal.hide(); |
| 257 | + _closeModal(); |
160 | 258 | location.reload(); |
161 | 259 | } else if (data.errors) { |
162 | | - displayErrors(data.errors); |
| 260 | + $confirmBtn.prop("disabled", false).removeClass("dma-loading").attr("aria-busy", "false"); |
| 261 | + _displayErrors(data.errors); |
163 | 262 | } |
164 | 263 | }, |
165 | 264 | error: function (jqXHR, textStatus, errorThrown) { |
166 | | - displayErrors({ |
167 | | - __all__: ["An unexpected error occurred. Please try again."], |
168 | | - }); |
| 265 | + $confirmBtn.prop("disabled", false).removeClass("dma-loading").attr("aria-busy", "false"); |
| 266 | + if (textStatus === 'timeout') { |
| 267 | + _displayErrors({ |
| 268 | + __all__: ["Request timed out. Please try again."], |
| 269 | + }); |
| 270 | + } else { |
| 271 | + _displayErrors({ |
| 272 | + __all__: ["An unexpected error occurred. Please try again."], |
| 273 | + }); |
| 274 | + } |
169 | 275 | }, |
170 | 276 | }); |
171 | 277 | }); |
172 | 278 |
|
173 | 279 | $(window).on("click", function (e) { |
174 | 280 | if ($(e.target).is(".dma-modal")) { |
175 | | - $modal.hide(); |
| 281 | + if (!$(".dma-confirm-btn").hasClass("dma-loading")) { |
| 282 | + _closeModal(); |
| 283 | + } |
176 | 284 | } |
177 | 285 | }); |
178 | 286 | }); |
|
0 commit comments