Skip to content

Commit

Permalink
Refactor trap focus in modal JS
Browse files Browse the repository at this point in the history
  • Loading branch information
jguo144 committed Nov 22, 2024
1 parent b5e20cf commit 65ea698
Showing 1 changed file with 69 additions and 72 deletions.
141 changes: 69 additions & 72 deletions ckanext/opendata_theme/opengov_custom_theme/assets/js/theme.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,88 +8,85 @@ $(document).on('keydown', function (e) {
}
});

this.ckan.module('resource-view-embed', function ($) {
var modal;
var self;
// Trap focus inside api-info modal
$(document).ready(function () {
$('a[data-module="api-info"]').on('click', function () {
trapApiInfoEmbed();
});

function initialize() {
self = this;
modal = $('#embed-'+this.options.id)
$('body').append(modal);
this.el.on('click', _onClick);
$('textarea', modal).on('focus', _selectAllCode).on('mouseup', _preventClick);
$('input', modal).on('keyup change', _updateValues);
_updateEmbedCode();

// Trap focus when modal is shown
modal.on('shown.bs.modal', _trapFocus);
modal.on('hidden.bs.modal', _restoreFocus);
}

function _onClick (event) {
event.preventDefault();
modal.modal('show');
}
$('a[data-module="api-info"]').on('keyup', function (e) {
if (e.key === 'Enter') {
trapApiInfoEmbed();
}
});

function _selectAllCode () {
$('textarea', modal).select();
function trapApiInfoEmbed() {
const modal = $(`#api-info-embed`);
if (modal.length) {
// Trap focus within the modal
trapFocus(modal);
// When closed remove event handlers
modal.find('.close').on('click', function () {
modal.off('keydown');
$('[data-module="api-info"]').focus();
});
}
}
});

function _updateValues () {
self.options.width = $('[name="width"]', modal).val();
self.options.height = $('[name="height"]', modal).val();
_updateEmbedCode();
}
// Trap focus inside resource-view-embed modal
$(document).ready(function () {
$('a[data-module="resource-view-embed"]').on('click', function () {
const modalId = $(this).data('module-id');
trapResourceViewEmbed(modalId);
});

function _updateEmbedCode () {
$('[name="code"]', modal).val(_embedCode());
}
$('a[data-module="resource-view-embed"]').on('keyup', function (e) {
if (e.key === 'Enter') {
const modalId = $(this).data('module-id');
trapResourceViewEmbed(modalId);
}
});

function _preventClick (event) {
event.preventDefault();
function trapResourceViewEmbed(modalId) {
const modal = $(`#embed-${modalId}`);
if (modal.length) {
// Trap focus within the modal
trapFocus(modal);
// When closed remove event handlers
modal.find('.close').on('click', function () {
modal.off('keydown');
$('[data-module="resource-view-embed"]').focus();
});
}
}
});

function _embedCode () {
return '<iframe width="' + self.options.width + '" height="' + self.options.height + '" src="' + self.options.url + '" frameBorder="0"></iframe>';
}
function trapFocus(modal) {
const focusableElements = modal.find('a, button, input, textarea');
const firstElement = focusableElements.first();
const lastElement = focusableElements.last();

// Trap focus
function _trapFocus() {
const focusableElements = modal.find('a, button, input, textarea');
const firstElement = focusableElements.first();
const lastElement = focusableElements.last();
// Focus on first element when the modal opens
modal.on('shown.bs.modal', function () {
firstElement.focus();
});

modal.on('keydown', function (e) {
if (e.key === 'Tab') {
if (e.shiftKey) { // Shift + Tab
if (document.activeElement === firstElement[0]) {
e.preventDefault();
lastElement.focus();
}
} else { // Tab
if (document.activeElement === lastElement[0]) {
e.preventDefault();
firstElement.focus();
}
modal.on('keydown', function (e) {
if (e.key === 'Tab') {
if (e.shiftKey) {
// Shift + Tab
if (document.activeElement === firstElement[0]) {
e.preventDefault();
lastElement.focus();
}
} else {
// Tab
if (document.activeElement === lastElement[0]) {
e.preventDefault();
firstElement.focus();
}
}
});

// Move focus to the modal
firstElement.focus();
}

function _restoreFocus() {
modal.off('keydown');
}

return {
initialize: initialize,
options: {
id: 0,
url: '#',
width: 700,
height: 400
}
}
});
});
}

0 comments on commit 65ea698

Please sign in to comment.