Files
notifications-admin/app/assets/javascripts/modules/show-hide-content.js
Alex Janousek 6f5750f095 Removed all govuk css (#2814)
* Removed all govuk css

* Updated reference files

* Removing govuk js

* Fixed casing for modules, removed unused page

* Got more reference images

* Updated template page

* Removed govuk padding util

* Updated hint to uswds hint

* More govuk cleanup

* Commiting backstopjs ref files

* Fixed all unit tests that broke due to brittleness around govuk styling

* Added new ref images

* Final removal of govuk

* Officially removed all govuk references

* Updated reference file

* Updated link to button

* UI modernization

* Cleanup

* removed govuk escaping tests since they are no longer needed

* Fix CodeQL security issue in escapeElementName function

- Escape backslashes first before other special characters
- Prevents potential double-escaping vulnerability
- Addresses CodeQL alert about improper string escaping

* Found more govuk removal. Fixed unit tests

* Add missing pipeline check to pre-commit

* updated test

* Updated e2e test

* More update to e2e test

* Fixed another e2e test

* Simple PR comments addressed

* More updates

* Updated backstop ref files

* Refactored folder selection for non-admins

* Updated redundant line

* Updated tests to include correct mocks

* Added more ref files

* Addressing carlos comments

* Addressing Carlo comments, cleanup of window initing

* More cleanup and addressing carlo comments

* Fixing a11 scan

* Fixed a few issues with javascript

* Fixed for pr

* Fixing e2e tests

* Tweaking e2e test

* Added more ref files and cleaned up urls.js

* Fixed bug with creating new template

* Removed brittle test - addressed code ql comment

* e2e race condition fix

* More e2e test fixes

* Updated e2e tests to not wait for text sent

* Updated test to not wait for button click response

* Made tear down more resilent if staging is down

* reverted e2e test to what was working before main merge

* Updated backstopRef images

* Updated gulp to include job-polling differently
2025-10-06 09:38:54 -04:00

155 lines
4.3 KiB
JavaScript

(function (window) {
'use strict';
var $ = window.jQuery;
function ShowHideContent () {
var self = this;
var selectors = {
namespace: 'ShowHideContent',
radio: '[data-target] > input[type="radio"]',
checkbox: '[data-target] > input[type="checkbox"]'
};
function initToggledContent () {
var $control = $(this);
var $content = getToggledContent($control);
if ($content.length) {
$control.attr('aria-controls', $content.attr('id'));
$control.attr('aria-expanded', 'false');
$content.attr('aria-hidden', 'true');
}
}
function getToggledContent ($control) {
try {
var id = $control.attr('aria-controls');
if (!id) {
id = $control.closest('[data-target]').data('target');
}
if (!id || !/^[\w-]+$/.test(id)) {
console.warn('Invalid element ID:', id);
return $();
}
return $('#' + id);
} catch (error) {
console.error('Error getting toggled content:', error);
return $();
}
}
function showToggledContent ($control, $content) {
if ($content.hasClass('display-none')) {
$content.removeClass('display-none');
$content.attr('aria-hidden', 'false');
if ($control.attr('aria-controls')) {
$control.attr('aria-expanded', 'true');
}
}
}
function hideToggledContent ($control, $content) {
$content.addClass('display-none');
$content.attr('aria-hidden', 'true');
if ($control.attr('aria-controls')) {
$control.attr('aria-expanded', 'false');
}
}
function handleRadioContent ($control, $content) {
var selector = selectors.radio + '[name=' + escapeElementName($control.attr('name')) + ']';
var $radios = $(selector);
$radios.each(function () {
hideToggledContent($(this), getToggledContent($(this)));
});
showToggledContent($control, $content);
}
function handleCheckboxContent ($control, $content) {
if ($control.is(':checked')) {
showToggledContent($control, $content);
} else {
hideToggledContent($control, $content);
}
}
function escapeElementName (str) {
// First escape backslashes, then escape other special characters
// This prevents double-escaping issues identified by CodeQL
return str
? str.replace(/\\/g, '\\\\').replace(/([!"#$%&'()*+,./:;<=>?@[\]^`{|}~])/g, '\\$1')
: str;
}
function setupHandlers () {
var $controls = $(selectors.radio + ', ' + selectors.checkbox);
$(selectors.radio).on('click.' + selectors.namespace, function () {
handleRadioContent($(this), getToggledContent($(this)));
});
$(selectors.checkbox).on('click.' + selectors.namespace, function () {
handleCheckboxContent($(this), getToggledContent($(this)));
});
if ($controls.filter(':checked').length) {
$controls.filter(':checked').each(function () {
var $control = $(this);
var $content = getToggledContent($control);
if ($control.is('[type=radio]')) {
handleRadioContent($control, $content);
} else {
handleCheckboxContent($control, $content);
}
});
}
}
self.destroy = function () {
var $controls = $(selectors.radio + ', ' + selectors.checkbox);
$controls.each(function () {
var $control = $(this);
var $content = getToggledContent($control);
$control.removeAttr('aria-controls aria-expanded');
$content.removeAttr('aria-hidden');
});
$(selectors.radio).off('.' + selectors.namespace);
$(selectors.checkbox).off('.' + selectors.namespace);
};
self.init = function () {
try {
$(selectors.radio + ', ' + selectors.checkbox).each(initToggledContent);
setupHandlers();
} catch (error) {
console.error('Error initializing show-hide content:', error);
}
};
}
ShowHideContent.prototype.start = function (element) {
try {
var instance = new ShowHideContent();
instance.init(element);
} catch (error) {
console.error('Failed to start show-hide content module:', error);
}
};
window.NotifyModules.ShowHideContent = ShowHideContent;
})(window);