govuk_publishing_components 44.1.0 → 44.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (139) hide show
  1. checksums.yaml +4 -4
  2. data/lib/govuk_publishing_components/version.rb +1 -1
  3. data/node_modules/govuk-frontend/dist/govuk/all.bundle.js +336 -225
  4. data/node_modules/govuk-frontend/dist/govuk/all.bundle.js.map +1 -1
  5. data/node_modules/govuk-frontend/dist/govuk/all.bundle.mjs +334 -226
  6. data/node_modules/govuk-frontend/dist/govuk/all.bundle.mjs.map +1 -1
  7. data/node_modules/govuk-frontend/dist/govuk/all.mjs +3 -0
  8. data/node_modules/govuk-frontend/dist/govuk/all.mjs.map +1 -1
  9. data/node_modules/govuk-frontend/dist/govuk/assets/images/govuk-crest.svg +1 -0
  10. data/node_modules/govuk-frontend/dist/govuk/common/govuk-frontend-version.mjs +1 -1
  11. data/node_modules/govuk-frontend/dist/govuk/common/index.mjs +21 -1
  12. data/node_modules/govuk-frontend/dist/govuk/common/index.mjs.map +1 -1
  13. data/node_modules/govuk-frontend/dist/govuk/components/_index.scss +1 -0
  14. data/node_modules/govuk-frontend/dist/govuk/components/_index.scss.map +1 -1
  15. data/node_modules/govuk-frontend/dist/govuk/components/accordion/accordion.bundle.js +92 -26
  16. data/node_modules/govuk-frontend/dist/govuk/components/accordion/accordion.bundle.js.map +1 -1
  17. data/node_modules/govuk-frontend/dist/govuk/components/accordion/accordion.bundle.mjs +92 -26
  18. data/node_modules/govuk-frontend/dist/govuk/components/accordion/accordion.bundle.mjs.map +1 -1
  19. data/node_modules/govuk-frontend/dist/govuk/components/accordion/accordion.mjs +12 -21
  20. data/node_modules/govuk-frontend/dist/govuk/components/accordion/accordion.mjs.map +1 -1
  21. data/node_modules/govuk-frontend/dist/govuk/components/button/button.bundle.js +86 -20
  22. data/node_modules/govuk-frontend/dist/govuk/components/button/button.bundle.js.map +1 -1
  23. data/node_modules/govuk-frontend/dist/govuk/components/button/button.bundle.mjs +86 -20
  24. data/node_modules/govuk-frontend/dist/govuk/components/button/button.bundle.mjs.map +1 -1
  25. data/node_modules/govuk-frontend/dist/govuk/components/button/button.mjs +6 -16
  26. data/node_modules/govuk-frontend/dist/govuk/components/button/button.mjs.map +1 -1
  27. data/node_modules/govuk-frontend/dist/govuk/components/character-count/character-count.bundle.js +89 -23
  28. data/node_modules/govuk-frontend/dist/govuk/components/character-count/character-count.bundle.js.map +1 -1
  29. data/node_modules/govuk-frontend/dist/govuk/components/character-count/character-count.bundle.mjs +89 -23
  30. data/node_modules/govuk-frontend/dist/govuk/components/character-count/character-count.bundle.mjs.map +1 -1
  31. data/node_modules/govuk-frontend/dist/govuk/components/character-count/character-count.mjs +10 -19
  32. data/node_modules/govuk-frontend/dist/govuk/components/character-count/character-count.mjs.map +1 -1
  33. data/node_modules/govuk-frontend/dist/govuk/components/checkboxes/checkboxes.bundle.js +113 -47
  34. data/node_modules/govuk-frontend/dist/govuk/components/checkboxes/checkboxes.bundle.js.map +1 -1
  35. data/node_modules/govuk-frontend/dist/govuk/components/checkboxes/checkboxes.bundle.mjs +113 -47
  36. data/node_modules/govuk-frontend/dist/govuk/components/checkboxes/checkboxes.bundle.mjs.map +1 -1
  37. data/node_modules/govuk-frontend/dist/govuk/components/checkboxes/checkboxes.mjs +7 -16
  38. data/node_modules/govuk-frontend/dist/govuk/components/checkboxes/checkboxes.mjs.map +1 -1
  39. data/node_modules/govuk-frontend/dist/govuk/components/details/_index.scss +7 -2
  40. data/node_modules/govuk-frontend/dist/govuk/components/details/_index.scss.map +1 -1
  41. data/node_modules/govuk-frontend/dist/govuk/components/error-summary/error-summary.bundle.js +86 -20
  42. data/node_modules/govuk-frontend/dist/govuk/components/error-summary/error-summary.bundle.js.map +1 -1
  43. data/node_modules/govuk-frontend/dist/govuk/components/error-summary/error-summary.bundle.mjs +86 -20
  44. data/node_modules/govuk-frontend/dist/govuk/components/error-summary/error-summary.bundle.mjs.map +1 -1
  45. data/node_modules/govuk-frontend/dist/govuk/components/error-summary/error-summary.mjs +6 -16
  46. data/node_modules/govuk-frontend/dist/govuk/components/error-summary/error-summary.mjs.map +1 -1
  47. data/node_modules/govuk-frontend/dist/govuk/components/exit-this-page/exit-this-page.bundle.js +87 -21
  48. data/node_modules/govuk-frontend/dist/govuk/components/exit-this-page/exit-this-page.bundle.js.map +1 -1
  49. data/node_modules/govuk-frontend/dist/govuk/components/exit-this-page/exit-this-page.bundle.mjs +87 -21
  50. data/node_modules/govuk-frontend/dist/govuk/components/exit-this-page/exit-this-page.bundle.mjs.map +1 -1
  51. data/node_modules/govuk-frontend/dist/govuk/components/exit-this-page/exit-this-page.mjs +7 -16
  52. data/node_modules/govuk-frontend/dist/govuk/components/exit-this-page/exit-this-page.mjs.map +1 -1
  53. data/node_modules/govuk-frontend/dist/govuk/components/footer/_index.scss +8 -10
  54. data/node_modules/govuk-frontend/dist/govuk/components/footer/_index.scss.map +1 -1
  55. data/node_modules/govuk-frontend/dist/govuk/components/header/_index.scss +8 -0
  56. data/node_modules/govuk-frontend/dist/govuk/components/header/_index.scss.map +1 -1
  57. data/node_modules/govuk-frontend/dist/govuk/components/header/fixtures.json +12 -0
  58. data/node_modules/govuk-frontend/dist/govuk/components/header/header.bundle.js +87 -21
  59. data/node_modules/govuk-frontend/dist/govuk/components/header/header.bundle.js.map +1 -1
  60. data/node_modules/govuk-frontend/dist/govuk/components/header/header.bundle.mjs +87 -21
  61. data/node_modules/govuk-frontend/dist/govuk/components/header/header.bundle.mjs.map +1 -1
  62. data/node_modules/govuk-frontend/dist/govuk/components/header/header.mjs +7 -16
  63. data/node_modules/govuk-frontend/dist/govuk/components/header/header.mjs.map +1 -1
  64. data/node_modules/govuk-frontend/dist/govuk/components/header/template-with-full-width-border.html +24 -0
  65. data/node_modules/govuk-frontend/dist/govuk/components/notification-banner/notification-banner.bundle.js +86 -20
  66. data/node_modules/govuk-frontend/dist/govuk/components/notification-banner/notification-banner.bundle.js.map +1 -1
  67. data/node_modules/govuk-frontend/dist/govuk/components/notification-banner/notification-banner.bundle.mjs +86 -20
  68. data/node_modules/govuk-frontend/dist/govuk/components/notification-banner/notification-banner.bundle.mjs.map +1 -1
  69. data/node_modules/govuk-frontend/dist/govuk/components/notification-banner/notification-banner.mjs +6 -16
  70. data/node_modules/govuk-frontend/dist/govuk/components/notification-banner/notification-banner.mjs.map +1 -1
  71. data/node_modules/govuk-frontend/dist/govuk/components/password-input/password-input.bundle.js +89 -23
  72. data/node_modules/govuk-frontend/dist/govuk/components/password-input/password-input.bundle.js.map +1 -1
  73. data/node_modules/govuk-frontend/dist/govuk/components/password-input/password-input.bundle.mjs +89 -23
  74. data/node_modules/govuk-frontend/dist/govuk/components/password-input/password-input.bundle.mjs.map +1 -1
  75. data/node_modules/govuk-frontend/dist/govuk/components/password-input/password-input.mjs +9 -18
  76. data/node_modules/govuk-frontend/dist/govuk/components/password-input/password-input.mjs.map +1 -1
  77. data/node_modules/govuk-frontend/dist/govuk/components/radios/radios.bundle.js +113 -47
  78. data/node_modules/govuk-frontend/dist/govuk/components/radios/radios.bundle.js.map +1 -1
  79. data/node_modules/govuk-frontend/dist/govuk/components/radios/radios.bundle.mjs +113 -47
  80. data/node_modules/govuk-frontend/dist/govuk/components/radios/radios.bundle.mjs.map +1 -1
  81. data/node_modules/govuk-frontend/dist/govuk/components/radios/radios.mjs +7 -16
  82. data/node_modules/govuk-frontend/dist/govuk/components/radios/radios.mjs.map +1 -1
  83. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/README.md +15 -0
  84. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/_index.scss +168 -0
  85. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/_index.scss.map +1 -0
  86. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/_service-navigation.scss +4 -0
  87. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/_service-navigation.scss.map +1 -0
  88. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/fixtures.json +464 -0
  89. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/macro-options.json +138 -0
  90. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/macro.njk +3 -0
  91. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/service-navigation.bundle.js +249 -0
  92. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/service-navigation.bundle.js.map +1 -0
  93. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/service-navigation.bundle.mjs +241 -0
  94. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/service-navigation.bundle.mjs.map +1 -0
  95. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/service-navigation.mjs +85 -0
  96. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/service-navigation.mjs.map +1 -0
  97. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-default.html +57 -0
  98. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-html-navigation-items.html +49 -0
  99. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-large-navigation.html +153 -0
  100. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-long-service-name.html +20 -0
  101. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-navigation-with-a-current-item.html +58 -0
  102. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-navigation-with-an-active-item.html +58 -0
  103. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-non-link-navigation-items.html +49 -0
  104. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-service-link.html +20 -0
  105. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-service-name-and-navigation.html +63 -0
  106. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template-with-service-name.html +18 -0
  107. data/node_modules/govuk-frontend/dist/govuk/components/service-navigation/template.njk +102 -0
  108. data/node_modules/govuk-frontend/dist/govuk/components/skip-link/skip-link.bundle.js +93 -26
  109. data/node_modules/govuk-frontend/dist/govuk/components/skip-link/skip-link.bundle.js.map +1 -1
  110. data/node_modules/govuk-frontend/dist/govuk/components/skip-link/skip-link.bundle.mjs +93 -26
  111. data/node_modules/govuk-frontend/dist/govuk/components/skip-link/skip-link.bundle.mjs.map +1 -1
  112. data/node_modules/govuk-frontend/dist/govuk/components/skip-link/skip-link.mjs +13 -21
  113. data/node_modules/govuk-frontend/dist/govuk/components/skip-link/skip-link.mjs.map +1 -1
  114. data/node_modules/govuk-frontend/dist/govuk/components/tabs/tabs.bundle.js +93 -27
  115. data/node_modules/govuk-frontend/dist/govuk/components/tabs/tabs.bundle.js.map +1 -1
  116. data/node_modules/govuk-frontend/dist/govuk/components/tabs/tabs.bundle.mjs +93 -27
  117. data/node_modules/govuk-frontend/dist/govuk/components/tabs/tabs.bundle.mjs.map +1 -1
  118. data/node_modules/govuk-frontend/dist/govuk/components/tabs/tabs.mjs +13 -22
  119. data/node_modules/govuk-frontend/dist/govuk/components/tabs/tabs.mjs.map +1 -1
  120. data/node_modules/govuk-frontend/dist/govuk/components/warning-text/_index.scss +4 -3
  121. data/node_modules/govuk-frontend/dist/govuk/components/warning-text/_index.scss.map +1 -1
  122. data/node_modules/govuk-frontend/dist/govuk/core/_govuk-frontend-properties.scss +1 -1
  123. data/node_modules/govuk-frontend/dist/govuk/errors/index.mjs +16 -3
  124. data/node_modules/govuk-frontend/dist/govuk/errors/index.mjs.map +1 -1
  125. data/node_modules/govuk-frontend/dist/govuk/govuk-frontend-component.mjs +49 -5
  126. data/node_modules/govuk-frontend/dist/govuk/govuk-frontend-component.mjs.map +1 -1
  127. data/node_modules/govuk-frontend/dist/govuk/govuk-frontend.min.css +2 -2
  128. data/node_modules/govuk-frontend/dist/govuk/govuk-frontend.min.css.map +1 -1
  129. data/node_modules/govuk-frontend/dist/govuk/govuk-frontend.min.js +1 -1
  130. data/node_modules/govuk-frontend/dist/govuk/govuk-frontend.min.js.map +1 -1
  131. data/node_modules/govuk-frontend/dist/govuk/init.mjs +72 -10
  132. data/node_modules/govuk-frontend/dist/govuk/init.mjs.map +1 -1
  133. data/node_modules/govuk-frontend/dist/govuk/settings/_colours-organisations.scss +3 -0
  134. data/node_modules/govuk-frontend/dist/govuk/settings/_colours-organisations.scss.map +1 -1
  135. data/node_modules/govuk-frontend/govuk-prototype-kit.config.json +5 -1
  136. data/node_modules/govuk-frontend/package.json +8 -8
  137. metadata +29 -4
  138. data/node_modules/govuk-frontend/dist/govuk/assets/images/govuk-crest-2x.png +0 -0
  139. data/node_modules/govuk-frontend/dist/govuk/assets/images/govuk-crest.png +0 -0
@@ -1 +1 @@
1
- {"version":3,"file":"govuk-frontend.min.js","sources":["../../src/govuk/common/govuk-frontend-version.mjs","../../src/govuk/common/normalise-string.mjs","../../src/govuk/common/index.mjs","../../src/govuk/common/normalise-dataset.mjs","../../src/govuk/errors/index.mjs","../../src/govuk/govuk-frontend-component.mjs","../../src/govuk/i18n.mjs","../../src/govuk/components/accordion/accordion.mjs","../../src/govuk/components/button/button.mjs","../../src/govuk/common/closest-attribute-value.mjs","../../src/govuk/components/character-count/character-count.mjs","../../src/govuk/components/checkboxes/checkboxes.mjs","../../src/govuk/components/error-summary/error-summary.mjs","../../src/govuk/components/exit-this-page/exit-this-page.mjs","../../src/govuk/components/header/header.mjs","../../src/govuk/components/notification-banner/notification-banner.mjs","../../src/govuk/components/password-input/password-input.mjs","../../src/govuk/components/radios/radios.mjs","../../src/govuk/components/skip-link/skip-link.mjs","../../src/govuk/components/tabs/tabs.mjs","../../src/govuk/init.mjs"],"sourcesContent":["/*\n * This variable is automatically overwritten during builds and releases.\n * It doesn't need to be updated manually.\n */\n\n/**\n * GOV.UK Frontend release version\n *\n * {@link https://github.com/alphagov/govuk-frontend/releases}\n */\nexport const version = 'development'\n","/**\n * Normalise string\n *\n * 'If it looks like a duck, and it quacks like a duck…' 🦆\n *\n * If the passed value looks like a boolean or a number, convert it to a boolean\n * or number.\n *\n * Designed to be used to convert config passed via data attributes (which are\n * always strings) into something sensible.\n *\n * @internal\n * @param {DOMStringMap[string]} value - The value to normalise\n * @param {SchemaProperty} [property] - Component schema property\n * @returns {string | boolean | number | undefined} Normalised data\n */\nexport function normaliseString(value, property) {\n const trimmedValue = value ? value.trim() : ''\n\n let output\n let outputType = property?.type\n\n // No schema type set? Determine automatically\n if (!outputType) {\n if (['true', 'false'].includes(trimmedValue)) {\n outputType = 'boolean'\n }\n\n // Empty / whitespace-only strings are considered finite so we need to check\n // the length of the trimmed string as well\n if (trimmedValue.length > 0 && isFinite(Number(trimmedValue))) {\n outputType = 'number'\n }\n }\n\n switch (outputType) {\n case 'boolean':\n output = trimmedValue === 'true'\n break\n\n case 'number':\n output = Number(trimmedValue)\n break\n\n default:\n output = value\n }\n\n return output\n}\n\n/**\n * @typedef {import('./index.mjs').SchemaProperty} SchemaProperty\n */\n","import { normaliseString } from './normalise-string.mjs'\n\n/**\n * Common helpers which do not require polyfill.\n *\n * IMPORTANT: If a helper require a polyfill, please isolate it in its own module\n * so that the polyfill can be properly tree-shaken and does not burden\n * the components that do not need that helper\n */\n\n/**\n * Config merging function\n *\n * Takes any number of objects and combines them together, with\n * greatest priority on the LAST item passed in.\n *\n * @internal\n * @param {...{ [key: string]: unknown }} configObjects - Config objects to merge\n * @returns {{ [key: string]: unknown }} A merged config object\n */\nexport function mergeConfigs(...configObjects) {\n // Start with an empty object as our base\n /** @type {{ [key: string]: unknown }} */\n const formattedConfigObject = {}\n\n // Loop through each of the passed objects\n for (const configObject of configObjects) {\n for (const key of Object.keys(configObject)) {\n const option = formattedConfigObject[key]\n const override = configObject[key]\n\n // Push their keys one-by-one into formattedConfigObject. Any duplicate\n // keys with object values will be merged, otherwise the new value will\n // override the existing value.\n if (isObject(option) && isObject(override)) {\n // @ts-expect-error Index signature for type 'string' is missing\n formattedConfigObject[key] = mergeConfigs(option, override)\n } else {\n // Apply override\n formattedConfigObject[key] = override\n }\n }\n }\n\n return formattedConfigObject\n}\n\n/**\n * Extracts keys starting with a particular namespace from dataset ('data-*')\n * object, removing the namespace in the process, normalising all values\n *\n * @internal\n * @param {{ schema: Schema }} Component - Component class\n * @param {DOMStringMap} dataset - The object to extract key-value pairs from\n * @param {string} namespace - The namespace to filter keys with\n * @returns {ObjectNested | undefined} Nested object with dot-separated key namespace removed\n */\nexport function extractConfigByNamespace(Component, dataset, namespace) {\n const property = Component.schema.properties[namespace]\n\n // Only extract configs for object schema properties\n if (property?.type !== 'object') {\n return\n }\n\n // Add default empty config\n const newObject = {\n [namespace]: /** @type {ObjectNested} */ ({})\n }\n\n for (const [key, value] of Object.entries(dataset)) {\n /** @type {ObjectNested | ObjectNested[NestedKey]} */\n let current = newObject\n\n // Split the key into parts, using . as our namespace separator\n const keyParts = key.split('.')\n\n /**\n * Create new level per part\n *\n * e.g. 'i18n.textareaDescription.other' becomes\n * `{ i18n: { textareaDescription: { other } } }`\n */\n for (const [index, name] of keyParts.entries()) {\n if (typeof current === 'object') {\n // Drop down to nested object until the last part\n if (index < keyParts.length - 1) {\n // New nested object (optionally) replaces existing value\n if (!isObject(current[name])) {\n current[name] = {}\n }\n\n // Drop down into new or existing nested object\n current = current[name]\n } else if (key !== namespace) {\n // Normalised value (optionally) replaces existing value\n current[name] = normaliseString(value)\n }\n }\n }\n }\n\n return newObject[namespace]\n}\n\n/**\n * Get hash fragment from URL\n *\n * Extract the hash fragment (everything after the hash) from a URL,\n * but not including the hash symbol\n *\n * @private\n * @param {string} url - URL\n * @returns {string | undefined} Fragment from URL, without the hash\n */\nexport function getFragmentFromUrl(url) {\n if (!url.includes('#')) {\n return undefined\n }\n\n return url.split('#').pop()\n}\n\n/**\n * Get GOV.UK Frontend breakpoint value from CSS custom property\n *\n * @private\n * @param {string} name - Breakpoint name\n * @returns {{ property: string, value?: string }} Breakpoint object\n */\nexport function getBreakpoint(name) {\n const property = `--govuk-frontend-breakpoint-${name}`\n\n // Get value from `<html>` with breakpoints on CSS :root\n const value = window\n .getComputedStyle(document.documentElement)\n .getPropertyValue(property)\n\n return {\n property,\n value: value || undefined\n }\n}\n\n/**\n * Move focus to element\n *\n * Sets tabindex to -1 to make the element programmatically focusable,\n * but removes it on blur as the element doesn't need to be focused again.\n *\n * @private\n * @template {HTMLElement} FocusElement\n * @param {FocusElement} $element - HTML element\n * @param {object} [options] - Handler options\n * @param {function(this: FocusElement): void} [options.onBeforeFocus] - Callback before focus\n * @param {function(this: FocusElement): void} [options.onBlur] - Callback on blur\n */\nexport function setFocus($element, options = {}) {\n const isFocusable = $element.getAttribute('tabindex')\n\n if (!isFocusable) {\n $element.setAttribute('tabindex', '-1')\n }\n\n /**\n * Handle element focus\n */\n function onFocus() {\n $element.addEventListener('blur', onBlur, { once: true })\n }\n\n /**\n * Handle element blur\n */\n function onBlur() {\n options.onBlur?.call($element)\n\n if (!isFocusable) {\n $element.removeAttribute('tabindex')\n }\n }\n\n // Add listener to reset element on blur, after focus\n $element.addEventListener('focus', onFocus, { once: true })\n\n // Focus element\n options.onBeforeFocus?.call($element)\n $element.focus()\n}\n\n/**\n * Checks if GOV.UK Frontend is supported on this page\n *\n * Some browsers will load and run our JavaScript but GOV.UK Frontend\n * won't be supported.\n *\n * @internal\n * @param {HTMLElement | null} [$scope] - HTML element `<body>` checked for browser support\n * @returns {boolean} Whether GOV.UK Frontend is supported on this page\n */\nexport function isSupported($scope = document.body) {\n if (!$scope) {\n return false\n }\n\n return $scope.classList.contains('govuk-frontend-supported')\n}\n\n/**\n * Validate component config by schema\n *\n * Follows limited examples in JSON schema for wider support in future\n *\n * {@link https://ajv.js.org/json-schema.html#compound-keywords}\n * {@link https://ajv.js.org/packages/ajv-errors.html#single-message}\n *\n * @internal\n * @param {Schema} schema - Config schema\n * @param {{ [key: string]: unknown }} config - Component config\n * @returns {string[]} List of validation errors\n */\nexport function validateConfig(schema, config) {\n const validationErrors = []\n\n // Check errors for each schema\n for (const [name, conditions] of Object.entries(schema)) {\n const errors = []\n\n // Check errors for each schema condition\n if (Array.isArray(conditions)) {\n for (const { required, errorMessage } of conditions) {\n if (!required.every((key) => !!config[key])) {\n errors.push(errorMessage) // Missing config key value\n }\n }\n\n // Check one condition passes or add errors\n if (name === 'anyOf' && !(conditions.length - errors.length >= 1)) {\n validationErrors.push(...errors)\n }\n }\n }\n\n return validationErrors\n}\n\n/**\n * Check for an array\n *\n * @internal\n * @param {unknown} option - Option to check\n * @returns {boolean} Whether the option is an array\n */\nfunction isArray(option) {\n return Array.isArray(option)\n}\n\n/**\n * Check for an object\n *\n * @internal\n * @param {unknown} option - Option to check\n * @returns {boolean} Whether the option is an object\n */\nfunction isObject(option) {\n return !!option && typeof option === 'object' && !isArray(option)\n}\n\n/**\n * Schema for component config\n *\n * @typedef {object} Schema\n * @property {{ [field: string]: SchemaProperty | undefined }} properties - Schema properties\n * @property {SchemaCondition[]} [anyOf] - List of schema conditions\n */\n\n/**\n * Schema property for component config\n *\n * @typedef {object} SchemaProperty\n * @property {'string' | 'boolean' | 'number' | 'object'} type - Property type\n */\n\n/**\n * Schema condition for component config\n *\n * @typedef {object} SchemaCondition\n * @property {string[]} required - List of required config fields\n * @property {string} errorMessage - Error message when required config fields not provided\n */\n\n/**\n * @internal\n * @typedef {keyof ObjectNested} NestedKey\n * @typedef {{ [key: string]: string | boolean | number | ObjectNested | undefined }} ObjectNested\n */\n","import { extractConfigByNamespace } from './index.mjs'\nimport { normaliseString } from './normalise-string.mjs'\n\n/**\n * Normalise dataset\n *\n * Loop over an object and normalise each value using {@link normaliseString},\n * optionally expanding nested `i18n.field`\n *\n * @internal\n * @param {{ schema: Schema }} Component - Component class\n * @param {DOMStringMap} dataset - HTML element dataset\n * @returns {ObjectNested} Normalised dataset\n */\nexport function normaliseDataset(Component, dataset) {\n const out = /** @type {ReturnType<typeof normaliseDataset>} */ ({})\n\n // Normalise top-level dataset ('data-*') values using schema types\n for (const [field, property] of Object.entries(Component.schema.properties)) {\n if (field in dataset) {\n out[field] = normaliseString(dataset[field], property)\n }\n\n /**\n * Extract and normalise nested object values automatically using\n * {@link normaliseString} but only schema object types are allowed\n */\n if (property?.type === 'object') {\n out[field] = extractConfigByNamespace(Component, dataset, field)\n }\n }\n\n return out\n}\n\n/**\n * @internal\n * @typedef {import('./index.mjs').ObjectNested} ObjectNested\n * @typedef {import('./index.mjs').Schema} Schema\n */\n","/**\n * GOV.UK Frontend error\n *\n * A base class for `Error`s thrown by GOV.UK Frontend.\n *\n * It is meant to be extended into specific types of errors\n * to be thrown by our code.\n *\n * @example\n * ```js\n * class MissingRootError extends GOVUKFrontendError {\n * // Setting an explicit name is important as extending the class will not\n * // set a new `name` on the subclass. The `name` property is important\n * // to ensure intelligible error names even if the class name gets\n * // mangled by a minifier\n * name = \"MissingRootError\"\n * }\n * ```\n * @abstract\n */\nexport class GOVUKFrontendError extends Error {\n name = 'GOVUKFrontendError'\n}\n\n/**\n * Indicates that GOV.UK Frontend is not supported\n */\nexport class SupportError extends GOVUKFrontendError {\n name = 'SupportError'\n\n /**\n * Checks if GOV.UK Frontend is supported on this page\n *\n * @param {HTMLElement | null} [$scope] - HTML element `<body>` checked for browser support\n */\n constructor($scope = document.body) {\n const supportMessage =\n 'noModule' in HTMLScriptElement.prototype\n ? 'GOV.UK Frontend initialised without `<body class=\"govuk-frontend-supported\">` from template `<script>` snippet'\n : 'GOV.UK Frontend is not supported in this browser'\n\n super(\n $scope\n ? supportMessage\n : 'GOV.UK Frontend initialised without `<script type=\"module\">`'\n )\n }\n}\n\n/**\n * Indicates that a component has received an illegal configuration\n */\nexport class ConfigError extends GOVUKFrontendError {\n name = 'ConfigError'\n}\n\n/**\n * Indicates an issue with an element (possibly `null` or `undefined`)\n */\nexport class ElementError extends GOVUKFrontendError {\n name = 'ElementError'\n\n /**\n * @internal\n * @overload\n * @param {string} message - Element error message\n */\n\n /**\n * @internal\n * @overload\n * @param {ElementErrorOptions} options - Element error options\n */\n\n /**\n * @internal\n * @param {string | ElementErrorOptions} messageOrOptions - Element error message or options\n */\n constructor(messageOrOptions) {\n let message = typeof messageOrOptions === 'string' ? messageOrOptions : ''\n\n // Build message from options\n if (typeof messageOrOptions === 'object') {\n const { componentName, identifier, element, expectedType } =\n messageOrOptions\n\n // Add prefix and identifier\n message = `${componentName}: ${identifier}`\n\n // Append reason\n message += element\n ? ` is not of type ${expectedType ?? 'HTMLElement'}`\n : ' not found'\n }\n\n super(message)\n }\n}\n\n/**\n * Element error options\n *\n * @internal\n * @typedef {object} ElementErrorOptions\n * @property {string} componentName - The name of the component throwing the error\n * @property {string} identifier - An identifier that'll let the user understand which element has an error. This is whatever makes the most sense\n * @property {Element | null} [element] - The element in error\n * @property {string} [expectedType] - The type that was expected for the identifier\n */\n","import { isSupported } from './common/index.mjs'\nimport { SupportError } from './errors/index.mjs'\n\n/**\n * Base Component class\n *\n * Centralises the behaviours shared by our components\n *\n * @internal\n * @abstract\n */\nexport class GOVUKFrontendComponent {\n /**\n * Constructs a new component, validating that GOV.UK Frontend is supported\n *\n * @internal\n */\n constructor() {\n this.checkSupport()\n }\n\n /**\n * Validates whether GOV.UK Frontend is supported\n *\n * @private\n * @throws {SupportError} when GOV.UK Frontend is not supported\n */\n checkSupport() {\n if (!isSupported()) {\n throw new SupportError()\n }\n }\n}\n","/**\n * Internal support for selecting messages to render, with placeholder\n * interpolation and locale-aware number formatting and pluralisation\n *\n * @internal\n */\nexport class I18n {\n translations\n locale\n\n /**\n * @internal\n * @param {{ [key: string]: string | TranslationPluralForms }} translations - Key-value pairs of the translation strings to use.\n * @param {object} [config] - Configuration options for the function.\n * @param {string | null} [config.locale] - An overriding locale for the PluralRules functionality.\n */\n constructor(translations = {}, config = {}) {\n // Make list of translations available throughout function\n this.translations = translations\n\n // The locale to use for PluralRules and NumberFormat\n this.locale = config.locale ?? (document.documentElement.lang || 'en')\n }\n\n /**\n * The most used function - takes the key for a given piece of UI text and\n * returns the appropriate string.\n *\n * @internal\n * @param {string} lookupKey - The lookup key of the string to use.\n * @param {{ [key: string]: unknown }} [options] - Any options passed with the translation string, e.g: for string interpolation.\n * @returns {string} The appropriate translation string.\n * @throws {Error} Lookup key required\n * @throws {Error} Options required for `${}` placeholders\n */\n t(lookupKey, options) {\n if (!lookupKey) {\n // Print a console error if no lookup key has been provided\n throw new Error('i18n: lookup key missing')\n }\n\n // Fetch the translation for that lookup key\n let translation = this.translations[lookupKey]\n\n // If the `count` option is set, determine which plural suffix is needed and\n // change the lookupKey to match. We check to see if it's numeric instead of\n // falsy, as this could legitimately be 0.\n if (typeof options?.count === 'number' && typeof translation === 'object') {\n const translationPluralForm =\n translation[this.getPluralSuffix(lookupKey, options.count)]\n\n // Update translation with plural suffix\n if (translationPluralForm) {\n translation = translationPluralForm\n }\n }\n\n if (typeof translation === 'string') {\n // Check for ${} placeholders in the translation string\n if (translation.match(/%{(.\\S+)}/)) {\n if (!options) {\n throw new Error(\n 'i18n: cannot replace placeholders in string if no option data provided'\n )\n }\n\n return this.replacePlaceholders(translation, options)\n }\n\n return translation\n }\n\n // If the key wasn't found in our translations object,\n // return the lookup key itself as the fallback\n return lookupKey\n }\n\n /**\n * Takes a translation string with placeholders, and replaces the placeholders\n * with the provided data\n *\n * @internal\n * @param {string} translationString - The translation string\n * @param {{ [key: string]: unknown }} options - Any options passed with the translation string, e.g: for string interpolation.\n * @returns {string} The translation string to output, with $\\{\\} placeholders replaced\n */\n replacePlaceholders(translationString, options) {\n const formatter = Intl.NumberFormat.supportedLocalesOf(this.locale).length\n ? new Intl.NumberFormat(this.locale)\n : undefined\n\n return translationString.replace(\n /%{(.\\S+)}/g,\n\n /**\n * Replace translation string placeholders\n *\n * @internal\n * @param {string} placeholderWithBraces - Placeholder with braces\n * @param {string} placeholderKey - Placeholder key\n * @returns {string} Placeholder value\n */\n function (placeholderWithBraces, placeholderKey) {\n if (Object.prototype.hasOwnProperty.call(options, placeholderKey)) {\n const placeholderValue = options[placeholderKey]\n\n // If a user has passed `false` as the value for the placeholder\n // treat it as though the value should not be displayed\n if (\n placeholderValue === false ||\n (typeof placeholderValue !== 'number' &&\n typeof placeholderValue !== 'string')\n ) {\n return ''\n }\n\n // If the placeholder's value is a number, localise the number formatting\n if (typeof placeholderValue === 'number') {\n return formatter\n ? formatter.format(placeholderValue)\n : `${placeholderValue}`\n }\n\n return placeholderValue\n }\n\n throw new Error(\n `i18n: no data found to replace ${placeholderWithBraces} placeholder in string`\n )\n }\n )\n }\n\n /**\n * Check to see if the browser supports Intl.PluralRules\n *\n * It requires all conditions to be met in order to be supported:\n * - The implementation of Intl supports PluralRules (NOT true in Safari 10–12)\n * - The browser/OS has plural rules for the current locale (browser dependent)\n *\n * {@link https://browsersl.ist/#q=supports+es6-module+and+not+supports+intl-pluralrules}\n *\n * @internal\n * @returns {boolean} Returns true if all conditions are met. Returns false otherwise.\n */\n hasIntlPluralRulesSupport() {\n return Boolean(\n 'PluralRules' in window.Intl &&\n Intl.PluralRules.supportedLocalesOf(this.locale).length\n )\n }\n\n /**\n * Get the appropriate suffix for the plural form.\n *\n * Uses Intl.PluralRules (or our own fallback implementation) to get the\n * 'preferred' form to use for the given count.\n *\n * Checks that a translation has been provided for that plural form – if it\n * hasn't, it'll fall back to the 'other' plural form (unless that doesn't exist\n * either, in which case an error will be thrown)\n *\n * @internal\n * @param {string} lookupKey - The lookup key of the string to use.\n * @param {number} count - Number used to determine which pluralisation to use.\n * @returns {PluralRule} The suffix associated with the correct pluralisation for this locale.\n * @throws {Error} Plural form `.other` required when preferred plural form is missing\n */\n getPluralSuffix(lookupKey, count) {\n // Validate that the number is actually a number.\n //\n // Number(count) will turn anything that can't be converted to a Number type\n // into 'NaN'. isFinite filters out NaN, as it isn't a finite number.\n count = Number(count)\n if (!isFinite(count)) {\n return 'other'\n }\n\n // Fetch the translation for that lookup key\n const translation = this.translations[lookupKey]\n\n // Check to verify that all the requirements for Intl.PluralRules are met.\n // If so, we can use that instead of our custom implementation. Otherwise,\n // use the hardcoded fallback.\n const preferredForm = this.hasIntlPluralRulesSupport()\n ? new Intl.PluralRules(this.locale).select(count)\n : this.selectPluralFormUsingFallbackRules(count)\n\n // Use the correct plural form if provided\n if (typeof translation === 'object') {\n if (preferredForm in translation) {\n return preferredForm\n // Fall back to `other` if the plural form is missing, but log a warning\n // to the console\n } else if ('other' in translation) {\n console.warn(\n `i18n: Missing plural form \".${preferredForm}\" for \"${this.locale}\" locale. Falling back to \".other\".`\n )\n\n return 'other'\n }\n }\n\n // If the required `other` plural form is missing, all we can do is error\n throw new Error(\n `i18n: Plural form \".other\" is required for \"${this.locale}\" locale`\n )\n }\n\n /**\n * Get the plural form using our fallback implementation\n *\n * This is split out into a separate function to make it easier to test the\n * fallback behaviour in an environment where Intl.PluralRules exists.\n *\n * @internal\n * @param {number} count - Number used to determine which pluralisation to use.\n * @returns {PluralRule} The pluralisation form for count in this locale.\n */\n selectPluralFormUsingFallbackRules(count) {\n // Currently our custom code can only handle positive integers, so let's\n // make sure our number is one of those.\n count = Math.abs(Math.floor(count))\n\n const ruleset = this.getPluralRulesForLocale()\n\n if (ruleset) {\n return I18n.pluralRules[ruleset](count)\n }\n\n return 'other'\n }\n\n /**\n * Work out which pluralisation rules to use for the current locale\n *\n * The locale may include a regional indicator (such as en-GB), but we don't\n * usually care about this part, as pluralisation rules are usually the same\n * regardless of region. There are exceptions, however, (e.g. Portuguese) so\n * this searches by both the full and shortened locale codes, just to be sure.\n *\n * @internal\n * @returns {string | undefined} The name of the pluralisation rule to use (a key for one\n * of the functions in this.pluralRules)\n */\n getPluralRulesForLocale() {\n const localeShort = this.locale.split('-')[0]\n\n // Look through the plural rules map to find which `pluralRule` is\n // appropriate for our current `locale`.\n for (const pluralRule in I18n.pluralRulesMap) {\n const languages = I18n.pluralRulesMap[pluralRule]\n if (languages.includes(this.locale) || languages.includes(localeShort)) {\n return pluralRule\n }\n }\n }\n\n /**\n * Map of plural rules to languages where those rules apply.\n *\n * Note: These groups are named for the most dominant or recognisable language\n * that uses each system. The groupings do not imply that the languages are\n * related to one another. Many languages have evolved the same systems\n * independently of one another.\n *\n * Code to support more languages can be found in the i18n spike:\n * {@link https://github.com/alphagov/govuk-frontend/blob/spike-i18n-support/src/govuk/i18n.mjs}\n *\n * Languages currently supported:\n *\n * Arabic: Arabic (ar)\n * Chinese: Burmese (my), Chinese (zh), Indonesian (id), Japanese (ja),\n * Javanese (jv), Korean (ko), Malay (ms), Thai (th), Vietnamese (vi)\n * French: Armenian (hy), Bangla (bn), French (fr), Gujarati (gu), Hindi (hi),\n * Persian Farsi (fa), Punjabi (pa), Zulu (zu)\n * German: Afrikaans (af), Albanian (sq), Azerbaijani (az), Basque (eu),\n * Bulgarian (bg), Catalan (ca), Danish (da), Dutch (nl), English (en),\n * Estonian (et), Finnish (fi), Georgian (ka), German (de), Greek (el),\n * Hungarian (hu), Luxembourgish (lb), Norwegian (no), Somali (so),\n * Swahili (sw), Swedish (sv), Tamil (ta), Telugu (te), Turkish (tr),\n * Urdu (ur)\n * Irish: Irish Gaelic (ga)\n * Russian: Russian (ru), Ukrainian (uk)\n * Scottish: Scottish Gaelic (gd)\n * Spanish: European Portuguese (pt-PT), Italian (it), Spanish (es)\n * Welsh: Welsh (cy)\n *\n * @internal\n * @type {{ [key: string]: string[] }}\n */\n static pluralRulesMap = {\n arabic: ['ar'],\n chinese: ['my', 'zh', 'id', 'ja', 'jv', 'ko', 'ms', 'th', 'vi'],\n french: ['hy', 'bn', 'fr', 'gu', 'hi', 'fa', 'pa', 'zu'],\n german: [\n 'af',\n 'sq',\n 'az',\n 'eu',\n 'bg',\n 'ca',\n 'da',\n 'nl',\n 'en',\n 'et',\n 'fi',\n 'ka',\n 'de',\n 'el',\n 'hu',\n 'lb',\n 'no',\n 'so',\n 'sw',\n 'sv',\n 'ta',\n 'te',\n 'tr',\n 'ur'\n ],\n irish: ['ga'],\n russian: ['ru', 'uk'],\n scottish: ['gd'],\n spanish: ['pt-PT', 'it', 'es'],\n welsh: ['cy']\n }\n\n /**\n * Different pluralisation rule sets\n *\n * Returns the appropriate suffix for the plural form associated with `n`.\n * Possible suffixes: 'zero', 'one', 'two', 'few', 'many', 'other' (the actual\n * meaning of each differs per locale). 'other' should always exist, even in\n * languages without plurals, such as Chinese.\n * {@link https://cldr.unicode.org/index/cldr-spec/plural-rules}\n *\n * The count must be a positive integer. Negative numbers and decimals aren't accounted for\n *\n * @internal\n * @type {{ [key: string]: (count: number) => PluralRule }}\n */\n static pluralRules = {\n arabic(n) {\n if (n === 0) {\n return 'zero'\n }\n if (n === 1) {\n return 'one'\n }\n if (n === 2) {\n return 'two'\n }\n if (n % 100 >= 3 && n % 100 <= 10) {\n return 'few'\n }\n if (n % 100 >= 11 && n % 100 <= 99) {\n return 'many'\n }\n return 'other'\n },\n chinese() {\n return 'other'\n },\n french(n) {\n return n === 0 || n === 1 ? 'one' : 'other'\n },\n german(n) {\n return n === 1 ? 'one' : 'other'\n },\n irish(n) {\n if (n === 1) {\n return 'one'\n }\n if (n === 2) {\n return 'two'\n }\n if (n >= 3 && n <= 6) {\n return 'few'\n }\n if (n >= 7 && n <= 10) {\n return 'many'\n }\n return 'other'\n },\n russian(n) {\n const lastTwo = n % 100\n const last = lastTwo % 10\n if (last === 1 && lastTwo !== 11) {\n return 'one'\n }\n if (last >= 2 && last <= 4 && !(lastTwo >= 12 && lastTwo <= 14)) {\n return 'few'\n }\n if (\n last === 0 ||\n (last >= 5 && last <= 9) ||\n (lastTwo >= 11 && lastTwo <= 14)\n ) {\n return 'many'\n }\n // Note: The 'other' suffix is only used by decimal numbers in Russian.\n // We don't anticipate it being used, but it's here for consistency.\n return 'other'\n },\n scottish(n) {\n if (n === 1 || n === 11) {\n return 'one'\n }\n if (n === 2 || n === 12) {\n return 'two'\n }\n if ((n >= 3 && n <= 10) || (n >= 13 && n <= 19)) {\n return 'few'\n }\n return 'other'\n },\n spanish(n) {\n if (n === 1) {\n return 'one'\n }\n if (n % 1000000 === 0 && n !== 0) {\n return 'many'\n }\n return 'other'\n },\n welsh(n) {\n if (n === 0) {\n return 'zero'\n }\n if (n === 1) {\n return 'one'\n }\n if (n === 2) {\n return 'two'\n }\n if (n === 3) {\n return 'few'\n }\n if (n === 6) {\n return 'many'\n }\n return 'other'\n }\n }\n}\n\n/**\n * Plural rule category mnemonic tags\n *\n * @internal\n * @typedef {'zero' | 'one' | 'two' | 'few' | 'many' | 'other'} PluralRule\n */\n\n/**\n * Translated message by plural rule they correspond to.\n *\n * Allows to group pluralised messages under a single key when passing\n * translations to a component's constructor\n *\n * @internal\n * @typedef {object} TranslationPluralForms\n * @property {string} [other] - General plural form\n * @property {string} [zero] - Plural form used with 0\n * @property {string} [one] - Plural form used with 1\n * @property {string} [two] - Plural form used with 2\n * @property {string} [few] - Plural form used for a few\n * @property {string} [many] - Plural form used for many\n */\n","import { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Accordion component\n *\n * This allows a collection of sections to be collapsed by default, showing only\n * their headers. Sections can be expanded or collapsed individually by clicking\n * their headers. A \"Show all sections\" button is also added to the top of the\n * accordion, which switches to \"Hide all sections\" when all the sections are\n * expanded.\n *\n * The state of each section is saved to the DOM via the `aria-expanded`\n * attribute, which also provides accessibility.\n *\n * @preserve\n */\nexport class Accordion extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /**\n * @private\n * @type {AccordionConfig}\n */\n config\n\n /** @private */\n i18n\n\n /** @private */\n controlsClass = 'govuk-accordion__controls'\n\n /** @private */\n showAllClass = 'govuk-accordion__show-all'\n\n /** @private */\n showAllTextClass = 'govuk-accordion__show-all-text'\n\n /** @private */\n sectionClass = 'govuk-accordion__section'\n\n /** @private */\n sectionExpandedClass = 'govuk-accordion__section--expanded'\n\n /** @private */\n sectionButtonClass = 'govuk-accordion__section-button'\n\n /** @private */\n sectionHeaderClass = 'govuk-accordion__section-header'\n\n /** @private */\n sectionHeadingClass = 'govuk-accordion__section-heading'\n\n /** @private */\n sectionHeadingDividerClass = 'govuk-accordion__section-heading-divider'\n\n /** @private */\n sectionHeadingTextClass = 'govuk-accordion__section-heading-text'\n\n /** @private */\n sectionHeadingTextFocusClass = 'govuk-accordion__section-heading-text-focus'\n\n /** @private */\n sectionShowHideToggleClass = 'govuk-accordion__section-toggle'\n\n /** @private */\n sectionShowHideToggleFocusClass = 'govuk-accordion__section-toggle-focus'\n\n /** @private */\n sectionShowHideTextClass = 'govuk-accordion__section-toggle-text'\n\n /** @private */\n upChevronIconClass = 'govuk-accordion-nav__chevron'\n\n /** @private */\n downChevronIconClass = 'govuk-accordion-nav__chevron--down'\n\n /** @private */\n sectionSummaryClass = 'govuk-accordion__section-summary'\n\n /** @private */\n sectionSummaryFocusClass = 'govuk-accordion__section-summary-focus'\n\n /** @private */\n sectionContentClass = 'govuk-accordion__section-content'\n\n /** @private */\n $sections\n\n /**\n * @private\n * @type {HTMLButtonElement | null}\n */\n $showAllButton = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $showAllIcon = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $showAllText = null\n\n /**\n * @param {Element | null} $module - HTML element to use for accordion\n * @param {AccordionConfig} [config] - Accordion config\n */\n constructor($module, config = {}) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Accordion',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n this.$module = $module\n\n this.config = mergeConfigs(\n Accordion.defaults,\n config,\n normaliseDataset(Accordion, $module.dataset)\n )\n\n this.i18n = new I18n(this.config.i18n)\n\n const $sections = this.$module.querySelectorAll(`.${this.sectionClass}`)\n if (!$sections.length) {\n throw new ElementError({\n componentName: 'Accordion',\n identifier: `Sections (\\`<div class=\"${this.sectionClass}\">\\`)`\n })\n }\n\n this.$sections = $sections\n\n this.initControls()\n this.initSectionHeaders()\n\n this.updateShowAllButton(this.areAllSectionsOpen())\n }\n\n /**\n * Initialise controls and set attributes\n *\n * @private\n */\n initControls() {\n // Create \"Show all\" button and set attributes\n this.$showAllButton = document.createElement('button')\n this.$showAllButton.setAttribute('type', 'button')\n this.$showAllButton.setAttribute('class', this.showAllClass)\n this.$showAllButton.setAttribute('aria-expanded', 'false')\n\n // Create icon, add to element\n this.$showAllIcon = document.createElement('span')\n this.$showAllIcon.classList.add(this.upChevronIconClass)\n this.$showAllButton.appendChild(this.$showAllIcon)\n\n // Create control wrapper and add controls to it\n const $accordionControls = document.createElement('div')\n $accordionControls.setAttribute('class', this.controlsClass)\n $accordionControls.appendChild(this.$showAllButton)\n this.$module.insertBefore($accordionControls, this.$module.firstChild)\n\n // Build additional wrapper for Show all toggle text and place after icon\n this.$showAllText = document.createElement('span')\n this.$showAllText.classList.add(this.showAllTextClass)\n this.$showAllButton.appendChild(this.$showAllText)\n\n // Handle click events on the show/hide all button\n this.$showAllButton.addEventListener('click', () =>\n this.onShowOrHideAllToggle()\n )\n\n // Handle 'beforematch' events, if the user agent supports them\n if ('onbeforematch' in document) {\n document.addEventListener('beforematch', (event) =>\n this.onBeforeMatch(event)\n )\n }\n }\n\n /**\n * Initialise section headers\n *\n * @private\n */\n initSectionHeaders() {\n this.$sections.forEach(($section, i) => {\n const $header = $section.querySelector(`.${this.sectionHeaderClass}`)\n if (!$header) {\n throw new ElementError({\n componentName: 'Accordion',\n identifier: `Section headers (\\`<div class=\"${this.sectionHeaderClass}\">\\`)`\n })\n }\n\n // Set header attributes\n this.constructHeaderMarkup($header, i)\n this.setExpanded(this.isExpanded($section), $section)\n\n // Handle events\n $header.addEventListener('click', () => this.onSectionToggle($section))\n\n // See if there is any state stored in sessionStorage and set the sections\n // to open or closed.\n this.setInitialState($section)\n })\n }\n\n /**\n * Construct section header\n *\n * @private\n * @param {Element} $header - Section header\n * @param {number} index - Section index\n */\n constructHeaderMarkup($header, index) {\n const $span = $header.querySelector(`.${this.sectionButtonClass}`)\n const $heading = $header.querySelector(`.${this.sectionHeadingClass}`)\n const $summary = $header.querySelector(`.${this.sectionSummaryClass}`)\n\n if (!$heading) {\n throw new ElementError({\n componentName: 'Accordion',\n identifier: `Section heading (\\`.${this.sectionHeadingClass}\\`)`\n })\n }\n\n if (!$span) {\n throw new ElementError({\n componentName: 'Accordion',\n identifier: `Section button placeholder (\\`<span class=\"${this.sectionButtonClass}\">\\`)`\n })\n }\n\n // Create a button element that will replace the\n // '.govuk-accordion__section-button' span\n const $button = document.createElement('button')\n $button.setAttribute('type', 'button')\n $button.setAttribute(\n 'aria-controls',\n `${this.$module.id}-content-${index + 1}`\n )\n\n // Copy all attributes from $span to $button (except `id`, which gets added\n // to the `$headingText` element)\n for (const attr of Array.from($span.attributes)) {\n if (attr.name !== 'id') {\n $button.setAttribute(attr.name, attr.value)\n }\n }\n\n // Create container for heading text so it can be styled\n const $headingText = document.createElement('span')\n $headingText.classList.add(this.sectionHeadingTextClass)\n // Copy the span ID to the heading text to allow it to be referenced by\n // `aria-labelledby` on the hidden content area without \"Show this section\"\n $headingText.id = $span.id\n\n // Create an inner heading text container to limit the width of the focus\n // state\n const $headingTextFocus = document.createElement('span')\n $headingTextFocus.classList.add(this.sectionHeadingTextFocusClass)\n $headingText.appendChild($headingTextFocus)\n // span could contain HTML elements\n // (see https://www.w3.org/TR/2011/WD-html5-20110525/content-models.html#phrasing-content)\n Array.from($span.childNodes).forEach(($child) =>\n $headingTextFocus.appendChild($child)\n )\n\n // Create container for show / hide icons and text.\n const $showHideToggle = document.createElement('span')\n $showHideToggle.classList.add(this.sectionShowHideToggleClass)\n // Tell Google not to index the 'show' text as part of the heading. Must be\n // set on the element before it's added to the DOM.\n // See https://developers.google.com/search/docs/advanced/robots/robots_meta_tag#data-nosnippet-attr\n $showHideToggle.setAttribute('data-nosnippet', '')\n // Create an inner container to limit the width of the focus state\n const $showHideToggleFocus = document.createElement('span')\n $showHideToggleFocus.classList.add(this.sectionShowHideToggleFocusClass)\n $showHideToggle.appendChild($showHideToggleFocus)\n // Create wrapper for the show / hide text. Append text after the show/hide icon\n const $showHideText = document.createElement('span')\n const $showHideIcon = document.createElement('span')\n $showHideIcon.classList.add(this.upChevronIconClass)\n $showHideToggleFocus.appendChild($showHideIcon)\n $showHideText.classList.add(this.sectionShowHideTextClass)\n $showHideToggleFocus.appendChild($showHideText)\n\n // Append elements to the button:\n // 1. Heading text\n // 2. Punctuation\n // 3. (Optional: Summary line followed by punctuation)\n // 4. Show / hide toggle\n $button.appendChild($headingText)\n $button.appendChild(this.getButtonPunctuationEl())\n\n // If summary content exists add to DOM in correct order\n if ($summary) {\n // Create a new `span` element and copy the summary line content from the\n // original `div` to the new `span`. This is because the summary line text\n // is now inside a button element, which can only contain phrasing\n // content.\n const $summarySpan = document.createElement('span')\n // Create an inner summary container to limit the width of the summary\n // focus state\n const $summarySpanFocus = document.createElement('span')\n $summarySpanFocus.classList.add(this.sectionSummaryFocusClass)\n $summarySpan.appendChild($summarySpanFocus)\n\n // Get original attributes, and pass them to the replacement\n for (const attr of Array.from($summary.attributes)) {\n $summarySpan.setAttribute(attr.name, attr.value)\n }\n\n // Copy original contents of summary to the new summary span\n Array.from($summary.childNodes).forEach(($child) =>\n $summarySpanFocus.appendChild($child)\n )\n\n // Replace the original summary `div` with the new summary `span`\n $summary.remove()\n\n $button.appendChild($summarySpan)\n $button.appendChild(this.getButtonPunctuationEl())\n }\n\n $button.appendChild($showHideToggle)\n\n $heading.removeChild($span)\n $heading.appendChild($button)\n }\n\n /**\n * When a section is opened by the user agent via the 'beforematch' event\n *\n * @private\n * @param {Event} event - Generic event\n */\n onBeforeMatch(event) {\n const $fragment = event.target\n\n // Handle elements with `.closest()` support only\n if (!($fragment instanceof Element)) {\n return\n }\n\n // Handle when fragment is inside section\n const $section = $fragment.closest(`.${this.sectionClass}`)\n if ($section) {\n this.setExpanded(true, $section)\n }\n }\n\n /**\n * When section toggled, set and store state\n *\n * @private\n * @param {Element} $section - Section element\n */\n onSectionToggle($section) {\n const nowExpanded = !this.isExpanded($section)\n this.setExpanded(nowExpanded, $section)\n\n // Store the state in sessionStorage when a change is triggered\n this.storeState($section, nowExpanded)\n }\n\n /**\n * When Open/Close All toggled, set and store state\n *\n * @private\n */\n onShowOrHideAllToggle() {\n const nowExpanded = !this.areAllSectionsOpen()\n\n this.$sections.forEach(($section) => {\n this.setExpanded(nowExpanded, $section)\n this.storeState($section, nowExpanded)\n })\n\n this.updateShowAllButton(nowExpanded)\n }\n\n /**\n * Set section attributes when opened/closed\n *\n * @private\n * @param {boolean} expanded - Section expanded\n * @param {Element} $section - Section element\n */\n setExpanded(expanded, $section) {\n const $showHideIcon = $section.querySelector(`.${this.upChevronIconClass}`)\n const $showHideText = $section.querySelector(\n `.${this.sectionShowHideTextClass}`\n )\n const $button = $section.querySelector(`.${this.sectionButtonClass}`)\n const $content = $section.querySelector(`.${this.sectionContentClass}`)\n\n if (!$content) {\n throw new ElementError({\n componentName: 'Accordion',\n identifier: `Section content (\\`<div class=\"${this.sectionContentClass}\">\\`)`\n })\n }\n\n if (!$showHideIcon || !$showHideText || !$button) {\n // Return early for elements we create\n return\n }\n\n const newButtonText = expanded\n ? this.i18n.t('hideSection')\n : this.i18n.t('showSection')\n\n $showHideText.textContent = newButtonText\n $button.setAttribute('aria-expanded', `${expanded}`)\n\n // Update aria-label combining\n const ariaLabelParts = []\n\n const $headingText = $section.querySelector(\n `.${this.sectionHeadingTextClass}`\n )\n if ($headingText) {\n ariaLabelParts.push(`${$headingText.textContent}`.trim())\n }\n\n const $summary = $section.querySelector(`.${this.sectionSummaryClass}`)\n if ($summary) {\n ariaLabelParts.push(`${$summary.textContent}`.trim())\n }\n\n const ariaLabelMessage = expanded\n ? this.i18n.t('hideSectionAriaLabel')\n : this.i18n.t('showSectionAriaLabel')\n ariaLabelParts.push(ariaLabelMessage)\n\n /*\n * Join with a comma to add pause for assistive technology.\n * Example: [heading]Section A ,[pause] Show this section.\n * https://accessibility.blog.gov.uk/2017/12/18/what-working-on-gov-uk-navigation-taught-us-about-accessibility/\n */\n $button.setAttribute('aria-label', ariaLabelParts.join(' , '))\n\n // Swap icon, change class\n if (expanded) {\n $content.removeAttribute('hidden')\n $section.classList.add(this.sectionExpandedClass)\n $showHideIcon.classList.remove(this.downChevronIconClass)\n } else {\n $content.setAttribute('hidden', 'until-found')\n $section.classList.remove(this.sectionExpandedClass)\n $showHideIcon.classList.add(this.downChevronIconClass)\n }\n\n // See if \"Show all sections\" button text should be updated\n this.updateShowAllButton(this.areAllSectionsOpen())\n }\n\n /**\n * Get state of section\n *\n * @private\n * @param {Element} $section - Section element\n * @returns {boolean} True if expanded\n */\n isExpanded($section) {\n return $section.classList.contains(this.sectionExpandedClass)\n }\n\n /**\n * Check if all sections are open\n *\n * @private\n * @returns {boolean} True if all sections are open\n */\n areAllSectionsOpen() {\n return Array.from(this.$sections).every(($section) =>\n this.isExpanded($section)\n )\n }\n\n /**\n * Update \"Show all sections\" button\n *\n * @private\n * @param {boolean} expanded - Section expanded\n */\n updateShowAllButton(expanded) {\n if (!this.$showAllButton || !this.$showAllText || !this.$showAllIcon) {\n return\n }\n\n this.$showAllButton.setAttribute('aria-expanded', expanded.toString())\n this.$showAllText.textContent = expanded\n ? this.i18n.t('hideAllSections')\n : this.i18n.t('showAllSections')\n this.$showAllIcon.classList.toggle(this.downChevronIconClass, !expanded)\n }\n\n /**\n * Get the identifier for a section\n *\n * We need a unique way of identifying each content in the Accordion.\n * Since an `#id` should be unique and an `id` is required for `aria-`\n * attributes `id` can be safely used.\n *\n * @param {Element} $section - Section element\n * @returns {string | undefined | null} Identifier for section\n */\n getIdentifier($section) {\n const $button = $section.querySelector(`.${this.sectionButtonClass}`)\n\n return $button?.getAttribute('aria-controls')\n }\n\n /**\n * Set the state of the accordions in sessionStorage\n *\n * @private\n * @param {Element} $section - Section element\n * @param {boolean} isExpanded - Whether the section is expanded\n */\n storeState($section, isExpanded) {\n if (!this.config.rememberExpanded) {\n return\n }\n\n const id = this.getIdentifier($section)\n\n if (id) {\n try {\n window.sessionStorage.setItem(id, isExpanded.toString())\n } catch (exception) {}\n }\n }\n\n /**\n * Read the state of the accordions from sessionStorage\n *\n * @private\n * @param {Element} $section - Section element\n */\n setInitialState($section) {\n if (!this.config.rememberExpanded) {\n return\n }\n\n const id = this.getIdentifier($section)\n\n if (id) {\n try {\n const state = window.sessionStorage.getItem(id)\n\n if (state !== null) {\n this.setExpanded(state === 'true', $section)\n }\n } catch (exception) {}\n }\n }\n\n /**\n * Create an element to improve semantics of the section button with\n * punctuation\n *\n * Adding punctuation to the button can also improve its general semantics by\n * dividing its contents into thematic chunks. See\n * https://github.com/alphagov/govuk-frontend/issues/2327#issuecomment-922957442\n *\n * @private\n * @returns {Element} DOM element\n */\n getButtonPunctuationEl() {\n const $punctuationEl = document.createElement('span')\n $punctuationEl.classList.add(\n 'govuk-visually-hidden',\n this.sectionHeadingDividerClass\n )\n $punctuationEl.textContent = ', '\n return $punctuationEl\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-accordion'\n\n /**\n * Accordion default config\n *\n * @see {@link AccordionConfig}\n * @constant\n * @type {AccordionConfig}\n */\n static defaults = Object.freeze({\n i18n: {\n hideAllSections: 'Hide all sections',\n hideSection: 'Hide',\n hideSectionAriaLabel: 'Hide this section',\n showAllSections: 'Show all sections',\n showSection: 'Show',\n showSectionAriaLabel: 'Show this section'\n },\n rememberExpanded: true\n })\n\n /**\n * Accordion config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' },\n rememberExpanded: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Accordion config\n *\n * @see {@link Accordion.defaults}\n * @typedef {object} AccordionConfig\n * @property {AccordionTranslations} [i18n=Accordion.defaults.i18n] - Accordion translations\n * @property {boolean} [rememberExpanded] - Whether the expanded and collapsed\n * state of each section is remembered and restored when navigating.\n */\n\n/**\n * Accordion translations\n *\n * @see {@link Accordion.defaults.i18n}\n * @typedef {object} AccordionTranslations\n *\n * Messages used by the component for the labels of its buttons. This includes\n * the visible text shown on screen, and text to help assistive technology users\n * for the buttons toggling each section.\n * @property {string} [hideAllSections] - The text content for the 'Hide all\n * sections' button, used when at least one section is expanded.\n * @property {string} [hideSection] - The text content for the 'Hide'\n * button, used when a section is expanded.\n * @property {string} [hideSectionAriaLabel] - The text content appended to the\n * 'Hide' button's accessible name when a section is expanded.\n * @property {string} [showAllSections] - The text content for the 'Show all\n * sections' button, used when all sections are collapsed.\n * @property {string} [showSection] - The text content for the 'Show'\n * button, used when a section is collapsed.\n * @property {string} [showSectionAriaLabel] - The text content appended to the\n * 'Show' button's accessible name when a section is expanded.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\nconst DEBOUNCE_TIMEOUT_IN_SECONDS = 1\n\n/**\n * JavaScript enhancements for the Button component\n *\n * @preserve\n */\nexport class Button extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /**\n * @private\n * @type {ButtonConfig}\n */\n config\n\n /**\n * @private\n * @type {number | null}\n */\n debounceFormSubmitTimer = null\n\n /**\n * @param {Element | null} $module - HTML element to use for button\n * @param {ButtonConfig} [config] - Button config\n */\n constructor($module, config = {}) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Button',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n this.$module = $module\n\n this.config = mergeConfigs(\n Button.defaults,\n config,\n normaliseDataset(Button, $module.dataset)\n )\n\n this.$module.addEventListener('keydown', (event) =>\n this.handleKeyDown(event)\n )\n this.$module.addEventListener('click', (event) => this.debounce(event))\n }\n\n /**\n * Trigger a click event when the space key is pressed\n *\n * Some screen readers tell users they can use the space bar to activate\n * things with the 'button' role, so we need to match the functionality of\n * native HTML buttons.\n *\n * See https://github.com/alphagov/govuk_elements/pull/272#issuecomment-233028270\n *\n * @private\n * @param {KeyboardEvent} event - Keydown event\n */\n handleKeyDown(event) {\n const $target = event.target\n\n // Handle space bar only\n if (event.key !== ' ') {\n return\n }\n\n // Handle elements with [role=\"button\"] only\n if (\n $target instanceof HTMLElement &&\n $target.getAttribute('role') === 'button'\n ) {\n event.preventDefault() // prevent the page from scrolling\n $target.click()\n }\n }\n\n /**\n * Debounce double-clicks\n *\n * If the click quickly succeeds a previous click then nothing will happen.\n * This stops people accidentally causing multiple form submissions by double\n * clicking buttons.\n *\n * @private\n * @param {MouseEvent} event - Mouse click event\n * @returns {undefined | false} Returns undefined, or false when debounced\n */\n debounce(event) {\n // Check the button that was clicked has preventDoubleClick enabled\n if (!this.config.preventDoubleClick) {\n return\n }\n\n // If the timer is still running, prevent the click from submitting the form\n if (this.debounceFormSubmitTimer) {\n event.preventDefault()\n return false\n }\n\n this.debounceFormSubmitTimer = window.setTimeout(() => {\n this.debounceFormSubmitTimer = null\n }, DEBOUNCE_TIMEOUT_IN_SECONDS * 1000)\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-button'\n\n /**\n * Button default config\n *\n * @see {@link ButtonConfig}\n * @constant\n * @type {ButtonConfig}\n */\n static defaults = Object.freeze({\n preventDoubleClick: false\n })\n\n /**\n * Button config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n preventDoubleClick: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Button config\n *\n * @typedef {object} ButtonConfig\n * @property {boolean} [preventDoubleClick=false] - Prevent accidental double\n * clicks on submit buttons from submitting forms multiple times.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","/**\n * Returns the value of the given attribute closest to the given element (including itself)\n *\n * @internal\n * @param {Element} $element - The element to start walking the DOM tree up\n * @param {string} attributeName - The name of the attribute\n * @returns {string | null} Attribute value\n */\nexport function closestAttributeValue($element, attributeName) {\n const $closestElementWithAttribute = $element.closest(`[${attributeName}]`)\n return $closestElementWithAttribute\n ? $closestElementWithAttribute.getAttribute(attributeName)\n : null\n}\n","import { closestAttributeValue } from '../../common/closest-attribute-value.mjs'\nimport { mergeConfigs, validateConfig } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ConfigError, ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Character count component\n *\n * Tracks the number of characters or words in the `.govuk-js-character-count`\n * `<textarea>` inside the element. Displays a message with the remaining number\n * of characters/words available, or the number of characters/words in excess.\n *\n * You can configure the message to only appear after a certain percentage\n * of the available characters/words has been entered.\n *\n * @preserve\n */\nexport class CharacterCount extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /** @private */\n $textarea\n\n /** @private */\n $visibleCountMessage\n\n /** @private */\n $screenReaderCountMessage\n\n /**\n * @private\n * @type {number | null}\n */\n lastInputTimestamp = null\n\n /** @private */\n lastInputValue = ''\n\n /**\n * @private\n * @type {number | null}\n */\n valueChecker = null\n\n /**\n * @private\n * @type {CharacterCountConfig}\n */\n config\n\n /** @private */\n i18n\n\n /** @private */\n maxLength\n\n /**\n * @param {Element | null} $module - HTML element to use for character count\n * @param {CharacterCountConfig} [config] - Character count config\n */\n constructor($module, config = {}) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Character count',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n const $textarea = $module.querySelector('.govuk-js-character-count')\n if (\n !(\n $textarea instanceof HTMLTextAreaElement ||\n $textarea instanceof HTMLInputElement\n )\n ) {\n throw new ElementError({\n componentName: 'Character count',\n element: $textarea,\n expectedType: 'HTMLTextareaElement or HTMLInputElement',\n identifier: 'Form field (`.govuk-js-character-count`)'\n })\n }\n\n // Read config set using dataset ('data-' values)\n const datasetConfig = normaliseDataset(CharacterCount, $module.dataset)\n\n // To ensure data-attributes take complete precedence, even if they change\n // the type of count, we need to reset the `maxlength` and `maxwords` from\n // the JavaScript config.\n //\n // We can't mutate `config`, though, as it may be shared across multiple\n // components inside `initAll`.\n /** @type {CharacterCountConfig} */\n let configOverrides = {}\n if ('maxwords' in datasetConfig || 'maxlength' in datasetConfig) {\n configOverrides = {\n maxlength: undefined,\n maxwords: undefined\n }\n }\n\n this.config = mergeConfigs(\n CharacterCount.defaults,\n config,\n configOverrides,\n datasetConfig\n )\n\n // Check for valid config\n const errors = validateConfig(CharacterCount.schema, this.config)\n if (errors[0]) {\n throw new ConfigError(`Character count: ${errors[0]}`)\n }\n\n this.i18n = new I18n(this.config.i18n, {\n // Read the fallback if necessary rather than have it set in the defaults\n locale: closestAttributeValue($module, 'lang')\n })\n\n // Determine the limit attribute (characters or words)\n this.maxLength = this.config.maxwords ?? this.config.maxlength ?? Infinity\n\n this.$module = $module\n this.$textarea = $textarea\n\n const textareaDescriptionId = `${this.$textarea.id}-info`\n const $textareaDescription = document.getElementById(textareaDescriptionId)\n if (!$textareaDescription) {\n throw new ElementError({\n componentName: 'Character count',\n element: $textareaDescription,\n identifier: `Count message (\\`id=\"${textareaDescriptionId}\"\\`)`\n })\n }\n\n // Inject a description for the textarea if none is present already\n // for when the component was rendered with no maxlength, maxwords\n // nor custom textareaDescriptionText\n if (`${$textareaDescription.textContent}`.match(/^\\s*$/)) {\n $textareaDescription.textContent = this.i18n.t('textareaDescription', {\n count: this.maxLength\n })\n }\n\n // Move the textarea description to be immediately after the textarea\n // Kept for backwards compatibility\n this.$textarea.insertAdjacentElement('afterend', $textareaDescription)\n\n // Create the *screen reader* specific live-updating counter\n // This doesn't need any styling classes, as it is never visible\n const $screenReaderCountMessage = document.createElement('div')\n $screenReaderCountMessage.className =\n 'govuk-character-count__sr-status govuk-visually-hidden'\n $screenReaderCountMessage.setAttribute('aria-live', 'polite')\n this.$screenReaderCountMessage = $screenReaderCountMessage\n $textareaDescription.insertAdjacentElement(\n 'afterend',\n $screenReaderCountMessage\n )\n\n // Create our live-updating counter element, copying the classes from the\n // textarea description for backwards compatibility as these may have been\n // configured\n const $visibleCountMessage = document.createElement('div')\n $visibleCountMessage.className = $textareaDescription.className\n $visibleCountMessage.classList.add('govuk-character-count__status')\n $visibleCountMessage.setAttribute('aria-hidden', 'true')\n this.$visibleCountMessage = $visibleCountMessage\n $textareaDescription.insertAdjacentElement('afterend', $visibleCountMessage)\n\n // Hide the textarea description\n $textareaDescription.classList.add('govuk-visually-hidden')\n\n // Remove hard limit if set\n this.$textarea.removeAttribute('maxlength')\n\n this.bindChangeEvents()\n\n // When the page is restored after navigating 'back' in some browsers the\n // state of form controls is not restored until *after* the DOMContentLoaded\n // event is fired, so we need to sync after the pageshow event.\n window.addEventListener('pageshow', () => this.updateCountMessage())\n\n // Although we've set up handlers to sync state on the pageshow event, init\n // could be called after those events have fired, for example if they are\n // added to the page dynamically, so update now too.\n this.updateCountMessage()\n }\n\n /**\n * Bind change events\n *\n * Set up event listeners on the $textarea so that the count messages update\n * when the user types.\n *\n * @private\n */\n bindChangeEvents() {\n this.$textarea.addEventListener('keyup', () => this.handleKeyUp())\n\n // Bind focus/blur events to start/stop polling\n this.$textarea.addEventListener('focus', () => this.handleFocus())\n this.$textarea.addEventListener('blur', () => this.handleBlur())\n }\n\n /**\n * Handle key up event\n *\n * Update the visible character counter and keep track of when the last update\n * happened for each keypress\n *\n * @private\n */\n handleKeyUp() {\n this.updateVisibleCountMessage()\n this.lastInputTimestamp = Date.now()\n }\n\n /**\n * Handle focus event\n *\n * Speech recognition software such as Dragon NaturallySpeaking will modify\n * the fields by directly changing its `value`. These changes don't trigger\n * events in JavaScript, so we need to poll to handle when and if they occur.\n *\n * Once the keyup event hasn't been detected for at least 1000 ms (1s), check\n * if the textarea value has changed and update the count message if it has.\n *\n * This is so that the update triggered by the manual comparison doesn't\n * conflict with debounced KeyboardEvent updates.\n *\n * @private\n */\n handleFocus() {\n this.valueChecker = window.setInterval(() => {\n if (\n !this.lastInputTimestamp ||\n Date.now() - 500 >= this.lastInputTimestamp\n ) {\n this.updateIfValueChanged()\n }\n }, 1000)\n }\n\n /**\n * Handle blur event\n *\n * Stop checking the textarea value once the textarea no longer has focus\n *\n * @private\n */\n handleBlur() {\n // Cancel value checking on blur\n if (this.valueChecker) {\n window.clearInterval(this.valueChecker)\n }\n }\n\n /**\n * Update count message if textarea value has changed\n *\n * @private\n */\n updateIfValueChanged() {\n if (this.$textarea.value !== this.lastInputValue) {\n this.lastInputValue = this.$textarea.value\n this.updateCountMessage()\n }\n }\n\n /**\n * Update count message\n *\n * Helper function to update both the visible and screen reader-specific\n * counters simultaneously (e.g. on init)\n *\n * @private\n */\n updateCountMessage() {\n this.updateVisibleCountMessage()\n this.updateScreenReaderCountMessage()\n }\n\n /**\n * Update visible count message\n *\n * @private\n */\n updateVisibleCountMessage() {\n const remainingNumber = this.maxLength - this.count(this.$textarea.value)\n const isError = remainingNumber < 0\n\n // If input is over the threshold, remove the disabled class which renders\n // the counter invisible.\n this.$visibleCountMessage.classList.toggle(\n 'govuk-character-count__message--disabled',\n !this.isOverThreshold()\n )\n\n // Update styles\n this.$textarea.classList.toggle('govuk-textarea--error', isError)\n this.$visibleCountMessage.classList.toggle('govuk-error-message', isError)\n this.$visibleCountMessage.classList.toggle('govuk-hint', !isError)\n\n // Update message\n this.$visibleCountMessage.textContent = this.getCountMessage()\n }\n\n /**\n * Update screen reader count message\n *\n * @private\n */\n updateScreenReaderCountMessage() {\n // If over the threshold, remove the aria-hidden attribute, allowing screen\n // readers to announce the content of the element.\n if (this.isOverThreshold()) {\n this.$screenReaderCountMessage.removeAttribute('aria-hidden')\n } else {\n this.$screenReaderCountMessage.setAttribute('aria-hidden', 'true')\n }\n\n // Update message\n this.$screenReaderCountMessage.textContent = this.getCountMessage()\n }\n\n /**\n * Count the number of characters (or words, if `config.maxwords` is set)\n * in the given text\n *\n * @private\n * @param {string} text - The text to count the characters of\n * @returns {number} the number of characters (or words) in the text\n */\n count(text) {\n if (this.config.maxwords) {\n const tokens = text.match(/\\S+/g) ?? [] // Matches consecutive non-whitespace chars\n return tokens.length\n }\n\n return text.length\n }\n\n /**\n * Get count message\n *\n * @private\n * @returns {string} Status message\n */\n getCountMessage() {\n const remainingNumber = this.maxLength - this.count(this.$textarea.value)\n const countType = this.config.maxwords ? 'words' : 'characters'\n return this.formatCountMessage(remainingNumber, countType)\n }\n\n /**\n * Formats the message shown to users according to what's counted\n * and how many remain\n *\n * @private\n * @param {number} remainingNumber - The number of words/characaters remaining\n * @param {string} countType - \"words\" or \"characters\"\n * @returns {string} Status message\n */\n formatCountMessage(remainingNumber, countType) {\n if (remainingNumber === 0) {\n return this.i18n.t(`${countType}AtLimit`)\n }\n\n const translationKeySuffix =\n remainingNumber < 0 ? 'OverLimit' : 'UnderLimit'\n\n return this.i18n.t(`${countType}${translationKeySuffix}`, {\n count: Math.abs(remainingNumber)\n })\n }\n\n /**\n * Check if count is over threshold\n *\n * Checks whether the value is over the configured threshold for the input.\n * If there is no configured threshold, it is set to 0 and this function will\n * always return true.\n *\n * @private\n * @returns {boolean} true if the current count is over the config.threshold\n * (or no threshold is set)\n */\n isOverThreshold() {\n // No threshold means we're always above threshold so save some computation\n if (!this.config.threshold) {\n return true\n }\n\n // Determine the remaining number of characters/words\n const currentLength = this.count(this.$textarea.value)\n const maxLength = this.maxLength\n\n const thresholdValue = (maxLength * this.config.threshold) / 100\n\n return thresholdValue <= currentLength\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-character-count'\n\n /**\n * Character count default config\n *\n * @see {@link CharacterCountConfig}\n * @constant\n * @type {CharacterCountConfig}\n */\n static defaults = Object.freeze({\n threshold: 0,\n i18n: {\n // Characters\n charactersUnderLimit: {\n one: 'You have %{count} character remaining',\n other: 'You have %{count} characters remaining'\n },\n charactersAtLimit: 'You have 0 characters remaining',\n charactersOverLimit: {\n one: 'You have %{count} character too many',\n other: 'You have %{count} characters too many'\n },\n // Words\n wordsUnderLimit: {\n one: 'You have %{count} word remaining',\n other: 'You have %{count} words remaining'\n },\n wordsAtLimit: 'You have 0 words remaining',\n wordsOverLimit: {\n one: 'You have %{count} word too many',\n other: 'You have %{count} words too many'\n },\n textareaDescription: {\n other: ''\n }\n }\n })\n\n /**\n * Character count config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' },\n maxwords: { type: 'number' },\n maxlength: { type: 'number' },\n threshold: { type: 'number' }\n },\n anyOf: [\n {\n required: ['maxwords'],\n errorMessage: 'Either \"maxlength\" or \"maxwords\" must be provided'\n },\n {\n required: ['maxlength'],\n errorMessage: 'Either \"maxlength\" or \"maxwords\" must be provided'\n }\n ]\n })\n}\n\n/**\n * Character count config\n *\n * @see {@link CharacterCount.defaults}\n * @typedef {object} CharacterCountConfig\n * @property {number} [maxlength] - The maximum number of characters.\n * If maxwords is provided, the maxlength option will be ignored.\n * @property {number} [maxwords] - The maximum number of words. If maxwords is\n * provided, the maxlength option will be ignored.\n * @property {number} [threshold=0] - The percentage value of the limit at\n * which point the count message is displayed. If this attribute is set, the\n * count message will be hidden by default.\n * @property {CharacterCountTranslations} [i18n=CharacterCount.defaults.i18n] - Character count translations\n */\n\n/**\n * Character count translations\n *\n * @see {@link CharacterCount.defaults.i18n}\n * @typedef {object} CharacterCountTranslations\n *\n * Messages shown to users as they type. It provides feedback on how many words\n * or characters they have remaining or if they are over the limit. This also\n * includes a message used as an accessible description for the textarea.\n * @property {TranslationPluralForms} [charactersUnderLimit] - Message displayed\n * when the number of characters is under the configured maximum, `maxlength`.\n * This message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining characters. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {string} [charactersAtLimit] - Message displayed when the number of\n * characters reaches the configured maximum, `maxlength`. This message is\n * displayed visually and through assistive technologies.\n * @property {TranslationPluralForms} [charactersOverLimit] - Message displayed\n * when the number of characters is over the configured maximum, `maxlength`.\n * This message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining characters. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {TranslationPluralForms} [wordsUnderLimit] - Message displayed when\n * the number of words is under the configured maximum, `maxlength`. This\n * message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining words. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {string} [wordsAtLimit] - Message displayed when the number of\n * words reaches the configured maximum, `maxlength`. This message is\n * displayed visually and through assistive technologies.\n * @property {TranslationPluralForms} [wordsOverLimit] - Message displayed when\n * the number of words is over the configured maximum, `maxlength`. This\n * message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining words. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {TranslationPluralForms} [textareaDescription] - Message made\n * available to assistive technologies, if none is already present in the\n * HTML, to describe that the component accepts only a limited amount of\n * content. It is visible on the page when JavaScript is unavailable. The\n * component will replace the `%{count}` placeholder with the value of the\n * `maxlength` or `maxwords` parameter.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n * @typedef {import('../../i18n.mjs').TranslationPluralForms} TranslationPluralForms\n */\n","import { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Checkboxes component\n *\n * @preserve\n */\nexport class Checkboxes extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /** @private */\n $inputs\n\n /**\n * Checkboxes can be associated with a 'conditionally revealed' content block\n * – for example, a checkbox for 'Phone' could reveal an additional form field\n * for the user to enter their phone number.\n *\n * These associations are made using a `data-aria-controls` attribute, which\n * is promoted to an aria-controls attribute during initialisation.\n *\n * We also need to restore the state of any conditional reveals on the page\n * (for example if the user has navigated back), and set up event handlers to\n * keep the reveal in sync with the checkbox state.\n *\n * @param {Element | null} $module - HTML element to use for checkboxes\n */\n constructor($module) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Checkboxes',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n const $inputs = $module.querySelectorAll('input[type=\"checkbox\"]')\n if (!$inputs.length) {\n throw new ElementError({\n componentName: 'Checkboxes',\n identifier: 'Form inputs (`<input type=\"checkbox\">`)'\n })\n }\n\n this.$module = $module\n this.$inputs = $inputs\n\n this.$inputs.forEach(($input) => {\n const targetId = $input.getAttribute('data-aria-controls')\n\n // Skip radios without data-aria-controls attributes\n if (!targetId) {\n return\n }\n\n // Throw if target conditional element does not exist.\n if (!document.getElementById(targetId)) {\n throw new ElementError({\n componentName: 'Checkboxes',\n identifier: `Conditional reveal (\\`id=\"${targetId}\"\\`)`\n })\n }\n\n // Promote the data-aria-controls attribute to a aria-controls attribute\n // so that the relationship is exposed in the AOM\n $input.setAttribute('aria-controls', targetId)\n $input.removeAttribute('data-aria-controls')\n })\n\n // When the page is restored after navigating 'back' in some browsers the\n // state of form controls is not restored until *after* the DOMContentLoaded\n // event is fired, so we need to sync after the pageshow event.\n window.addEventListener('pageshow', () => this.syncAllConditionalReveals())\n\n // Although we've set up handlers to sync state on the pageshow event, init\n // could be called after those events have fired, for example if they are\n // added to the page dynamically, so sync now too.\n this.syncAllConditionalReveals()\n\n // Handle events\n this.$module.addEventListener('click', (event) => this.handleClick(event))\n }\n\n /**\n * Sync the conditional reveal states for all checkboxes in this $module.\n *\n * @private\n */\n syncAllConditionalReveals() {\n this.$inputs.forEach(($input) =>\n this.syncConditionalRevealWithInputState($input)\n )\n }\n\n /**\n * Sync conditional reveal with the input state\n *\n * Synchronise the visibility of the conditional reveal, and its accessible\n * state, with the input's checked state.\n *\n * @private\n * @param {HTMLInputElement} $input - Checkbox input\n */\n syncConditionalRevealWithInputState($input) {\n const targetId = $input.getAttribute('aria-controls')\n if (!targetId) {\n return\n }\n\n const $target = document.getElementById(targetId)\n if ($target?.classList.contains('govuk-checkboxes__conditional')) {\n const inputIsChecked = $input.checked\n\n $input.setAttribute('aria-expanded', inputIsChecked.toString())\n $target.classList.toggle(\n 'govuk-checkboxes__conditional--hidden',\n !inputIsChecked\n )\n }\n }\n\n /**\n * Uncheck other checkboxes\n *\n * Find any other checkbox inputs with the same name value, and uncheck them.\n * This is useful for when a “None of these\" checkbox is checked.\n *\n * @private\n * @param {HTMLInputElement} $input - Checkbox input\n */\n unCheckAllInputsExcept($input) {\n const allInputsWithSameName = document.querySelectorAll(\n `input[type=\"checkbox\"][name=\"${$input.name}\"]`\n )\n\n allInputsWithSameName.forEach(($inputWithSameName) => {\n const hasSameFormOwner = $input.form === $inputWithSameName.form\n if (hasSameFormOwner && $inputWithSameName !== $input) {\n $inputWithSameName.checked = false\n this.syncConditionalRevealWithInputState($inputWithSameName)\n }\n })\n }\n\n /**\n * Uncheck exclusive checkboxes\n *\n * Find any checkbox inputs with the same name value and the 'exclusive'\n * behaviour, and uncheck them. This helps prevent someone checking both a\n * regular checkbox and a \"None of these\" checkbox in the same fieldset.\n *\n * @private\n * @param {HTMLInputElement} $input - Checkbox input\n */\n unCheckExclusiveInputs($input) {\n const allInputsWithSameNameAndExclusiveBehaviour =\n document.querySelectorAll(\n `input[data-behaviour=\"exclusive\"][type=\"checkbox\"][name=\"${$input.name}\"]`\n )\n\n allInputsWithSameNameAndExclusiveBehaviour.forEach(($exclusiveInput) => {\n const hasSameFormOwner = $input.form === $exclusiveInput.form\n if (hasSameFormOwner) {\n $exclusiveInput.checked = false\n this.syncConditionalRevealWithInputState($exclusiveInput)\n }\n })\n }\n\n /**\n * Click event handler\n *\n * Handle a click within the $module – if the click occurred on a checkbox,\n * sync the state of any associated conditional reveal with the checkbox\n * state.\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n handleClick(event) {\n const $clickedInput = event.target\n\n // Ignore clicks on things that aren't checkbox inputs\n if (\n !($clickedInput instanceof HTMLInputElement) ||\n $clickedInput.type !== 'checkbox'\n ) {\n return\n }\n\n // If the checkbox conditionally-reveals some content, sync the state\n const hasAriaControls = $clickedInput.getAttribute('aria-controls')\n if (hasAriaControls) {\n this.syncConditionalRevealWithInputState($clickedInput)\n }\n\n // No further behaviour needed for unchecking\n if (!$clickedInput.checked) {\n return\n }\n\n // Handle 'exclusive' checkbox behaviour (ie \"None of these\")\n const hasBehaviourExclusive =\n $clickedInput.getAttribute('data-behaviour') === 'exclusive'\n if (hasBehaviourExclusive) {\n this.unCheckAllInputsExcept($clickedInput)\n } else {\n this.unCheckExclusiveInputs($clickedInput)\n }\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-checkboxes'\n}\n","import {\n getFragmentFromUrl,\n mergeConfigs,\n setFocus\n} from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Error summary component\n *\n * Takes focus on initialisation for accessible announcement, unless disabled in\n * configuration.\n *\n * @preserve\n */\nexport class ErrorSummary extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /**\n * @private\n * @type {ErrorSummaryConfig}\n */\n config\n\n /**\n * @param {Element | null} $module - HTML element to use for error summary\n * @param {ErrorSummaryConfig} [config] - Error summary config\n */\n constructor($module, config = {}) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Error summary',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n this.$module = $module\n\n this.config = mergeConfigs(\n ErrorSummary.defaults,\n config,\n normaliseDataset(ErrorSummary, $module.dataset)\n )\n\n /**\n * Focus the error summary\n */\n if (!this.config.disableAutoFocus) {\n setFocus(this.$module)\n }\n\n this.$module.addEventListener('click', (event) => this.handleClick(event))\n }\n\n /**\n * Click event handler\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n handleClick(event) {\n const $target = event.target\n if ($target && this.focusTarget($target)) {\n event.preventDefault()\n }\n }\n\n /**\n * Focus the target element\n *\n * By default, the browser will scroll the target into view. Because our\n * labels or legends appear above the input, this means the user will be\n * presented with an input without any context, as the label or legend will be\n * off the top of the screen.\n *\n * Manually handling the click event, scrolling the question into view and\n * then focussing the element solves this.\n *\n * This also results in the label and/or legend being announced correctly in\n * NVDA (as tested in 2018.3.2) - without this only the field type is\n * announced (e.g. \"Edit, has autocomplete\").\n *\n * @private\n * @param {EventTarget} $target - Event target\n * @returns {boolean} True if the target was able to be focussed\n */\n focusTarget($target) {\n // If the element that was clicked was not a link, return early\n if (!($target instanceof HTMLAnchorElement)) {\n return false\n }\n\n const inputId = getFragmentFromUrl($target.href)\n if (!inputId) {\n return false\n }\n\n const $input = document.getElementById(inputId)\n if (!$input) {\n return false\n }\n\n const $legendOrLabel = this.getAssociatedLegendOrLabel($input)\n if (!$legendOrLabel) {\n return false\n }\n\n // Scroll the legend or label into view *before* calling focus on the input\n // to avoid extra scrolling in browsers that don't support `preventScroll`\n // (which at time of writing is most of them...)\n $legendOrLabel.scrollIntoView()\n $input.focus({ preventScroll: true })\n\n return true\n }\n\n /**\n * Get associated legend or label\n *\n * Returns the first element that exists from this list:\n *\n * - The `<legend>` associated with the closest `<fieldset>` ancestor, as long\n * as the top of it is no more than half a viewport height away from the\n * bottom of the input\n * - The first `<label>` that is associated with the input using for=\"inputId\"\n * - The closest parent `<label>`\n *\n * @private\n * @param {Element} $input - The input\n * @returns {Element | null} Associated legend or label, or null if no\n * associated legend or label can be found\n */\n getAssociatedLegendOrLabel($input) {\n const $fieldset = $input.closest('fieldset')\n\n if ($fieldset) {\n const $legends = $fieldset.getElementsByTagName('legend')\n\n if ($legends.length) {\n const $candidateLegend = $legends[0]\n\n // If the input type is radio or checkbox, always use the legend if\n // there is one.\n if (\n $input instanceof HTMLInputElement &&\n ($input.type === 'checkbox' || $input.type === 'radio')\n ) {\n return $candidateLegend\n }\n\n // For other input types, only scroll to the fieldset’s legend (instead\n // of the label associated with the input) if the input would end up in\n // the top half of the screen.\n //\n // This should avoid situations where the input either ends up off the\n // screen, or obscured by a software keyboard.\n const legendTop = $candidateLegend.getBoundingClientRect().top\n const inputRect = $input.getBoundingClientRect()\n\n // If the browser doesn't support Element.getBoundingClientRect().height\n // or window.innerHeight (like IE8), bail and just link to the label.\n if (inputRect.height && window.innerHeight) {\n const inputBottom = inputRect.top + inputRect.height\n\n if (inputBottom - legendTop < window.innerHeight / 2) {\n return $candidateLegend\n }\n }\n }\n }\n\n return (\n document.querySelector(`label[for='${$input.getAttribute('id')}']`) ??\n $input.closest('label')\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-error-summary'\n\n /**\n * Error summary default config\n *\n * @see {@link ErrorSummaryConfig}\n * @constant\n * @type {ErrorSummaryConfig}\n */\n static defaults = Object.freeze({\n disableAutoFocus: false\n })\n\n /**\n * Error summary config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n disableAutoFocus: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Error summary config\n *\n * @typedef {object} ErrorSummaryConfig\n * @property {boolean} [disableAutoFocus=false] - If set to `true` the error\n * summary will not be focussed when the page loads.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Exit this page component\n *\n * @preserve\n */\nexport class ExitThisPage extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /**\n * @private\n * @type {ExitThisPageConfig}\n */\n config\n\n /** @private */\n i18n\n\n /** @private */\n $button\n\n /**\n * @private\n * @type {HTMLAnchorElement | null}\n */\n $skiplinkButton = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $updateSpan = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $indicatorContainer = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $overlay = null\n\n /** @private */\n keypressCounter = 0\n\n /** @private */\n lastKeyWasModified = false\n\n /** @private */\n timeoutTime = 5000 // milliseconds\n\n // Store the timeout events so that we can clear them to avoid user keypresses overlapping\n // setTimeout returns an id that we can use to clear it with clearTimeout,\n // hence the 'Id' suffix\n\n /**\n * @private\n * @type {number | null}\n */\n keypressTimeoutId = null\n\n /**\n * @private\n * @type {number | null}\n */\n timeoutMessageId = null\n\n /**\n * @param {Element | null} $module - HTML element that wraps the Exit This Page button\n * @param {ExitThisPageConfig} [config] - Exit This Page config\n */\n constructor($module, config = {}) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Exit this page',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n const $button = $module.querySelector('.govuk-exit-this-page__button')\n if (!($button instanceof HTMLAnchorElement)) {\n throw new ElementError({\n componentName: 'Exit this page',\n element: $button,\n expectedType: 'HTMLAnchorElement',\n identifier: 'Button (`.govuk-exit-this-page__button`)'\n })\n }\n\n this.config = mergeConfigs(\n ExitThisPage.defaults,\n config,\n normaliseDataset(ExitThisPage, $module.dataset)\n )\n\n this.i18n = new I18n(this.config.i18n)\n this.$module = $module\n this.$button = $button\n\n const $skiplinkButton = document.querySelector(\n '.govuk-js-exit-this-page-skiplink'\n )\n if ($skiplinkButton instanceof HTMLAnchorElement) {\n this.$skiplinkButton = $skiplinkButton\n }\n\n this.buildIndicator()\n this.initUpdateSpan()\n this.initButtonClickHandler()\n\n // Check to see if this has already been done by a previous initialisation of ExitThisPage\n if (!('govukFrontendExitThisPageKeypress' in document.body.dataset)) {\n document.addEventListener('keyup', this.handleKeypress.bind(this), true)\n document.body.dataset.govukFrontendExitThisPageKeypress = 'true'\n }\n\n // When the page is restored after navigating 'back' in some browsers the\n // blank overlay remains present, rendering the page unusable. Here, we check\n // to see if it's present on page (re)load, and remove it if so.\n window.addEventListener('pageshow', this.resetPage.bind(this))\n }\n\n /**\n * Create the <span> we use for screen reader announcements.\n *\n * @private\n */\n initUpdateSpan() {\n this.$updateSpan = document.createElement('span')\n this.$updateSpan.setAttribute('role', 'status')\n this.$updateSpan.className = 'govuk-visually-hidden'\n\n this.$module.appendChild(this.$updateSpan)\n }\n\n /**\n * Create button click handlers.\n *\n * @private\n */\n initButtonClickHandler() {\n // Main EtP button\n this.$button.addEventListener('click', this.handleClick.bind(this))\n\n // EtP secondary link\n if (this.$skiplinkButton) {\n this.$skiplinkButton.addEventListener(\n 'click',\n this.handleClick.bind(this)\n )\n }\n }\n\n /**\n * Create the HTML for the 'three lights' indicator on the button.\n *\n * @private\n */\n buildIndicator() {\n // Build container\n // Putting `aria-hidden` on it as it won't contain any readable information\n this.$indicatorContainer = document.createElement('div')\n this.$indicatorContainer.className = 'govuk-exit-this-page__indicator'\n this.$indicatorContainer.setAttribute('aria-hidden', 'true')\n\n // Create three 'lights' and place them within the container\n for (let i = 0; i < 3; i++) {\n const $indicator = document.createElement('div')\n $indicator.className = 'govuk-exit-this-page__indicator-light'\n this.$indicatorContainer.appendChild($indicator)\n }\n\n // Append it all to the module\n this.$button.appendChild(this.$indicatorContainer)\n }\n\n /**\n * Update whether the lights are visible and which ones are lit up depending on\n * the value of `keypressCounter`.\n *\n * @private\n */\n updateIndicator() {\n if (!this.$indicatorContainer) {\n return\n }\n\n // Show or hide the indicator container depending on keypressCounter value\n this.$indicatorContainer.classList.toggle(\n 'govuk-exit-this-page__indicator--visible',\n this.keypressCounter > 0\n )\n\n // Turn on only the indicators we want on\n const $indicators = this.$indicatorContainer.querySelectorAll(\n '.govuk-exit-this-page__indicator-light'\n )\n $indicators.forEach(($indicator, index) => {\n $indicator.classList.toggle(\n 'govuk-exit-this-page__indicator-light--on',\n index < this.keypressCounter\n )\n })\n }\n\n /**\n * Initiates the redirection away from the current page.\n * Includes the loading overlay functionality, which covers the current page with a\n * white overlay so that the contents are not visible during the loading\n * process. This is particularly important on slow network connections.\n *\n * @private\n */\n exitPage() {\n if (!this.$updateSpan) {\n return\n }\n\n this.$updateSpan.textContent = ''\n\n // Blank the page\n // As well as creating an overlay with text, we also set the body to hidden\n // to prevent screen reader and sequential navigation users potentially\n // navigating through the page behind the overlay during loading\n document.body.classList.add('govuk-exit-this-page-hide-content')\n this.$overlay = document.createElement('div')\n this.$overlay.className = 'govuk-exit-this-page-overlay'\n this.$overlay.setAttribute('role', 'alert')\n\n // we do these this way round, thus incurring a second paint, because changing\n // the element text after adding it means that screen readers pick up the\n // announcement more reliably.\n document.body.appendChild(this.$overlay)\n this.$overlay.textContent = this.i18n.t('activated')\n\n window.location.href = this.$button.href\n }\n\n /**\n * Pre-activation logic for when the button is clicked/activated via mouse or\n * pointer.\n *\n * We do this to differentiate it from the keyboard activation event because we\n * need to run `e.preventDefault` as the button or skiplink are both links and we\n * want to apply some additional logic in `exitPage` before navigating.\n *\n * @private\n * @param {MouseEvent} event - mouse click event\n */\n handleClick(event) {\n event.preventDefault()\n this.exitPage()\n }\n\n /**\n * Logic for the 'quick escape' keyboard sequence functionality (pressing the\n * Shift key three times without interruption, within a time limit).\n *\n * @private\n * @param {KeyboardEvent} event - keyup event\n */\n handleKeypress(event) {\n if (!this.$updateSpan) {\n return\n }\n\n // Detect if the 'Shift' key has been pressed. We want to only do things if it\n // was pressed by itself and not in a combination with another key—so we keep\n // track of whether the preceding keyup had shiftKey: true on it, and if it\n // did, we ignore the next Shift keyup event.\n //\n // This works because using Shift as a modifier key (e.g. pressing Shift + A)\n // will fire TWO keyup events, one for A (with e.shiftKey: true) and the other\n // for Shift (with e.shiftKey: false).\n if (event.key === 'Shift' && !this.lastKeyWasModified) {\n this.keypressCounter += 1\n\n // Update the indicator before the below if statement can reset it back to 0\n this.updateIndicator()\n\n // Clear the timeout for the keypress timeout message clearing itself\n if (this.timeoutMessageId) {\n window.clearTimeout(this.timeoutMessageId)\n this.timeoutMessageId = null\n }\n\n if (this.keypressCounter >= 3) {\n this.keypressCounter = 0\n\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n this.keypressTimeoutId = null\n }\n\n this.exitPage()\n } else {\n if (this.keypressCounter === 1) {\n this.$updateSpan.textContent = this.i18n.t('pressTwoMoreTimes')\n } else {\n this.$updateSpan.textContent = this.i18n.t('pressOneMoreTime')\n }\n }\n\n this.setKeypressTimer()\n } else if (this.keypressTimeoutId) {\n // If the user pressed any key other than 'Shift', after having pressed\n // 'Shift' and activating the timer, stop and reset the timer.\n this.resetKeypressTimer()\n }\n\n // Keep track of whether the Shift modifier key was held during this keypress\n this.lastKeyWasModified = event.shiftKey\n }\n\n /**\n * Starts the 'quick escape' keyboard sequence timer.\n *\n * This can be invoked several times. We want this to be possible so that the\n * timer is restarted each time the shortcut key is pressed (e.g. the user has\n * up to n seconds between each keypress, rather than n seconds to invoke the\n * entire sequence.)\n *\n * @private\n */\n setKeypressTimer() {\n // Clear any existing timeout. This is so only one timer is running even if\n // there are multiple keypresses in quick succession.\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n }\n\n // Set a fresh timeout\n this.keypressTimeoutId = window.setTimeout(\n this.resetKeypressTimer.bind(this),\n this.timeoutTime\n )\n }\n\n /**\n * Stops and resets the 'quick escape' keyboard sequence timer.\n *\n * @private\n */\n resetKeypressTimer() {\n if (!this.$updateSpan) {\n return\n }\n\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n this.keypressTimeoutId = null\n }\n\n const $updateSpan = this.$updateSpan\n\n this.keypressCounter = 0\n $updateSpan.textContent = this.i18n.t('timedOut')\n\n this.timeoutMessageId = window.setTimeout(() => {\n $updateSpan.textContent = ''\n }, this.timeoutTime)\n\n this.updateIndicator()\n }\n\n /**\n * Reset the page using the EtP button\n *\n * We use this in situations where a user may re-enter a page using the browser\n * back button. In these cases, the browser can choose to restore the state of\n * the page as it was previously, including restoring the 'ghost page' overlay,\n * the announcement span having it's role set to \"alert\" and the keypress\n * indicator still active, leaving the page in an unusable state.\n *\n * By running this check when the page is shown, we can programatically restore\n * the page and the component to a \"default\" state\n *\n * @private\n */\n resetPage() {\n // If an overlay is set, remove it and reset the value\n document.body.classList.remove('govuk-exit-this-page-hide-content')\n\n if (this.$overlay) {\n this.$overlay.remove()\n this.$overlay = null\n }\n\n // Ensure the announcement span's role is status, not alert and clear any text\n if (this.$updateSpan) {\n this.$updateSpan.setAttribute('role', 'status')\n this.$updateSpan.textContent = ''\n }\n\n // Sync the keypress indicator lights\n this.updateIndicator()\n\n // If the timeouts are active, clear them\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n }\n\n if (this.timeoutMessageId) {\n window.clearTimeout(this.timeoutMessageId)\n }\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-exit-this-page'\n\n /**\n * Exit this page default config\n *\n * @see {@link ExitThisPageConfig}\n * @constant\n * @type {ExitThisPageConfig}\n */\n static defaults = Object.freeze({\n i18n: {\n activated: 'Loading.',\n timedOut: 'Exit this page expired.',\n pressTwoMoreTimes: 'Shift, press 2 more times to exit.',\n pressOneMoreTime: 'Shift, press 1 more time to exit.'\n }\n })\n\n /**\n * Exit this page config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' }\n }\n })\n}\n\n/**\n * Exit this Page config\n *\n * @see {@link ExitThisPage.defaults}\n * @typedef {object} ExitThisPageConfig\n * @property {ExitThisPageTranslations} [i18n=ExitThisPage.defaults.i18n] - Exit this page translations\n */\n\n/**\n * Exit this Page translations\n *\n * @see {@link ExitThisPage.defaults.i18n}\n * @typedef {object} ExitThisPageTranslations\n *\n * Messages used by the component programatically inserted text, including\n * overlay text and screen reader announcements.\n * @property {string} [activated] - Screen reader announcement for when EtP\n * keypress functionality has been successfully activated.\n * @property {string} [timedOut] - Screen reader announcement for when the EtP\n * keypress functionality has timed out.\n * @property {string} [pressTwoMoreTimes] - Screen reader announcement informing\n * the user they must press the activation key two more times.\n * @property {string} [pressOneMoreTime] - Screen reader announcement informing\n * the user they must press the activation key one more time.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { getBreakpoint } from '../../common/index.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Header component\n *\n * @preserve\n */\nexport class Header extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /** @private */\n $menuButton\n\n /** @private */\n $menu\n\n /**\n * Save the opened/closed state for the nav in memory so that we can\n * accurately maintain state when the screen is changed from small to big and\n * back to small\n *\n * @private\n */\n menuIsOpen = false\n\n /**\n * A global const for storing a matchMedia instance which we'll use to detect\n * when a screen size change happens. We rely on it being null if the feature\n * isn't available to initially apply hidden attributes\n *\n * @private\n * @type {MediaQueryList | null}\n */\n mql = null\n\n /**\n * Apply a matchMedia for desktop which will trigger a state sync if the\n * browser viewport moves between states.\n *\n * @param {Element | null} $module - HTML element to use for header\n */\n constructor($module) {\n super()\n\n if (!$module) {\n throw new ElementError({\n componentName: 'Header',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n this.$module = $module\n const $menuButton = $module.querySelector('.govuk-js-header-toggle')\n\n // Headers don't necessarily have a navigation. When they don't, the menu\n // toggle won't be rendered by our macro (or may be omitted when writing\n // plain HTML)\n if (!$menuButton) {\n return this\n }\n\n const menuId = $menuButton.getAttribute('aria-controls')\n if (!menuId) {\n throw new ElementError({\n componentName: 'Header',\n identifier:\n 'Navigation button (`<button class=\"govuk-js-header-toggle\">`) attribute (`aria-controls`)'\n })\n }\n\n const $menu = document.getElementById(menuId)\n if (!$menu) {\n throw new ElementError({\n componentName: 'Header',\n element: $menu,\n identifier: `Navigation (\\`<ul id=\"${menuId}\">\\`)`\n })\n }\n\n this.$menu = $menu\n this.$menuButton = $menuButton\n\n this.setupResponsiveChecks()\n\n this.$menuButton.addEventListener('click', () =>\n this.handleMenuButtonClick()\n )\n }\n\n /**\n * Setup viewport resize check\n *\n * @private\n */\n setupResponsiveChecks() {\n const breakpoint = getBreakpoint('desktop')\n\n if (!breakpoint.value) {\n throw new ElementError({\n componentName: 'Header',\n identifier: `CSS custom property (\\`${breakpoint.property}\\`) on pseudo-class \\`:root\\``\n })\n }\n\n // Media query list for GOV.UK Frontend desktop breakpoint\n this.mql = window.matchMedia(`(min-width: ${breakpoint.value})`)\n\n // MediaQueryList.addEventListener isn't supported by Safari < 14 so we need\n // to be able to fall back to the deprecated MediaQueryList.addListener\n if ('addEventListener' in this.mql) {\n this.mql.addEventListener('change', () => this.checkMode())\n } else {\n // @ts-expect-error Property 'addListener' does not exist\n // eslint-disable-next-line @typescript-eslint/no-unsafe-call\n this.mql.addListener(() => this.checkMode())\n }\n\n this.checkMode()\n }\n\n /**\n * Sync menu state\n *\n * Uses the global variable menuIsOpen to correctly set the accessible and\n * visual states of the menu and the menu button.\n * Additionally will force the menu to be visible and the menu button to be\n * hidden if the matchMedia is triggered to desktop.\n *\n * @private\n */\n checkMode() {\n if (!this.mql || !this.$menu || !this.$menuButton) {\n return\n }\n\n if (this.mql.matches) {\n this.$menu.removeAttribute('hidden')\n this.$menuButton.setAttribute('hidden', '')\n } else {\n this.$menuButton.removeAttribute('hidden')\n this.$menuButton.setAttribute('aria-expanded', this.menuIsOpen.toString())\n\n if (this.menuIsOpen) {\n this.$menu.removeAttribute('hidden')\n } else {\n this.$menu.setAttribute('hidden', '')\n }\n }\n }\n\n /**\n * Handle menu button click\n *\n * When the menu button is clicked, change the visibility of the menu and then\n * sync the accessibility state and menu button state\n *\n * @private\n */\n handleMenuButtonClick() {\n this.menuIsOpen = !this.menuIsOpen\n this.checkMode()\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-header'\n}\n","import { mergeConfigs, setFocus } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Notification Banner component\n *\n * @preserve\n */\nexport class NotificationBanner extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /**\n * @private\n * @type {NotificationBannerConfig}\n */\n config\n\n /**\n * @param {Element | null} $module - HTML element to use for notification banner\n * @param {NotificationBannerConfig} [config] - Notification banner config\n */\n constructor($module, config = {}) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Notification banner',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n this.$module = $module\n\n this.config = mergeConfigs(\n NotificationBanner.defaults,\n config,\n normaliseDataset(NotificationBanner, $module.dataset)\n )\n\n /**\n * Focus the notification banner\n *\n * If `role=\"alert\"` is set, focus the element to help some assistive\n * technologies prioritise announcing it.\n *\n * You can turn off the auto-focus functionality by setting\n * `data-disable-auto-focus=\"true\"` in the component HTML. You might wish to\n * do this based on user research findings, or to avoid a clash with another\n * element which should be focused when the page loads.\n */\n if (\n this.$module.getAttribute('role') === 'alert' &&\n !this.config.disableAutoFocus\n ) {\n setFocus(this.$module)\n }\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-notification-banner'\n\n /**\n * Notification banner default config\n *\n * @see {@link NotificationBannerConfig}\n * @constant\n * @type {NotificationBannerConfig}\n */\n static defaults = Object.freeze({\n disableAutoFocus: false\n })\n\n /**\n * Notification banner config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n disableAutoFocus: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Notification banner config\n *\n * @typedef {object} NotificationBannerConfig\n * @property {boolean} [disableAutoFocus=false] - If set to `true` the\n * notification banner will not be focussed when the page loads. This only\n * applies if the component has a `role` of `alert` – in other cases the\n * component will not be focused on page load, regardless of this option.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { closestAttributeValue } from '../../common/closest-attribute-value.mjs'\nimport { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Password input component\n *\n * @preserve\n */\nexport class PasswordInput extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /**\n * @private\n * @type {PasswordInputConfig}\n */\n config\n\n /** @private */\n i18n\n\n /**\n * @private\n * @type {HTMLInputElement}\n */\n $input\n\n /**\n * @private\n * @type {HTMLButtonElement}\n */\n $showHideButton\n\n /** @private */\n $screenReaderStatusMessage\n\n /**\n * @param {Element | null} $module - HTML element to use for password input\n * @param {PasswordInputConfig} [config] - Password input config\n */\n constructor($module, config = {}) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Password input',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n const $input = $module.querySelector('.govuk-js-password-input-input')\n if (!($input instanceof HTMLInputElement)) {\n throw new ElementError({\n componentName: 'Password input',\n element: $input,\n expectedType: 'HTMLInputElement',\n identifier: 'Form field (`.govuk-js-password-input-input`)'\n })\n }\n\n if ($input.type !== 'password') {\n throw new ElementError(\n 'Password input: Form field (`.govuk-js-password-input-input`) must be of type `password`.'\n )\n }\n\n const $showHideButton = $module.querySelector(\n '.govuk-js-password-input-toggle'\n )\n if (!($showHideButton instanceof HTMLButtonElement)) {\n throw new ElementError({\n componentName: 'Password input',\n element: $showHideButton,\n expectedType: 'HTMLButtonElement',\n identifier: 'Button (`.govuk-js-password-input-toggle`)'\n })\n }\n\n if ($showHideButton.type !== 'button') {\n throw new ElementError(\n 'Password input: Button (`.govuk-js-password-input-toggle`) must be of type `button`.'\n )\n }\n\n this.$module = $module\n this.$input = $input\n this.$showHideButton = $showHideButton\n\n this.config = mergeConfigs(\n PasswordInput.defaults,\n config,\n normaliseDataset(PasswordInput, $module.dataset)\n )\n\n this.i18n = new I18n(this.config.i18n, {\n // Read the fallback if necessary rather than have it set in the defaults\n locale: closestAttributeValue($module, 'lang')\n })\n\n // Show the toggle button element\n this.$showHideButton.removeAttribute('hidden')\n\n // Create and append the status text for screen readers.\n // This is injected between the input and button so that users get a sensible reading order if\n // moving through the page content linearly:\n // [password input] -> [your password is visible/hidden] -> [show/hide password]\n const $screenReaderStatusMessage = document.createElement('div')\n $screenReaderStatusMessage.className =\n 'govuk-password-input__sr-status govuk-visually-hidden'\n $screenReaderStatusMessage.setAttribute('aria-live', 'polite')\n this.$screenReaderStatusMessage = $screenReaderStatusMessage\n this.$input.insertAdjacentElement('afterend', $screenReaderStatusMessage)\n\n // Bind toggle button\n this.$showHideButton.addEventListener('click', this.toggle.bind(this))\n\n // Bind event to revert the password visibility to hidden\n if (this.$input.form) {\n this.$input.form.addEventListener('submit', () => this.hide())\n }\n\n // If the page is restored from bfcache and the password is visible, hide it again\n window.addEventListener('pageshow', (event) => {\n if (event.persisted && this.$input.type !== 'password') {\n this.hide()\n }\n })\n\n // Default the component to having the password hidden.\n this.hide()\n }\n\n /**\n * Toggle the visibility of the password input\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n toggle(event) {\n event.preventDefault()\n\n // If on this click, the field is type=\"password\", show the value\n if (this.$input.type === 'password') {\n this.show()\n return\n }\n\n // Otherwise, hide it\n // Being defensive - hiding should always be the default\n this.hide()\n }\n\n /**\n * Show the password input value in plain text.\n *\n * @private\n */\n show() {\n this.setType('text')\n }\n\n /**\n * Hide the password input value.\n *\n * @private\n */\n hide() {\n this.setType('password')\n }\n\n /**\n * Set the password input type\n *\n * @param {'text' | 'password'} type - Input type\n * @private\n */\n setType(type) {\n if (type === this.$input.type) {\n return\n }\n\n // Update input type\n this.$input.setAttribute('type', type)\n\n const isHidden = type === 'password'\n const prefixButton = isHidden ? 'show' : 'hide'\n const prefixStatus = isHidden ? 'passwordHidden' : 'passwordShown'\n\n // Update button text\n this.$showHideButton.innerText = this.i18n.t(`${prefixButton}Password`)\n\n // Update button aria-label\n this.$showHideButton.setAttribute(\n 'aria-label',\n this.i18n.t(`${prefixButton}PasswordAriaLabel`)\n )\n\n // Update status change text\n this.$screenReaderStatusMessage.innerText = this.i18n.t(\n `${prefixStatus}Announcement`\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-password-input'\n\n /**\n * Password input default config\n *\n * @see {@link PasswordInputConfig}\n * @constant\n * @default\n * @type {PasswordInputConfig}\n */\n static defaults = Object.freeze({\n i18n: {\n showPassword: 'Show',\n hidePassword: 'Hide',\n showPasswordAriaLabel: 'Show password',\n hidePasswordAriaLabel: 'Hide password',\n passwordShownAnnouncement: 'Your password is visible',\n passwordHiddenAnnouncement: 'Your password is hidden'\n }\n })\n\n /**\n * Password input config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' }\n }\n })\n}\n\n/**\n * Password input config\n *\n * @typedef {object} PasswordInputConfig\n * @property {PasswordInputTranslations} [i18n=PasswordInput.defaults.i18n] - Password input translations\n */\n\n/**\n * Password input translations\n *\n * @see {@link PasswordInput.defaults.i18n}\n * @typedef {object} PasswordInputTranslations\n *\n * Messages displayed to the user indicating the state of the show/hide toggle.\n * @property {string} [showPassword] - Visible text of the button when the\n * password is currently hidden. Plain text only.\n * @property {string} [hidePassword] - Visible text of the button when the\n * password is currently visible. Plain text only.\n * @property {string} [showPasswordAriaLabel] - aria-label of the button when\n * the password is currently hidden. Plain text only.\n * @property {string} [hidePasswordAriaLabel] - aria-label of the button when\n * the password is currently visible. Plain text only.\n * @property {string} [passwordShownAnnouncement] - Screen reader\n * announcement to make when the password has just become visible.\n * Plain text only.\n * @property {string} [passwordHiddenAnnouncement] - Screen reader\n * announcement to make when the password has just been hidden.\n * Plain text only.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n * @typedef {import('../../i18n.mjs').TranslationPluralForms} TranslationPluralForms\n */\n","import { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Radios component\n *\n * @preserve\n */\nexport class Radios extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /** @private */\n $inputs\n\n /**\n * Radios can be associated with a 'conditionally revealed' content block –\n * for example, a radio for 'Phone' could reveal an additional form field for\n * the user to enter their phone number.\n *\n * These associations are made using a `data-aria-controls` attribute, which\n * is promoted to an aria-controls attribute during initialisation.\n *\n * We also need to restore the state of any conditional reveals on the page\n * (for example if the user has navigated back), and set up event handlers to\n * keep the reveal in sync with the radio state.\n *\n * @param {Element | null} $module - HTML element to use for radios\n */\n constructor($module) {\n super()\n\n if (!($module instanceof HTMLElement)) {\n throw new ElementError({\n componentName: 'Radios',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n const $inputs = $module.querySelectorAll('input[type=\"radio\"]')\n if (!$inputs.length) {\n throw new ElementError({\n componentName: 'Radios',\n identifier: 'Form inputs (`<input type=\"radio\">`)'\n })\n }\n\n this.$module = $module\n this.$inputs = $inputs\n\n this.$inputs.forEach(($input) => {\n const targetId = $input.getAttribute('data-aria-controls')\n\n // Skip radios without data-aria-controls attributes\n if (!targetId) {\n return\n }\n\n // Throw if target conditional element does not exist.\n if (!document.getElementById(targetId)) {\n throw new ElementError({\n componentName: 'Radios',\n identifier: `Conditional reveal (\\`id=\"${targetId}\"\\`)`\n })\n }\n\n // Promote the data-aria-controls attribute to a aria-controls attribute\n // so that the relationship is exposed in the AOM\n $input.setAttribute('aria-controls', targetId)\n $input.removeAttribute('data-aria-controls')\n })\n\n // When the page is restored after navigating 'back' in some browsers the\n // state of form controls is not restored until *after* the DOMContentLoaded\n // event is fired, so we need to sync after the pageshow event.\n window.addEventListener('pageshow', () => this.syncAllConditionalReveals())\n\n // Although we've set up handlers to sync state on the pageshow event, init\n // could be called after those events have fired, for example if they are\n // added to the page dynamically, so sync now too.\n this.syncAllConditionalReveals()\n\n // Handle events\n this.$module.addEventListener('click', (event) => this.handleClick(event))\n }\n\n /**\n * Sync the conditional reveal states for all radio buttons in this $module.\n *\n * @private\n */\n syncAllConditionalReveals() {\n this.$inputs.forEach(($input) =>\n this.syncConditionalRevealWithInputState($input)\n )\n }\n\n /**\n * Sync conditional reveal with the input state\n *\n * Synchronise the visibility of the conditional reveal, and its accessible\n * state, with the input's checked state.\n *\n * @private\n * @param {HTMLInputElement} $input - Radio input\n */\n syncConditionalRevealWithInputState($input) {\n const targetId = $input.getAttribute('aria-controls')\n if (!targetId) {\n return\n }\n\n const $target = document.getElementById(targetId)\n if ($target?.classList.contains('govuk-radios__conditional')) {\n const inputIsChecked = $input.checked\n\n $input.setAttribute('aria-expanded', inputIsChecked.toString())\n $target.classList.toggle(\n 'govuk-radios__conditional--hidden',\n !inputIsChecked\n )\n }\n }\n\n /**\n * Click event handler\n *\n * Handle a click within the $module – if the click occurred on a radio, sync\n * the state of the conditional reveal for all radio buttons in the same form\n * with the same name (because checking one radio could have un-checked a\n * radio in another $module)\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n handleClick(event) {\n const $clickedInput = event.target\n\n // Ignore clicks on things that aren't radio buttons\n if (\n !($clickedInput instanceof HTMLInputElement) ||\n $clickedInput.type !== 'radio'\n ) {\n return\n }\n\n // We only need to consider radios with conditional reveals, which will have\n // aria-controls attributes.\n const $allInputs = document.querySelectorAll(\n 'input[type=\"radio\"][aria-controls]'\n )\n\n const $clickedInputForm = $clickedInput.form\n const $clickedInputName = $clickedInput.name\n\n $allInputs.forEach(($input) => {\n const hasSameFormOwner = $input.form === $clickedInputForm\n const hasSameName = $input.name === $clickedInputName\n\n if (hasSameName && hasSameFormOwner) {\n this.syncConditionalRevealWithInputState($input)\n }\n })\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-radios'\n}\n","import { getFragmentFromUrl, setFocus } from '../../common/index.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Skip link component\n *\n * @preserve\n */\nexport class SkipLink extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /**\n * @param {Element | null} $module - HTML element to use for skip link\n * @throws {ElementError} when $module is not set or the wrong type\n * @throws {ElementError} when $module.hash does not contain a hash\n * @throws {ElementError} when the linked element is missing or the wrong type\n */\n constructor($module) {\n super()\n\n if (!($module instanceof HTMLAnchorElement)) {\n throw new ElementError({\n componentName: 'Skip link',\n element: $module,\n expectedType: 'HTMLAnchorElement',\n identifier: 'Root element (`$module`)'\n })\n }\n\n this.$module = $module\n\n const hash = this.$module.hash\n const href = this.$module.getAttribute('href') ?? ''\n\n /** @type {URL | undefined} */\n let url\n\n /**\n * Check for valid link URL\n *\n * {@link https://caniuse.com/url}\n * {@link https://url.spec.whatwg.org}\n *\n */\n try {\n url = new window.URL(this.$module.href)\n } catch (error) {\n throw new ElementError(\n `Skip link: Target link (\\`href=\"${href}\"\\`) is invalid`\n )\n }\n\n // Return early for external URLs or links to other pages\n if (\n url.origin !== window.location.origin ||\n url.pathname !== window.location.pathname\n ) {\n return\n }\n\n const linkedElementId = getFragmentFromUrl(hash)\n\n // Check link path matching current page\n if (!linkedElementId) {\n throw new ElementError(\n `Skip link: Target link (\\`href=\"${href}\"\\`) has no hash fragment`\n )\n }\n\n const $linkedElement = document.getElementById(linkedElementId)\n\n // Check for link target element\n if (!$linkedElement) {\n throw new ElementError({\n componentName: 'Skip link',\n element: $linkedElement,\n identifier: `Target content (\\`id=\"${linkedElementId}\"\\`)`\n })\n }\n\n /**\n * Focus the linked element on click\n *\n * Adds a helper CSS class to hide native focus styles,\n * but removes it on blur to restore native focus styles\n */\n this.$module.addEventListener('click', () =>\n setFocus($linkedElement, {\n onBeforeFocus() {\n $linkedElement.classList.add('govuk-skip-link-focused-element')\n },\n onBlur() {\n $linkedElement.classList.remove('govuk-skip-link-focused-element')\n }\n })\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-skip-link'\n}\n","import { getBreakpoint, getFragmentFromUrl } from '../../common/index.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Tabs component\n *\n * @preserve\n */\nexport class Tabs extends GOVUKFrontendComponent {\n /** @private */\n $module\n\n /** @private */\n $tabs\n\n /** @private */\n $tabList\n\n /** @private */\n $tabListItems\n\n /** @private */\n jsHiddenClass = 'govuk-tabs__panel--hidden'\n\n /** @private */\n changingHash = false\n\n /** @private */\n boundTabClick\n\n /** @private */\n boundTabKeydown\n\n /** @private */\n boundOnHashChange\n\n /**\n * @private\n * @type {MediaQueryList | null}\n */\n mql = null\n\n /**\n * @param {Element | null} $module - HTML element to use for tabs\n */\n constructor($module) {\n super()\n\n if (!$module) {\n throw new ElementError({\n componentName: 'Tabs',\n element: $module,\n identifier: 'Root element (`$module`)'\n })\n }\n\n const $tabs = $module.querySelectorAll('a.govuk-tabs__tab')\n if (!$tabs.length) {\n throw new ElementError({\n componentName: 'Tabs',\n identifier: 'Links (`<a class=\"govuk-tabs__tab\">`)'\n })\n }\n\n this.$module = $module\n this.$tabs = $tabs\n\n // Save bound functions so we can remove event listeners during teardown\n this.boundTabClick = this.onTabClick.bind(this)\n this.boundTabKeydown = this.onTabKeydown.bind(this)\n this.boundOnHashChange = this.onHashChange.bind(this)\n\n const $tabList = this.$module.querySelector('.govuk-tabs__list')\n const $tabListItems = this.$module.querySelectorAll(\n 'li.govuk-tabs__list-item'\n )\n\n if (!$tabList) {\n throw new ElementError({\n componentName: 'Tabs',\n identifier: 'List (`<ul class=\"govuk-tabs__list\">`)'\n })\n }\n\n if (!$tabListItems.length) {\n throw new ElementError({\n componentName: 'Tabs',\n identifier: 'List items (`<li class=\"govuk-tabs__list-item\">`)'\n })\n }\n\n this.$tabList = $tabList\n this.$tabListItems = $tabListItems\n\n this.setupResponsiveChecks()\n }\n\n /**\n * Setup viewport resize check\n *\n * @private\n */\n setupResponsiveChecks() {\n const breakpoint = getBreakpoint('tablet')\n\n if (!breakpoint.value) {\n throw new ElementError({\n componentName: 'Tabs',\n identifier: `CSS custom property (\\`${breakpoint.property}\\`) on pseudo-class \\`:root\\``\n })\n }\n\n // Media query list for GOV.UK Frontend tablet breakpoint\n this.mql = window.matchMedia(`(min-width: ${breakpoint.value})`)\n\n // MediaQueryList.addEventListener isn't supported by Safari < 14 so we need\n // to be able to fall back to the deprecated MediaQueryList.addListener\n if ('addEventListener' in this.mql) {\n this.mql.addEventListener('change', () => this.checkMode())\n } else {\n // @ts-expect-error Property 'addListener' does not exist\n // eslint-disable-next-line @typescript-eslint/no-unsafe-call\n this.mql.addListener(() => this.checkMode())\n }\n\n this.checkMode()\n }\n\n /**\n * Setup or teardown handler for viewport resize check\n *\n * @private\n */\n checkMode() {\n if (this.mql?.matches) {\n this.setup()\n } else {\n this.teardown()\n }\n }\n\n /**\n * Setup tab component\n *\n * @private\n */\n setup() {\n this.$tabList.setAttribute('role', 'tablist')\n\n this.$tabListItems.forEach(($item) => {\n $item.setAttribute('role', 'presentation')\n })\n\n this.$tabs.forEach(($tab) => {\n // Set HTML attributes\n this.setAttributes($tab)\n\n // Handle events\n $tab.addEventListener('click', this.boundTabClick, true)\n $tab.addEventListener('keydown', this.boundTabKeydown, true)\n\n // Remove old active panels\n this.hideTab($tab)\n })\n\n // Show either the active tab according to the URL's hash or the first tab\n const $activeTab = this.getTab(window.location.hash) ?? this.$tabs[0]\n\n this.showTab($activeTab)\n\n // Handle hashchange events\n window.addEventListener('hashchange', this.boundOnHashChange, true)\n }\n\n /**\n * Teardown tab component\n *\n * @private\n */\n teardown() {\n this.$tabList.removeAttribute('role')\n\n this.$tabListItems.forEach(($item) => {\n $item.removeAttribute('role')\n })\n\n this.$tabs.forEach(($tab) => {\n // Remove events\n $tab.removeEventListener('click', this.boundTabClick, true)\n $tab.removeEventListener('keydown', this.boundTabKeydown, true)\n\n // Unset HTML attributes\n this.unsetAttributes($tab)\n })\n\n // Remove hashchange event handler\n window.removeEventListener('hashchange', this.boundOnHashChange, true)\n }\n\n /**\n * Handle hashchange event\n *\n * @private\n * @returns {void | undefined} Returns void, or undefined when prevented\n */\n onHashChange() {\n const hash = window.location.hash\n const $tabWithHash = this.getTab(hash)\n if (!$tabWithHash) {\n return\n }\n\n // Prevent changing the hash\n if (this.changingHash) {\n this.changingHash = false\n return\n }\n\n // Show either the active tab according to the URL's hash or the first tab\n const $previousTab = this.getCurrentTab()\n if (!$previousTab) {\n return\n }\n\n this.hideTab($previousTab)\n this.showTab($tabWithHash)\n $tabWithHash.focus()\n }\n\n /**\n * Hide panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n hideTab($tab) {\n this.unhighlightTab($tab)\n this.hidePanel($tab)\n }\n\n /**\n * Show panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n showTab($tab) {\n this.highlightTab($tab)\n this.showPanel($tab)\n }\n\n /**\n * Get tab link by hash\n *\n * @private\n * @param {string} hash - Hash fragment including #\n * @returns {HTMLAnchorElement | null} Tab link\n */\n getTab(hash) {\n return this.$module.querySelector(`a.govuk-tabs__tab[href=\"${hash}\"]`)\n }\n\n /**\n * Set tab link and panel attributes\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n setAttributes($tab) {\n const panelId = getFragmentFromUrl($tab.href)\n if (!panelId) {\n return\n }\n\n // Set tab attributes\n $tab.setAttribute('id', `tab_${panelId}`)\n $tab.setAttribute('role', 'tab')\n $tab.setAttribute('aria-controls', panelId)\n $tab.setAttribute('aria-selected', 'false')\n $tab.setAttribute('tabindex', '-1')\n\n // Set panel attributes\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.setAttribute('role', 'tabpanel')\n $panel.setAttribute('aria-labelledby', $tab.id)\n $panel.classList.add(this.jsHiddenClass)\n }\n\n /**\n * Unset tab link and panel attributes\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n unsetAttributes($tab) {\n // unset tab attributes\n $tab.removeAttribute('id')\n $tab.removeAttribute('role')\n $tab.removeAttribute('aria-controls')\n $tab.removeAttribute('aria-selected')\n $tab.removeAttribute('tabindex')\n\n // unset panel attributes\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.removeAttribute('role')\n $panel.removeAttribute('aria-labelledby')\n $panel.classList.remove(this.jsHiddenClass)\n }\n\n /**\n * Handle tab link clicks\n *\n * @private\n * @param {MouseEvent} event - Mouse click event\n * @returns {void} Returns void\n */\n onTabClick(event) {\n const $currentTab = this.getCurrentTab()\n const $nextTab = event.currentTarget\n\n if (!$currentTab || !($nextTab instanceof HTMLAnchorElement)) {\n return\n }\n\n event.preventDefault()\n\n this.hideTab($currentTab)\n this.showTab($nextTab)\n this.createHistoryEntry($nextTab)\n }\n\n /**\n * Update browser URL hash fragment for tab\n *\n * - Allows back/forward to navigate tabs\n * - Avoids page jump when hash changes\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n createHistoryEntry($tab) {\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n // Save and restore the id so the page doesn't jump when a user clicks a tab\n // (which changes the hash)\n const panelId = $panel.id\n $panel.id = ''\n this.changingHash = true\n window.location.hash = panelId\n $panel.id = panelId\n }\n\n /**\n * Handle tab keydown event\n *\n * - Press right arrow for next tab\n * - Press left arrow for previous tab\n *\n * @private\n * @param {KeyboardEvent} event - Keydown event\n */\n onTabKeydown(event) {\n switch (event.key) {\n // 'Left' and 'Right' required for Edge 16 support.\n case 'ArrowLeft':\n case 'Left':\n this.activatePreviousTab()\n event.preventDefault()\n break\n case 'ArrowRight':\n case 'Right':\n this.activateNextTab()\n event.preventDefault()\n break\n }\n }\n\n /**\n * Activate next tab\n *\n * @private\n */\n activateNextTab() {\n const $currentTab = this.getCurrentTab()\n if (!$currentTab?.parentElement) {\n return\n }\n\n const $nextTabListItem = $currentTab.parentElement.nextElementSibling\n if (!$nextTabListItem) {\n return\n }\n\n const $nextTab = $nextTabListItem.querySelector('a.govuk-tabs__tab')\n if (!$nextTab) {\n return\n }\n\n this.hideTab($currentTab)\n this.showTab($nextTab)\n $nextTab.focus()\n this.createHistoryEntry($nextTab)\n }\n\n /**\n * Activate previous tab\n *\n * @private\n */\n activatePreviousTab() {\n const $currentTab = this.getCurrentTab()\n if (!$currentTab?.parentElement) {\n return\n }\n\n const $previousTabListItem =\n $currentTab.parentElement.previousElementSibling\n if (!$previousTabListItem) {\n return\n }\n\n const $previousTab = $previousTabListItem.querySelector('a.govuk-tabs__tab')\n if (!$previousTab) {\n return\n }\n\n this.hideTab($currentTab)\n this.showTab($previousTab)\n $previousTab.focus()\n this.createHistoryEntry($previousTab)\n }\n\n /**\n * Get tab panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n * @returns {Element | null} Tab panel\n */\n getPanel($tab) {\n const panelId = getFragmentFromUrl($tab.href)\n if (!panelId) {\n return null\n }\n\n return this.$module.querySelector(`#${panelId}`)\n }\n\n /**\n * Show tab panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n showPanel($tab) {\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.classList.remove(this.jsHiddenClass)\n }\n\n /**\n * Hide tab panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n hidePanel($tab) {\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.classList.add(this.jsHiddenClass)\n }\n\n /**\n * Unset 'selected' state for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n unhighlightTab($tab) {\n if (!$tab.parentElement) {\n return\n }\n\n $tab.setAttribute('aria-selected', 'false')\n $tab.parentElement.classList.remove('govuk-tabs__list-item--selected')\n $tab.setAttribute('tabindex', '-1')\n }\n\n /**\n * Set 'selected' state for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n highlightTab($tab) {\n if (!$tab.parentElement) {\n return\n }\n\n $tab.setAttribute('aria-selected', 'true')\n $tab.parentElement.classList.add('govuk-tabs__list-item--selected')\n $tab.setAttribute('tabindex', '0')\n }\n\n /**\n * Get current tab link\n *\n * @private\n * @returns {HTMLAnchorElement | null} Tab link\n */\n getCurrentTab() {\n return this.$module.querySelector(\n '.govuk-tabs__list-item--selected a.govuk-tabs__tab'\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-tabs'\n}\n","import { isSupported } from './common/index.mjs'\nimport { Accordion } from './components/accordion/accordion.mjs'\nimport { Button } from './components/button/button.mjs'\nimport { CharacterCount } from './components/character-count/character-count.mjs'\nimport { Checkboxes } from './components/checkboxes/checkboxes.mjs'\nimport { ErrorSummary } from './components/error-summary/error-summary.mjs'\nimport { ExitThisPage } from './components/exit-this-page/exit-this-page.mjs'\nimport { Header } from './components/header/header.mjs'\nimport { NotificationBanner } from './components/notification-banner/notification-banner.mjs'\nimport { PasswordInput } from './components/password-input/password-input.mjs'\nimport { Radios } from './components/radios/radios.mjs'\nimport { SkipLink } from './components/skip-link/skip-link.mjs'\nimport { Tabs } from './components/tabs/tabs.mjs'\nimport { SupportError } from './errors/index.mjs'\n\n/**\n * Initialise all components\n *\n * Use the `data-module` attributes to find, instantiate and init all of the\n * components provided as part of GOV.UK Frontend.\n *\n * @param {Config & { scope?: Element }} [config] - Config for all components (with optional scope)\n */\nfunction initAll(config) {\n config = typeof config !== 'undefined' ? config : {}\n\n // Skip initialisation when GOV.UK Frontend is not supported\n if (!isSupported()) {\n console.log(new SupportError())\n return\n }\n\n const components = /** @type {const} */ ([\n [Accordion, config.accordion],\n [Button, config.button],\n [CharacterCount, config.characterCount],\n [Checkboxes],\n [ErrorSummary, config.errorSummary],\n [ExitThisPage, config.exitThisPage],\n [Header],\n [NotificationBanner, config.notificationBanner],\n [PasswordInput, config.passwordInput],\n [Radios],\n [SkipLink],\n [Tabs]\n ])\n\n // Allow the user to initialise GOV.UK Frontend in only certain sections of the page\n // Defaults to the entire document if nothing is set.\n const $scope = config.scope ?? document\n\n components.forEach(([Component, config]) => {\n createAll(Component, config, $scope)\n })\n}\n\n/**\n * Create all instances of a specific component on the page\n *\n * Uses the `data-module` attribute to find all elements matching the specified\n * component on the page, creating instances of the component object for each\n * of them.\n *\n * Any component errors will be caught and logged to the console.\n *\n * @template {CompatibleClass} T\n * @param {T} Component - class of the component to create\n * @param {T[\"defaults\"]} [config] - config for the component\n * @param {Element|Document} [$scope] - scope of the document to search within\n * @returns {Array<InstanceType<T>>} - array of instantiated components\n */\nfunction createAll(Component, config, $scope = document) {\n const $elements = $scope.querySelectorAll(\n `[data-module=\"${Component.moduleName}\"]`\n )\n\n /* eslint-disable-next-line @typescript-eslint/no-unsafe-return --\n * We can't define CompatibleClass as `{new(): CompatibleClass, moduleName: string}`,\n * as when doing `typeof Accordion` (or any component), TypeScript doesn't seem\n * to acknowledge the static `moduleName` that's set in our component classes.\n * This means we have to set the constructor of `CompatibleClass` as `{new(): any}`,\n * leading to ESLint frowning that we're returning `any[]`.\n */\n return Array.from($elements)\n .map(($element) => {\n try {\n // Only pass config to components that accept it\n // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n return 'defaults' in Component && typeof config !== 'undefined'\n ? new Component($element, config)\n : new Component($element)\n } catch (error) {\n console.log(error)\n return null\n }\n })\n .filter(Boolean) // Exclude components that errored\n}\n\nexport { initAll, createAll }\n\n/* eslint-disable jsdoc/valid-types --\n * `{new(...args: any[] ): object}` is not recognised as valid\n * https://github.com/gajus/eslint-plugin-jsdoc/issues/145#issuecomment-1308722878\n * https://github.com/jsdoc-type-pratt-parser/jsdoc-type-pratt-parser/issues/131\n **/\n\n/**\n * @typedef {{new (...args: any[]): any, defaults?: object, moduleName: string}} CompatibleClass\n */\n\n/* eslint-enable jsdoc/valid-types */\n\n/**\n * Config for all components via `initAll()`\n *\n * @typedef {object} Config\n * @property {AccordionConfig} [accordion] - Accordion config\n * @property {ButtonConfig} [button] - Button config\n * @property {CharacterCountConfig} [characterCount] - Character Count config\n * @property {ErrorSummaryConfig} [errorSummary] - Error Summary config\n * @property {ExitThisPageConfig} [exitThisPage] - Exit This Page config\n * @property {NotificationBannerConfig} [notificationBanner] - Notification Banner config\n * @property {PasswordInputConfig} [passwordInput] - Password input config\n */\n\n/**\n * Config for individual components\n *\n * @typedef {import('./components/accordion/accordion.mjs').AccordionConfig} AccordionConfig\n * @typedef {import('./components/accordion/accordion.mjs').AccordionTranslations} AccordionTranslations\n * @typedef {import('./components/button/button.mjs').ButtonConfig} ButtonConfig\n * @typedef {import('./components/character-count/character-count.mjs').CharacterCountConfig} CharacterCountConfig\n * @typedef {import('./components/character-count/character-count.mjs').CharacterCountTranslations} CharacterCountTranslations\n * @typedef {import('./components/error-summary/error-summary.mjs').ErrorSummaryConfig} ErrorSummaryConfig\n * @typedef {import('./components/exit-this-page/exit-this-page.mjs').ExitThisPageConfig} ExitThisPageConfig\n * @typedef {import('./components/exit-this-page/exit-this-page.mjs').ExitThisPageTranslations} ExitThisPageTranslations\n * @typedef {import('./components/notification-banner/notification-banner.mjs').NotificationBannerConfig} NotificationBannerConfig\n * @typedef {import('./components/password-input/password-input.mjs').PasswordInputConfig} PasswordInputConfig\n */\n\n/**\n * Component config keys, e.g. `accordion` and `characterCount`\n *\n * @typedef {keyof Config} ConfigKey\n */\n"],"names":["version","normaliseString","value","property","trimmedValue","trim","output","outputType","type","includes","length","isFinite","Number","mergeConfigs","configObjects","formattedConfigObject","configObject","key","Object","keys","option","override","isObject","extractConfigByNamespace","Component","dataset","namespace","schema","properties","newObject","entries","current","keyParts","split","index","name","getFragmentFromUrl","url","pop","getBreakpoint","window","getComputedStyle","document","documentElement","getPropertyValue","undefined","setFocus","$element","options","_options$onBeforeFocu","isFocusable","getAttribute","onBlur","_options$onBlur","call","removeAttribute","setAttribute","addEventListener","once","onBeforeFocus","focus","isSupported","$scope","body","classList","contains","Array","isArray","normaliseDataset","out","field","GOVUKFrontendError","Error","constructor","args","super","this","SupportError","supportMessage","HTMLScriptElement","prototype","ConfigError","ElementError","messageOrOptions","message","componentName","identifier","element","expectedType","GOVUKFrontendComponent","checkSupport","I18n","translations","config","_config$locale","locale","lang","t","lookupKey","translation","count","translationPluralForm","getPluralSuffix","match","replacePlaceholders","translationString","formatter","Intl","NumberFormat","supportedLocalesOf","replace","placeholderWithBraces","placeholderKey","hasOwnProperty","placeholderValue","format","hasIntlPluralRulesSupport","Boolean","PluralRules","preferredForm","select","selectPluralFormUsingFallbackRules","console","warn","Math","abs","floor","ruleset","getPluralRulesForLocale","pluralRules","localeShort","pluralRule","pluralRulesMap","languages","arabic","chinese","french","german","irish","russian","scottish","spanish","welsh","n","lastTwo","last","Accordion","$module","i18n","controlsClass","showAllClass","showAllTextClass","sectionClass","sectionExpandedClass","sectionButtonClass","sectionHeaderClass","sectionHeadingClass","sectionHeadingDividerClass","sectionHeadingTextClass","sectionHeadingTextFocusClass","sectionShowHideToggleClass","sectionShowHideToggleFocusClass","sectionShowHideTextClass","upChevronIconClass","downChevronIconClass","sectionSummaryClass","sectionSummaryFocusClass","sectionContentClass","$sections","$showAllButton","$showAllIcon","$showAllText","HTMLElement","defaults","querySelectorAll","initControls","initSectionHeaders","updateShowAllButton","areAllSectionsOpen","createElement","add","appendChild","$accordionControls","insertBefore","firstChild","onShowOrHideAllToggle","event","onBeforeMatch","forEach","$section","i","$header","querySelector","constructHeaderMarkup","setExpanded","isExpanded","onSectionToggle","setInitialState","$span","$heading","$summary","$button","id","attr","from","attributes","$headingText","$headingTextFocus","childNodes","$child","$showHideToggle","$showHideToggleFocus","$showHideText","$showHideIcon","getButtonPunctuationEl","$summarySpan","$summarySpanFocus","remove","removeChild","$fragment","target","Element","closest","nowExpanded","storeState","expanded","$content","newButtonText","textContent","ariaLabelParts","push","ariaLabelMessage","join","every","toString","toggle","getIdentifier","rememberExpanded","sessionStorage","setItem","exception","state","getItem","$punctuationEl","moduleName","freeze","hideAllSections","hideSection","hideSectionAriaLabel","showAllSections","showSection","showSectionAriaLabel","Button","debounceFormSubmitTimer","handleKeyDown","debounce","$target","preventDefault","click","preventDoubleClick","setTimeout","DEBOUNCE_TIMEOUT_IN_SECONDS","closestAttributeValue","attributeName","$closestElementWithAttribute","CharacterCount","_ref","_this$config$maxwords","$textarea","$visibleCountMessage","$screenReaderCountMessage","lastInputTimestamp","lastInputValue","valueChecker","maxLength","HTMLTextAreaElement","HTMLInputElement","datasetConfig","configOverrides","maxlength","maxwords","errors","validationErrors","conditions","required","errorMessage","validateConfig","Infinity","textareaDescriptionId","$textareaDescription","getElementById","insertAdjacentElement","className","bindChangeEvents","updateCountMessage","handleKeyUp","handleFocus","handleBlur","updateVisibleCountMessage","Date","now","setInterval","updateIfValueChanged","clearInterval","updateScreenReaderCountMessage","isError","isOverThreshold","getCountMessage","text","_text$match","remainingNumber","countType","formatCountMessage","translationKeySuffix","threshold","currentLength","charactersUnderLimit","one","other","charactersAtLimit","charactersOverLimit","wordsUnderLimit","wordsAtLimit","wordsOverLimit","textareaDescription","anyOf","Checkboxes","$inputs","$input","targetId","syncAllConditionalReveals","handleClick","syncConditionalRevealWithInputState","inputIsChecked","checked","unCheckAllInputsExcept","$inputWithSameName","form","unCheckExclusiveInputs","$exclusiveInput","$clickedInput","ErrorSummary","disableAutoFocus","focusTarget","HTMLAnchorElement","inputId","href","$legendOrLabel","getAssociatedLegendOrLabel","scrollIntoView","preventScroll","_document$querySelect","$fieldset","$legends","getElementsByTagName","$candidateLegend","legendTop","getBoundingClientRect","top","inputRect","height","innerHeight","ExitThisPage","$skiplinkButton","$updateSpan","$indicatorContainer","$overlay","keypressCounter","lastKeyWasModified","timeoutTime","keypressTimeoutId","timeoutMessageId","buildIndicator","initUpdateSpan","initButtonClickHandler","handleKeypress","bind","govukFrontendExitThisPageKeypress","resetPage","$indicator","updateIndicator","exitPage","location","resetKeypressTimer","clearTimeout","setKeypressTimer","shiftKey","activated","timedOut","pressTwoMoreTimes","pressOneMoreTime","Header","$menuButton","$menu","menuIsOpen","mql","menuId","setupResponsiveChecks","handleMenuButtonClick","breakpoint","matchMedia","checkMode","addListener","matches","NotificationBanner","PasswordInput","$showHideButton","$screenReaderStatusMessage","HTMLButtonElement","hide","persisted","show","setType","isHidden","prefixButton","prefixStatus","innerText","showPassword","hidePassword","showPasswordAriaLabel","hidePasswordAriaLabel","passwordShownAnnouncement","passwordHiddenAnnouncement","Radios","$allInputs","$clickedInputForm","$clickedInputName","hasSameFormOwner","SkipLink","_this$$module$getAttr","hash","URL","error","origin","pathname","linkedElementId","$linkedElement","Tabs","$tabs","$tabList","$tabListItems","jsHiddenClass","changingHash","boundTabClick","boundTabKeydown","boundOnHashChange","onTabClick","onTabKeydown","onHashChange","_this$mql","setup","teardown","_this$getTab","$item","$tab","setAttributes","hideTab","$activeTab","getTab","showTab","removeEventListener","unsetAttributes","$tabWithHash","$previousTab","getCurrentTab","unhighlightTab","hidePanel","highlightTab","showPanel","panelId","$panel","getPanel","$currentTab","$nextTab","currentTarget","createHistoryEntry","activatePreviousTab","activateNextTab","parentElement","$nextTabListItem","nextElementSibling","$previousTabListItem","previousElementSibling","initAll","_config$scope","log","components","accordion","button","characterCount","errorSummary","exitThisPage","notificationBanner","passwordInput","scope","createAll","$elements","map","filter"],"mappings":"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"}
1
+ {"version":3,"file":"govuk-frontend.min.js","sources":["../../src/govuk/common/govuk-frontend-version.mjs","../../src/govuk/common/normalise-string.mjs","../../src/govuk/common/index.mjs","../../src/govuk/common/normalise-dataset.mjs","../../src/govuk/errors/index.mjs","../../src/govuk/govuk-frontend-component.mjs","../../src/govuk/i18n.mjs","../../src/govuk/components/accordion/accordion.mjs","../../src/govuk/components/button/button.mjs","../../src/govuk/common/closest-attribute-value.mjs","../../src/govuk/components/character-count/character-count.mjs","../../src/govuk/components/checkboxes/checkboxes.mjs","../../src/govuk/components/error-summary/error-summary.mjs","../../src/govuk/components/exit-this-page/exit-this-page.mjs","../../src/govuk/components/header/header.mjs","../../src/govuk/components/notification-banner/notification-banner.mjs","../../src/govuk/components/password-input/password-input.mjs","../../src/govuk/components/radios/radios.mjs","../../src/govuk/components/service-navigation/service-navigation.mjs","../../src/govuk/components/skip-link/skip-link.mjs","../../src/govuk/components/tabs/tabs.mjs","../../src/govuk/init.mjs"],"sourcesContent":["/*\n * This variable is automatically overwritten during builds and releases.\n * It doesn't need to be updated manually.\n */\n\n/**\n * GOV.UK Frontend release version\n *\n * {@link https://github.com/alphagov/govuk-frontend/releases}\n */\nexport const version = 'development'\n","/**\n * Normalise string\n *\n * 'If it looks like a duck, and it quacks like a duck…' 🦆\n *\n * If the passed value looks like a boolean or a number, convert it to a boolean\n * or number.\n *\n * Designed to be used to convert config passed via data attributes (which are\n * always strings) into something sensible.\n *\n * @internal\n * @param {DOMStringMap[string]} value - The value to normalise\n * @param {SchemaProperty} [property] - Component schema property\n * @returns {string | boolean | number | undefined} Normalised data\n */\nexport function normaliseString(value, property) {\n const trimmedValue = value ? value.trim() : ''\n\n let output\n let outputType = property?.type\n\n // No schema type set? Determine automatically\n if (!outputType) {\n if (['true', 'false'].includes(trimmedValue)) {\n outputType = 'boolean'\n }\n\n // Empty / whitespace-only strings are considered finite so we need to check\n // the length of the trimmed string as well\n if (trimmedValue.length > 0 && isFinite(Number(trimmedValue))) {\n outputType = 'number'\n }\n }\n\n switch (outputType) {\n case 'boolean':\n output = trimmedValue === 'true'\n break\n\n case 'number':\n output = Number(trimmedValue)\n break\n\n default:\n output = value\n }\n\n return output\n}\n\n/**\n * @typedef {import('./index.mjs').SchemaProperty} SchemaProperty\n */\n","import { normaliseString } from './normalise-string.mjs'\n\n/**\n * Common helpers which do not require polyfill.\n *\n * IMPORTANT: If a helper require a polyfill, please isolate it in its own module\n * so that the polyfill can be properly tree-shaken and does not burden\n * the components that do not need that helper\n */\n\n/**\n * Config merging function\n *\n * Takes any number of objects and combines them together, with\n * greatest priority on the LAST item passed in.\n *\n * @internal\n * @param {...{ [key: string]: unknown }} configObjects - Config objects to merge\n * @returns {{ [key: string]: unknown }} A merged config object\n */\nexport function mergeConfigs(...configObjects) {\n // Start with an empty object as our base\n /** @type {{ [key: string]: unknown }} */\n const formattedConfigObject = {}\n\n // Loop through each of the passed objects\n for (const configObject of configObjects) {\n for (const key of Object.keys(configObject)) {\n const option = formattedConfigObject[key]\n const override = configObject[key]\n\n // Push their keys one-by-one into formattedConfigObject. Any duplicate\n // keys with object values will be merged, otherwise the new value will\n // override the existing value.\n if (isObject(option) && isObject(override)) {\n // @ts-expect-error Index signature for type 'string' is missing\n formattedConfigObject[key] = mergeConfigs(option, override)\n } else {\n // Apply override\n formattedConfigObject[key] = override\n }\n }\n }\n\n return formattedConfigObject\n}\n\n/**\n * Extracts keys starting with a particular namespace from dataset ('data-*')\n * object, removing the namespace in the process, normalising all values\n *\n * @internal\n * @param {{ schema: Schema }} Component - Component class\n * @param {DOMStringMap} dataset - The object to extract key-value pairs from\n * @param {string} namespace - The namespace to filter keys with\n * @returns {ObjectNested | undefined} Nested object with dot-separated key namespace removed\n */\nexport function extractConfigByNamespace(Component, dataset, namespace) {\n const property = Component.schema.properties[namespace]\n\n // Only extract configs for object schema properties\n if (property?.type !== 'object') {\n return\n }\n\n // Add default empty config\n const newObject = {\n [namespace]: /** @type {ObjectNested} */ ({})\n }\n\n for (const [key, value] of Object.entries(dataset)) {\n /** @type {ObjectNested | ObjectNested[NestedKey]} */\n let current = newObject\n\n // Split the key into parts, using . as our namespace separator\n const keyParts = key.split('.')\n\n /**\n * Create new level per part\n *\n * e.g. 'i18n.textareaDescription.other' becomes\n * `{ i18n: { textareaDescription: { other } } }`\n */\n for (const [index, name] of keyParts.entries()) {\n if (typeof current === 'object') {\n // Drop down to nested object until the last part\n if (index < keyParts.length - 1) {\n // New nested object (optionally) replaces existing value\n if (!isObject(current[name])) {\n current[name] = {}\n }\n\n // Drop down into new or existing nested object\n current = current[name]\n } else if (key !== namespace) {\n // Normalised value (optionally) replaces existing value\n current[name] = normaliseString(value)\n }\n }\n }\n }\n\n return newObject[namespace]\n}\n\n/**\n * Get hash fragment from URL\n *\n * Extract the hash fragment (everything after the hash) from a URL,\n * but not including the hash symbol\n *\n * @private\n * @param {string} url - URL\n * @returns {string | undefined} Fragment from URL, without the hash\n */\nexport function getFragmentFromUrl(url) {\n if (!url.includes('#')) {\n return undefined\n }\n\n return url.split('#').pop()\n}\n\n/**\n * Get GOV.UK Frontend breakpoint value from CSS custom property\n *\n * @private\n * @param {string} name - Breakpoint name\n * @returns {{ property: string, value?: string }} Breakpoint object\n */\nexport function getBreakpoint(name) {\n const property = `--govuk-frontend-breakpoint-${name}`\n\n // Get value from `<html>` with breakpoints on CSS :root\n const value = window\n .getComputedStyle(document.documentElement)\n .getPropertyValue(property)\n\n return {\n property,\n value: value || undefined\n }\n}\n\n/**\n * Move focus to element\n *\n * Sets tabindex to -1 to make the element programmatically focusable,\n * but removes it on blur as the element doesn't need to be focused again.\n *\n * @private\n * @template {HTMLElement} FocusElement\n * @param {FocusElement} $element - HTML element\n * @param {object} [options] - Handler options\n * @param {function(this: FocusElement): void} [options.onBeforeFocus] - Callback before focus\n * @param {function(this: FocusElement): void} [options.onBlur] - Callback on blur\n */\nexport function setFocus($element, options = {}) {\n const isFocusable = $element.getAttribute('tabindex')\n\n if (!isFocusable) {\n $element.setAttribute('tabindex', '-1')\n }\n\n /**\n * Handle element focus\n */\n function onFocus() {\n $element.addEventListener('blur', onBlur, { once: true })\n }\n\n /**\n * Handle element blur\n */\n function onBlur() {\n options.onBlur?.call($element)\n\n if (!isFocusable) {\n $element.removeAttribute('tabindex')\n }\n }\n\n // Add listener to reset element on blur, after focus\n $element.addEventListener('focus', onFocus, { once: true })\n\n // Focus element\n options.onBeforeFocus?.call($element)\n $element.focus()\n}\n\n/**\n * Checks if component is already initialised\n *\n * @internal\n * @param {Element} $root - HTML element to be checked\n * @param {string} moduleName - name of component module\n * @returns {boolean} Whether component is already initialised\n */\nexport function isInitialised($root, moduleName) {\n return (\n $root instanceof HTMLElement &&\n $root.hasAttribute(`data-${moduleName}-init`)\n )\n}\n\n/**\n * Checks if GOV.UK Frontend is supported on this page\n *\n * Some browsers will load and run our JavaScript but GOV.UK Frontend\n * won't be supported.\n *\n * @param {HTMLElement | null} [$scope] - (internal) `<body>` HTML element checked for browser support\n * @returns {boolean} Whether GOV.UK Frontend is supported on this page\n */\nexport function isSupported($scope = document.body) {\n if (!$scope) {\n return false\n }\n\n return $scope.classList.contains('govuk-frontend-supported')\n}\n\n/**\n * Validate component config by schema\n *\n * Follows limited examples in JSON schema for wider support in future\n *\n * {@link https://ajv.js.org/json-schema.html#compound-keywords}\n * {@link https://ajv.js.org/packages/ajv-errors.html#single-message}\n *\n * @internal\n * @param {Schema} schema - Config schema\n * @param {{ [key: string]: unknown }} config - Component config\n * @returns {string[]} List of validation errors\n */\nexport function validateConfig(schema, config) {\n const validationErrors = []\n\n // Check errors for each schema\n for (const [name, conditions] of Object.entries(schema)) {\n const errors = []\n\n // Check errors for each schema condition\n if (Array.isArray(conditions)) {\n for (const { required, errorMessage } of conditions) {\n if (!required.every((key) => !!config[key])) {\n errors.push(errorMessage) // Missing config key value\n }\n }\n\n // Check one condition passes or add errors\n if (name === 'anyOf' && !(conditions.length - errors.length >= 1)) {\n validationErrors.push(...errors)\n }\n }\n }\n\n return validationErrors\n}\n\n/**\n * Check for an array\n *\n * @internal\n * @param {unknown} option - Option to check\n * @returns {boolean} Whether the option is an array\n */\nfunction isArray(option) {\n return Array.isArray(option)\n}\n\n/**\n * Check for an object\n *\n * @internal\n * @param {unknown} option - Option to check\n * @returns {boolean} Whether the option is an object\n */\nfunction isObject(option) {\n return !!option && typeof option === 'object' && !isArray(option)\n}\n\n/**\n * Format error message\n *\n * @internal\n * @param {ComponentWithModuleName} Component - Component that threw the error\n * @param {string} message - Error message\n * @returns {string} - Formatted error message\n */\nexport function formatErrorMessage(Component, message) {\n return `${Component.moduleName}: ${message}`\n}\n\n/**\n * Schema for component config\n *\n * @typedef {object} Schema\n * @property {{ [field: string]: SchemaProperty | undefined }} properties - Schema properties\n * @property {SchemaCondition[]} [anyOf] - List of schema conditions\n */\n\n/**\n * Schema property for component config\n *\n * @typedef {object} SchemaProperty\n * @property {'string' | 'boolean' | 'number' | 'object'} type - Property type\n */\n\n/**\n * Schema condition for component config\n *\n * @typedef {object} SchemaCondition\n * @property {string[]} required - List of required config fields\n * @property {string} errorMessage - Error message when required config fields not provided\n */\n\n/**\n * @internal\n * @typedef {keyof ObjectNested} NestedKey\n * @typedef {{ [key: string]: string | boolean | number | ObjectNested | undefined }} ObjectNested\n */\n\n/* eslint-disable jsdoc/valid-types --\n * `{new(...args: any[] ): object}` is not recognised as valid\n * https://github.com/gajus/eslint-plugin-jsdoc/issues/145#issuecomment-1308722878\n * https://github.com/jsdoc-type-pratt-parser/jsdoc-type-pratt-parser/issues/131\n **/\n\n/**\n * @typedef ComponentWithModuleName\n * @property {string} moduleName - Name of the component\n */\n\n/* eslint-enable jsdoc/valid-types */\n","import { extractConfigByNamespace } from './index.mjs'\nimport { normaliseString } from './normalise-string.mjs'\n\n/**\n * Normalise dataset\n *\n * Loop over an object and normalise each value using {@link normaliseString},\n * optionally expanding nested `i18n.field`\n *\n * @internal\n * @param {{ schema: Schema }} Component - Component class\n * @param {DOMStringMap} dataset - HTML element dataset\n * @returns {ObjectNested} Normalised dataset\n */\nexport function normaliseDataset(Component, dataset) {\n const out = /** @type {ReturnType<typeof normaliseDataset>} */ ({})\n\n // Normalise top-level dataset ('data-*') values using schema types\n for (const [field, property] of Object.entries(Component.schema.properties)) {\n if (field in dataset) {\n out[field] = normaliseString(dataset[field], property)\n }\n\n /**\n * Extract and normalise nested object values automatically using\n * {@link normaliseString} but only schema object types are allowed\n */\n if (property?.type === 'object') {\n out[field] = extractConfigByNamespace(Component, dataset, field)\n }\n }\n\n return out\n}\n\n/**\n * @internal\n * @typedef {import('./index.mjs').ObjectNested} ObjectNested\n * @typedef {import('./index.mjs').Schema} Schema\n */\n","import { formatErrorMessage } from '../common/index.mjs'\n\n/**\n * GOV.UK Frontend error\n *\n * A base class for `Error`s thrown by GOV.UK Frontend.\n *\n * It is meant to be extended into specific types of errors\n * to be thrown by our code.\n *\n * @example\n * ```js\n * class MissingRootError extends GOVUKFrontendError {\n * // Setting an explicit name is important as extending the class will not\n * // set a new `name` on the subclass. The `name` property is important\n * // to ensure intelligible error names even if the class name gets\n * // mangled by a minifier\n * name = \"MissingRootError\"\n * }\n * ```\n * @virtual\n */\nexport class GOVUKFrontendError extends Error {\n name = 'GOVUKFrontendError'\n}\n\n/**\n * Indicates that GOV.UK Frontend is not supported\n */\nexport class SupportError extends GOVUKFrontendError {\n name = 'SupportError'\n\n /**\n * Checks if GOV.UK Frontend is supported on this page\n *\n * @param {HTMLElement | null} [$scope] - HTML element `<body>` checked for browser support\n */\n constructor($scope = document.body) {\n const supportMessage =\n 'noModule' in HTMLScriptElement.prototype\n ? 'GOV.UK Frontend initialised without `<body class=\"govuk-frontend-supported\">` from template `<script>` snippet'\n : 'GOV.UK Frontend is not supported in this browser'\n\n super(\n $scope\n ? supportMessage\n : 'GOV.UK Frontend initialised without `<script type=\"module\">`'\n )\n }\n}\n\n/**\n * Indicates that a component has received an illegal configuration\n */\nexport class ConfigError extends GOVUKFrontendError {\n name = 'ConfigError'\n}\n\n/**\n * Indicates an issue with an element (possibly `null` or `undefined`)\n */\nexport class ElementError extends GOVUKFrontendError {\n name = 'ElementError'\n\n /**\n * @internal\n * @overload\n * @param {string} message - Element error message\n */\n\n /**\n * @internal\n * @overload\n * @param {ElementErrorOptions} options - Element error options\n */\n\n /**\n * @internal\n * @param {string | ElementErrorOptions} messageOrOptions - Element error message or options\n */\n constructor(messageOrOptions) {\n let message = typeof messageOrOptions === 'string' ? messageOrOptions : ''\n\n // Build message from options\n if (typeof messageOrOptions === 'object') {\n const { component, identifier, element, expectedType } = messageOrOptions\n\n message = identifier\n\n // Append reason\n message += element\n ? ` is not of type ${expectedType ?? 'HTMLElement'}`\n : ' not found'\n\n message = formatErrorMessage(component, message)\n }\n\n super(message)\n }\n}\n\n/**\n * Indicates that a component is already initialised\n */\nexport class InitError extends GOVUKFrontendError {\n name = 'InitError'\n\n /**\n * @internal\n * @param {ComponentWithModuleName | string} componentOrMessage - name of the component module\n */\n constructor(componentOrMessage) {\n const message =\n typeof componentOrMessage === 'string'\n ? componentOrMessage\n : formatErrorMessage(\n componentOrMessage,\n `Root element (\\`$root\\`) already initialised`\n )\n\n super(message)\n }\n}\n\n/**\n * Element error options\n *\n * @internal\n * @typedef {object} ElementErrorOptions\n * @property {string} identifier - An identifier that'll let the user understand which element has an error. This is whatever makes the most sense\n * @property {Element | null} [element] - The element in error\n * @property {string} [expectedType] - The type that was expected for the identifier\n * @property {ComponentWithModuleName} component - Component throwing the error\n */\n\n/**\n * @typedef {import('../common/index.mjs').ComponentWithModuleName} ComponentWithModuleName\n */\n","import { isInitialised, isSupported } from './common/index.mjs'\nimport { ElementError, InitError, SupportError } from './errors/index.mjs'\n\n/**\n * Base Component class\n *\n * Centralises the behaviours shared by our components\n *\n * @virtual\n * @template {Element} [RootElementType=HTMLElement]\n */\nexport class GOVUKFrontendComponent {\n /**\n * @type {typeof Element}\n */\n static elementType = HTMLElement\n\n // allows Typescript user to work around the lack of types\n // in GOVUKFrontend package, Typescript is not aware of $root\n // in components that extend GOVUKFrontendComponent\n /**\n * Returns the root element of the component\n *\n * @protected\n * @returns {RootElementType} - the root element of component\n */\n get $root() {\n return this._$root\n }\n\n /**\n * @protected\n * @type {RootElementType}\n */\n _$root\n\n /**\n * Constructs a new component, validating that GOV.UK Frontend is supported\n *\n * @internal\n * @param {Element | null} [$root] - HTML element to use for component\n */\n constructor($root) {\n const childConstructor = /** @type {ChildClassConstructor} */ (\n this.constructor\n )\n\n // TypeScript does not enforce that inheriting classes will define a `moduleName`\n // (even if we add a `@virtual` `static moduleName` property to this class).\n // While we trust users to do this correctly, we do a little check to provide them\n // a helpful error message.\n //\n // After this, we'll be sure that `childConstructor` has a `moduleName`\n // as expected of the `ChildClassConstructor` we've cast `this.constructor` to.\n if (typeof childConstructor.moduleName !== 'string') {\n throw new InitError(`\\`moduleName\\` not defined in component`)\n }\n\n if (!($root instanceof childConstructor.elementType)) {\n throw new ElementError({\n element: $root,\n component: childConstructor,\n identifier: 'Root element (`$root`)',\n expectedType: childConstructor.elementType.name\n })\n } else {\n this._$root = /** @type {RootElementType} */ ($root)\n }\n\n childConstructor.checkSupport()\n\n this.checkInitialised()\n\n const moduleName = childConstructor.moduleName\n\n this.$root.setAttribute(`data-${moduleName}-init`, '')\n }\n\n /**\n * Validates whether component is already initialised\n *\n * @private\n * @throws {InitError} when component is already initialised\n */\n checkInitialised() {\n const constructor = /** @type {ChildClassConstructor} */ (this.constructor)\n const moduleName = constructor.moduleName\n\n if (moduleName && isInitialised(this.$root, moduleName)) {\n throw new InitError(constructor)\n }\n }\n\n /**\n * Validates whether components are supported\n *\n * @throws {SupportError} when the components are not supported\n */\n static checkSupport() {\n if (!isSupported()) {\n throw new SupportError()\n }\n }\n}\n\n/**\n * @typedef ChildClass\n * @property {string} moduleName - The module name that'll be looked for in the DOM when initialising the component\n */\n\n/**\n * @typedef {typeof GOVUKFrontendComponent & ChildClass} ChildClassConstructor\n */\n","/**\n * Internal support for selecting messages to render, with placeholder\n * interpolation and locale-aware number formatting and pluralisation\n *\n * @internal\n */\nexport class I18n {\n translations\n locale\n\n /**\n * @internal\n * @param {{ [key: string]: string | TranslationPluralForms }} translations - Key-value pairs of the translation strings to use.\n * @param {object} [config] - Configuration options for the function.\n * @param {string | null} [config.locale] - An overriding locale for the PluralRules functionality.\n */\n constructor(translations = {}, config = {}) {\n // Make list of translations available throughout function\n this.translations = translations\n\n // The locale to use for PluralRules and NumberFormat\n this.locale = config.locale ?? (document.documentElement.lang || 'en')\n }\n\n /**\n * The most used function - takes the key for a given piece of UI text and\n * returns the appropriate string.\n *\n * @internal\n * @param {string} lookupKey - The lookup key of the string to use.\n * @param {{ [key: string]: unknown }} [options] - Any options passed with the translation string, e.g: for string interpolation.\n * @returns {string} The appropriate translation string.\n * @throws {Error} Lookup key required\n * @throws {Error} Options required for `${}` placeholders\n */\n t(lookupKey, options) {\n if (!lookupKey) {\n // Print a console error if no lookup key has been provided\n throw new Error('i18n: lookup key missing')\n }\n\n // Fetch the translation for that lookup key\n let translation = this.translations[lookupKey]\n\n // If the `count` option is set, determine which plural suffix is needed and\n // change the lookupKey to match. We check to see if it's numeric instead of\n // falsy, as this could legitimately be 0.\n if (typeof options?.count === 'number' && typeof translation === 'object') {\n const translationPluralForm =\n translation[this.getPluralSuffix(lookupKey, options.count)]\n\n // Update translation with plural suffix\n if (translationPluralForm) {\n translation = translationPluralForm\n }\n }\n\n if (typeof translation === 'string') {\n // Check for ${} placeholders in the translation string\n if (translation.match(/%{(.\\S+)}/)) {\n if (!options) {\n throw new Error(\n 'i18n: cannot replace placeholders in string if no option data provided'\n )\n }\n\n return this.replacePlaceholders(translation, options)\n }\n\n return translation\n }\n\n // If the key wasn't found in our translations object,\n // return the lookup key itself as the fallback\n return lookupKey\n }\n\n /**\n * Takes a translation string with placeholders, and replaces the placeholders\n * with the provided data\n *\n * @internal\n * @param {string} translationString - The translation string\n * @param {{ [key: string]: unknown }} options - Any options passed with the translation string, e.g: for string interpolation.\n * @returns {string} The translation string to output, with $\\{\\} placeholders replaced\n */\n replacePlaceholders(translationString, options) {\n const formatter = Intl.NumberFormat.supportedLocalesOf(this.locale).length\n ? new Intl.NumberFormat(this.locale)\n : undefined\n\n return translationString.replace(\n /%{(.\\S+)}/g,\n\n /**\n * Replace translation string placeholders\n *\n * @internal\n * @param {string} placeholderWithBraces - Placeholder with braces\n * @param {string} placeholderKey - Placeholder key\n * @returns {string} Placeholder value\n */\n function (placeholderWithBraces, placeholderKey) {\n if (Object.prototype.hasOwnProperty.call(options, placeholderKey)) {\n const placeholderValue = options[placeholderKey]\n\n // If a user has passed `false` as the value for the placeholder\n // treat it as though the value should not be displayed\n if (\n placeholderValue === false ||\n (typeof placeholderValue !== 'number' &&\n typeof placeholderValue !== 'string')\n ) {\n return ''\n }\n\n // If the placeholder's value is a number, localise the number formatting\n if (typeof placeholderValue === 'number') {\n return formatter\n ? formatter.format(placeholderValue)\n : `${placeholderValue}`\n }\n\n return placeholderValue\n }\n\n throw new Error(\n `i18n: no data found to replace ${placeholderWithBraces} placeholder in string`\n )\n }\n )\n }\n\n /**\n * Check to see if the browser supports Intl.PluralRules\n *\n * It requires all conditions to be met in order to be supported:\n * - The implementation of Intl supports PluralRules (NOT true in Safari 10–12)\n * - The browser/OS has plural rules for the current locale (browser dependent)\n *\n * {@link https://browsersl.ist/#q=supports+es6-module+and+not+supports+intl-pluralrules}\n *\n * @internal\n * @returns {boolean} Returns true if all conditions are met. Returns false otherwise.\n */\n hasIntlPluralRulesSupport() {\n return Boolean(\n 'PluralRules' in window.Intl &&\n Intl.PluralRules.supportedLocalesOf(this.locale).length\n )\n }\n\n /**\n * Get the appropriate suffix for the plural form.\n *\n * Uses Intl.PluralRules (or our own fallback implementation) to get the\n * 'preferred' form to use for the given count.\n *\n * Checks that a translation has been provided for that plural form – if it\n * hasn't, it'll fall back to the 'other' plural form (unless that doesn't exist\n * either, in which case an error will be thrown)\n *\n * @internal\n * @param {string} lookupKey - The lookup key of the string to use.\n * @param {number} count - Number used to determine which pluralisation to use.\n * @returns {PluralRule} The suffix associated with the correct pluralisation for this locale.\n * @throws {Error} Plural form `.other` required when preferred plural form is missing\n */\n getPluralSuffix(lookupKey, count) {\n // Validate that the number is actually a number.\n //\n // Number(count) will turn anything that can't be converted to a Number type\n // into 'NaN'. isFinite filters out NaN, as it isn't a finite number.\n count = Number(count)\n if (!isFinite(count)) {\n return 'other'\n }\n\n // Fetch the translation for that lookup key\n const translation = this.translations[lookupKey]\n\n // Check to verify that all the requirements for Intl.PluralRules are met.\n // If so, we can use that instead of our custom implementation. Otherwise,\n // use the hardcoded fallback.\n const preferredForm = this.hasIntlPluralRulesSupport()\n ? new Intl.PluralRules(this.locale).select(count)\n : this.selectPluralFormUsingFallbackRules(count)\n\n // Use the correct plural form if provided\n if (typeof translation === 'object') {\n if (preferredForm in translation) {\n return preferredForm\n // Fall back to `other` if the plural form is missing, but log a warning\n // to the console\n } else if ('other' in translation) {\n console.warn(\n `i18n: Missing plural form \".${preferredForm}\" for \"${this.locale}\" locale. Falling back to \".other\".`\n )\n\n return 'other'\n }\n }\n\n // If the required `other` plural form is missing, all we can do is error\n throw new Error(\n `i18n: Plural form \".other\" is required for \"${this.locale}\" locale`\n )\n }\n\n /**\n * Get the plural form using our fallback implementation\n *\n * This is split out into a separate function to make it easier to test the\n * fallback behaviour in an environment where Intl.PluralRules exists.\n *\n * @internal\n * @param {number} count - Number used to determine which pluralisation to use.\n * @returns {PluralRule} The pluralisation form for count in this locale.\n */\n selectPluralFormUsingFallbackRules(count) {\n // Currently our custom code can only handle positive integers, so let's\n // make sure our number is one of those.\n count = Math.abs(Math.floor(count))\n\n const ruleset = this.getPluralRulesForLocale()\n\n if (ruleset) {\n return I18n.pluralRules[ruleset](count)\n }\n\n return 'other'\n }\n\n /**\n * Work out which pluralisation rules to use for the current locale\n *\n * The locale may include a regional indicator (such as en-GB), but we don't\n * usually care about this part, as pluralisation rules are usually the same\n * regardless of region. There are exceptions, however, (e.g. Portuguese) so\n * this searches by both the full and shortened locale codes, just to be sure.\n *\n * @internal\n * @returns {string | undefined} The name of the pluralisation rule to use (a key for one\n * of the functions in this.pluralRules)\n */\n getPluralRulesForLocale() {\n const localeShort = this.locale.split('-')[0]\n\n // Look through the plural rules map to find which `pluralRule` is\n // appropriate for our current `locale`.\n for (const pluralRule in I18n.pluralRulesMap) {\n const languages = I18n.pluralRulesMap[pluralRule]\n if (languages.includes(this.locale) || languages.includes(localeShort)) {\n return pluralRule\n }\n }\n }\n\n /**\n * Map of plural rules to languages where those rules apply.\n *\n * Note: These groups are named for the most dominant or recognisable language\n * that uses each system. The groupings do not imply that the languages are\n * related to one another. Many languages have evolved the same systems\n * independently of one another.\n *\n * Code to support more languages can be found in the i18n spike:\n * {@link https://github.com/alphagov/govuk-frontend/blob/spike-i18n-support/src/govuk/i18n.mjs}\n *\n * Languages currently supported:\n *\n * Arabic: Arabic (ar)\n * Chinese: Burmese (my), Chinese (zh), Indonesian (id), Japanese (ja),\n * Javanese (jv), Korean (ko), Malay (ms), Thai (th), Vietnamese (vi)\n * French: Armenian (hy), Bangla (bn), French (fr), Gujarati (gu), Hindi (hi),\n * Persian Farsi (fa), Punjabi (pa), Zulu (zu)\n * German: Afrikaans (af), Albanian (sq), Azerbaijani (az), Basque (eu),\n * Bulgarian (bg), Catalan (ca), Danish (da), Dutch (nl), English (en),\n * Estonian (et), Finnish (fi), Georgian (ka), German (de), Greek (el),\n * Hungarian (hu), Luxembourgish (lb), Norwegian (no), Somali (so),\n * Swahili (sw), Swedish (sv), Tamil (ta), Telugu (te), Turkish (tr),\n * Urdu (ur)\n * Irish: Irish Gaelic (ga)\n * Russian: Russian (ru), Ukrainian (uk)\n * Scottish: Scottish Gaelic (gd)\n * Spanish: European Portuguese (pt-PT), Italian (it), Spanish (es)\n * Welsh: Welsh (cy)\n *\n * @internal\n * @type {{ [key: string]: string[] }}\n */\n static pluralRulesMap = {\n arabic: ['ar'],\n chinese: ['my', 'zh', 'id', 'ja', 'jv', 'ko', 'ms', 'th', 'vi'],\n french: ['hy', 'bn', 'fr', 'gu', 'hi', 'fa', 'pa', 'zu'],\n german: [\n 'af',\n 'sq',\n 'az',\n 'eu',\n 'bg',\n 'ca',\n 'da',\n 'nl',\n 'en',\n 'et',\n 'fi',\n 'ka',\n 'de',\n 'el',\n 'hu',\n 'lb',\n 'no',\n 'so',\n 'sw',\n 'sv',\n 'ta',\n 'te',\n 'tr',\n 'ur'\n ],\n irish: ['ga'],\n russian: ['ru', 'uk'],\n scottish: ['gd'],\n spanish: ['pt-PT', 'it', 'es'],\n welsh: ['cy']\n }\n\n /**\n * Different pluralisation rule sets\n *\n * Returns the appropriate suffix for the plural form associated with `n`.\n * Possible suffixes: 'zero', 'one', 'two', 'few', 'many', 'other' (the actual\n * meaning of each differs per locale). 'other' should always exist, even in\n * languages without plurals, such as Chinese.\n * {@link https://cldr.unicode.org/index/cldr-spec/plural-rules}\n *\n * The count must be a positive integer. Negative numbers and decimals aren't accounted for\n *\n * @internal\n * @type {{ [key: string]: (count: number) => PluralRule }}\n */\n static pluralRules = {\n arabic(n) {\n if (n === 0) {\n return 'zero'\n }\n if (n === 1) {\n return 'one'\n }\n if (n === 2) {\n return 'two'\n }\n if (n % 100 >= 3 && n % 100 <= 10) {\n return 'few'\n }\n if (n % 100 >= 11 && n % 100 <= 99) {\n return 'many'\n }\n return 'other'\n },\n chinese() {\n return 'other'\n },\n french(n) {\n return n === 0 || n === 1 ? 'one' : 'other'\n },\n german(n) {\n return n === 1 ? 'one' : 'other'\n },\n irish(n) {\n if (n === 1) {\n return 'one'\n }\n if (n === 2) {\n return 'two'\n }\n if (n >= 3 && n <= 6) {\n return 'few'\n }\n if (n >= 7 && n <= 10) {\n return 'many'\n }\n return 'other'\n },\n russian(n) {\n const lastTwo = n % 100\n const last = lastTwo % 10\n if (last === 1 && lastTwo !== 11) {\n return 'one'\n }\n if (last >= 2 && last <= 4 && !(lastTwo >= 12 && lastTwo <= 14)) {\n return 'few'\n }\n if (\n last === 0 ||\n (last >= 5 && last <= 9) ||\n (lastTwo >= 11 && lastTwo <= 14)\n ) {\n return 'many'\n }\n // Note: The 'other' suffix is only used by decimal numbers in Russian.\n // We don't anticipate it being used, but it's here for consistency.\n return 'other'\n },\n scottish(n) {\n if (n === 1 || n === 11) {\n return 'one'\n }\n if (n === 2 || n === 12) {\n return 'two'\n }\n if ((n >= 3 && n <= 10) || (n >= 13 && n <= 19)) {\n return 'few'\n }\n return 'other'\n },\n spanish(n) {\n if (n === 1) {\n return 'one'\n }\n if (n % 1000000 === 0 && n !== 0) {\n return 'many'\n }\n return 'other'\n },\n welsh(n) {\n if (n === 0) {\n return 'zero'\n }\n if (n === 1) {\n return 'one'\n }\n if (n === 2) {\n return 'two'\n }\n if (n === 3) {\n return 'few'\n }\n if (n === 6) {\n return 'many'\n }\n return 'other'\n }\n }\n}\n\n/**\n * Plural rule category mnemonic tags\n *\n * @internal\n * @typedef {'zero' | 'one' | 'two' | 'few' | 'many' | 'other'} PluralRule\n */\n\n/**\n * Translated message by plural rule they correspond to.\n *\n * Allows to group pluralised messages under a single key when passing\n * translations to a component's constructor\n *\n * @internal\n * @typedef {object} TranslationPluralForms\n * @property {string} [other] - General plural form\n * @property {string} [zero] - Plural form used with 0\n * @property {string} [one] - Plural form used with 1\n * @property {string} [two] - Plural form used with 2\n * @property {string} [few] - Plural form used for a few\n * @property {string} [many] - Plural form used for many\n */\n","import { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Accordion component\n *\n * This allows a collection of sections to be collapsed by default, showing only\n * their headers. Sections can be expanded or collapsed individually by clicking\n * their headers. A \"Show all sections\" button is also added to the top of the\n * accordion, which switches to \"Hide all sections\" when all the sections are\n * expanded.\n *\n * The state of each section is saved to the DOM via the `aria-expanded`\n * attribute, which also provides accessibility.\n *\n * @preserve\n */\nexport class Accordion extends GOVUKFrontendComponent {\n /**\n * @private\n * @type {AccordionConfig}\n */\n config\n\n /** @private */\n i18n\n\n /** @private */\n controlsClass = 'govuk-accordion__controls'\n\n /** @private */\n showAllClass = 'govuk-accordion__show-all'\n\n /** @private */\n showAllTextClass = 'govuk-accordion__show-all-text'\n\n /** @private */\n sectionClass = 'govuk-accordion__section'\n\n /** @private */\n sectionExpandedClass = 'govuk-accordion__section--expanded'\n\n /** @private */\n sectionButtonClass = 'govuk-accordion__section-button'\n\n /** @private */\n sectionHeaderClass = 'govuk-accordion__section-header'\n\n /** @private */\n sectionHeadingClass = 'govuk-accordion__section-heading'\n\n /** @private */\n sectionHeadingDividerClass = 'govuk-accordion__section-heading-divider'\n\n /** @private */\n sectionHeadingTextClass = 'govuk-accordion__section-heading-text'\n\n /** @private */\n sectionHeadingTextFocusClass = 'govuk-accordion__section-heading-text-focus'\n\n /** @private */\n sectionShowHideToggleClass = 'govuk-accordion__section-toggle'\n\n /** @private */\n sectionShowHideToggleFocusClass = 'govuk-accordion__section-toggle-focus'\n\n /** @private */\n sectionShowHideTextClass = 'govuk-accordion__section-toggle-text'\n\n /** @private */\n upChevronIconClass = 'govuk-accordion-nav__chevron'\n\n /** @private */\n downChevronIconClass = 'govuk-accordion-nav__chevron--down'\n\n /** @private */\n sectionSummaryClass = 'govuk-accordion__section-summary'\n\n /** @private */\n sectionSummaryFocusClass = 'govuk-accordion__section-summary-focus'\n\n /** @private */\n sectionContentClass = 'govuk-accordion__section-content'\n\n /** @private */\n $sections\n\n /**\n * @private\n * @type {HTMLButtonElement | null}\n */\n $showAllButton = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $showAllIcon = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $showAllText = null\n\n /**\n * @param {Element | null} $root - HTML element to use for accordion\n * @param {AccordionConfig} [config] - Accordion config\n */\n constructor($root, config = {}) {\n super($root)\n\n this.config = mergeConfigs(\n Accordion.defaults,\n config,\n normaliseDataset(Accordion, this.$root.dataset)\n )\n\n this.i18n = new I18n(this.config.i18n)\n\n const $sections = this.$root.querySelectorAll(`.${this.sectionClass}`)\n if (!$sections.length) {\n throw new ElementError({\n component: Accordion,\n identifier: `Sections (\\`<div class=\"${this.sectionClass}\">\\`)`\n })\n }\n\n this.$sections = $sections\n\n this.initControls()\n this.initSectionHeaders()\n\n this.updateShowAllButton(this.areAllSectionsOpen())\n }\n\n /**\n * Initialise controls and set attributes\n *\n * @private\n */\n initControls() {\n // Create \"Show all\" button and set attributes\n this.$showAllButton = document.createElement('button')\n this.$showAllButton.setAttribute('type', 'button')\n this.$showAllButton.setAttribute('class', this.showAllClass)\n this.$showAllButton.setAttribute('aria-expanded', 'false')\n\n // Create icon, add to element\n this.$showAllIcon = document.createElement('span')\n this.$showAllIcon.classList.add(this.upChevronIconClass)\n this.$showAllButton.appendChild(this.$showAllIcon)\n\n // Create control wrapper and add controls to it\n const $accordionControls = document.createElement('div')\n $accordionControls.setAttribute('class', this.controlsClass)\n $accordionControls.appendChild(this.$showAllButton)\n this.$root.insertBefore($accordionControls, this.$root.firstChild)\n\n // Build additional wrapper for Show all toggle text and place after icon\n this.$showAllText = document.createElement('span')\n this.$showAllText.classList.add(this.showAllTextClass)\n this.$showAllButton.appendChild(this.$showAllText)\n\n // Handle click events on the show/hide all button\n this.$showAllButton.addEventListener('click', () =>\n this.onShowOrHideAllToggle()\n )\n\n // Handle 'beforematch' events, if the user agent supports them\n if ('onbeforematch' in document) {\n document.addEventListener('beforematch', (event) =>\n this.onBeforeMatch(event)\n )\n }\n }\n\n /**\n * Initialise section headers\n *\n * @private\n */\n initSectionHeaders() {\n this.$sections.forEach(($section, i) => {\n const $header = $section.querySelector(`.${this.sectionHeaderClass}`)\n if (!$header) {\n throw new ElementError({\n component: Accordion,\n identifier: `Section headers (\\`<div class=\"${this.sectionHeaderClass}\">\\`)`\n })\n }\n\n // Set header attributes\n this.constructHeaderMarkup($header, i)\n this.setExpanded(this.isExpanded($section), $section)\n\n // Handle events\n $header.addEventListener('click', () => this.onSectionToggle($section))\n\n // See if there is any state stored in sessionStorage and set the sections\n // to open or closed.\n this.setInitialState($section)\n })\n }\n\n /**\n * Construct section header\n *\n * @private\n * @param {Element} $header - Section header\n * @param {number} index - Section index\n */\n constructHeaderMarkup($header, index) {\n const $span = $header.querySelector(`.${this.sectionButtonClass}`)\n const $heading = $header.querySelector(`.${this.sectionHeadingClass}`)\n const $summary = $header.querySelector(`.${this.sectionSummaryClass}`)\n\n if (!$heading) {\n throw new ElementError({\n component: Accordion,\n identifier: `Section heading (\\`.${this.sectionHeadingClass}\\`)`\n })\n }\n\n if (!$span) {\n throw new ElementError({\n component: Accordion,\n identifier: `Section button placeholder (\\`<span class=\"${this.sectionButtonClass}\">\\`)`\n })\n }\n\n // Create a button element that will replace the\n // '.govuk-accordion__section-button' span\n const $button = document.createElement('button')\n $button.setAttribute('type', 'button')\n $button.setAttribute(\n 'aria-controls',\n `${this.$root.id}-content-${index + 1}`\n )\n\n // Copy all attributes from $span to $button (except `id`, which gets added\n // to the `$headingText` element)\n for (const attr of Array.from($span.attributes)) {\n if (attr.name !== 'id') {\n $button.setAttribute(attr.name, attr.value)\n }\n }\n\n // Create container for heading text so it can be styled\n const $headingText = document.createElement('span')\n $headingText.classList.add(this.sectionHeadingTextClass)\n // Copy the span ID to the heading text to allow it to be referenced by\n // `aria-labelledby` on the hidden content area without \"Show this section\"\n $headingText.id = $span.id\n\n // Create an inner heading text container to limit the width of the focus\n // state\n const $headingTextFocus = document.createElement('span')\n $headingTextFocus.classList.add(this.sectionHeadingTextFocusClass)\n $headingText.appendChild($headingTextFocus)\n // span could contain HTML elements\n // (see https://www.w3.org/TR/2011/WD-html5-20110525/content-models.html#phrasing-content)\n Array.from($span.childNodes).forEach(($child) =>\n $headingTextFocus.appendChild($child)\n )\n\n // Create container for show / hide icons and text.\n const $showHideToggle = document.createElement('span')\n $showHideToggle.classList.add(this.sectionShowHideToggleClass)\n // Tell Google not to index the 'show' text as part of the heading. Must be\n // set on the element before it's added to the DOM.\n // See https://developers.google.com/search/docs/advanced/robots/robots_meta_tag#data-nosnippet-attr\n $showHideToggle.setAttribute('data-nosnippet', '')\n // Create an inner container to limit the width of the focus state\n const $showHideToggleFocus = document.createElement('span')\n $showHideToggleFocus.classList.add(this.sectionShowHideToggleFocusClass)\n $showHideToggle.appendChild($showHideToggleFocus)\n // Create wrapper for the show / hide text. Append text after the show/hide icon\n const $showHideText = document.createElement('span')\n const $showHideIcon = document.createElement('span')\n $showHideIcon.classList.add(this.upChevronIconClass)\n $showHideToggleFocus.appendChild($showHideIcon)\n $showHideText.classList.add(this.sectionShowHideTextClass)\n $showHideToggleFocus.appendChild($showHideText)\n\n // Append elements to the button:\n // 1. Heading text\n // 2. Punctuation\n // 3. (Optional: Summary line followed by punctuation)\n // 4. Show / hide toggle\n $button.appendChild($headingText)\n $button.appendChild(this.getButtonPunctuationEl())\n\n // If summary content exists add to DOM in correct order\n if ($summary) {\n // Create a new `span` element and copy the summary line content from the\n // original `div` to the new `span`. This is because the summary line text\n // is now inside a button element, which can only contain phrasing\n // content.\n const $summarySpan = document.createElement('span')\n // Create an inner summary container to limit the width of the summary\n // focus state\n const $summarySpanFocus = document.createElement('span')\n $summarySpanFocus.classList.add(this.sectionSummaryFocusClass)\n $summarySpan.appendChild($summarySpanFocus)\n\n // Get original attributes, and pass them to the replacement\n for (const attr of Array.from($summary.attributes)) {\n $summarySpan.setAttribute(attr.name, attr.value)\n }\n\n // Copy original contents of summary to the new summary span\n Array.from($summary.childNodes).forEach(($child) =>\n $summarySpanFocus.appendChild($child)\n )\n\n // Replace the original summary `div` with the new summary `span`\n $summary.remove()\n\n $button.appendChild($summarySpan)\n $button.appendChild(this.getButtonPunctuationEl())\n }\n\n $button.appendChild($showHideToggle)\n\n $heading.removeChild($span)\n $heading.appendChild($button)\n }\n\n /**\n * When a section is opened by the user agent via the 'beforematch' event\n *\n * @private\n * @param {Event} event - Generic event\n */\n onBeforeMatch(event) {\n const $fragment = event.target\n\n // Handle elements with `.closest()` support only\n if (!($fragment instanceof Element)) {\n return\n }\n\n // Handle when fragment is inside section\n const $section = $fragment.closest(`.${this.sectionClass}`)\n if ($section) {\n this.setExpanded(true, $section)\n }\n }\n\n /**\n * When section toggled, set and store state\n *\n * @private\n * @param {Element} $section - Section element\n */\n onSectionToggle($section) {\n const nowExpanded = !this.isExpanded($section)\n this.setExpanded(nowExpanded, $section)\n\n // Store the state in sessionStorage when a change is triggered\n this.storeState($section, nowExpanded)\n }\n\n /**\n * When Open/Close All toggled, set and store state\n *\n * @private\n */\n onShowOrHideAllToggle() {\n const nowExpanded = !this.areAllSectionsOpen()\n\n this.$sections.forEach(($section) => {\n this.setExpanded(nowExpanded, $section)\n this.storeState($section, nowExpanded)\n })\n\n this.updateShowAllButton(nowExpanded)\n }\n\n /**\n * Set section attributes when opened/closed\n *\n * @private\n * @param {boolean} expanded - Section expanded\n * @param {Element} $section - Section element\n */\n setExpanded(expanded, $section) {\n const $showHideIcon = $section.querySelector(`.${this.upChevronIconClass}`)\n const $showHideText = $section.querySelector(\n `.${this.sectionShowHideTextClass}`\n )\n const $button = $section.querySelector(`.${this.sectionButtonClass}`)\n const $content = $section.querySelector(`.${this.sectionContentClass}`)\n\n if (!$content) {\n throw new ElementError({\n component: Accordion,\n identifier: `Section content (\\`<div class=\"${this.sectionContentClass}\">\\`)`\n })\n }\n\n if (!$showHideIcon || !$showHideText || !$button) {\n // Return early for elements we create\n return\n }\n\n const newButtonText = expanded\n ? this.i18n.t('hideSection')\n : this.i18n.t('showSection')\n\n $showHideText.textContent = newButtonText\n $button.setAttribute('aria-expanded', `${expanded}`)\n\n // Update aria-label combining\n const ariaLabelParts = []\n\n const $headingText = $section.querySelector(\n `.${this.sectionHeadingTextClass}`\n )\n if ($headingText) {\n ariaLabelParts.push(`${$headingText.textContent}`.trim())\n }\n\n const $summary = $section.querySelector(`.${this.sectionSummaryClass}`)\n if ($summary) {\n ariaLabelParts.push(`${$summary.textContent}`.trim())\n }\n\n const ariaLabelMessage = expanded\n ? this.i18n.t('hideSectionAriaLabel')\n : this.i18n.t('showSectionAriaLabel')\n ariaLabelParts.push(ariaLabelMessage)\n\n /*\n * Join with a comma to add pause for assistive technology.\n * Example: [heading]Section A ,[pause] Show this section.\n * https://accessibility.blog.gov.uk/2017/12/18/what-working-on-gov-uk-navigation-taught-us-about-accessibility/\n */\n $button.setAttribute('aria-label', ariaLabelParts.join(' , '))\n\n // Swap icon, change class\n if (expanded) {\n $content.removeAttribute('hidden')\n $section.classList.add(this.sectionExpandedClass)\n $showHideIcon.classList.remove(this.downChevronIconClass)\n } else {\n $content.setAttribute('hidden', 'until-found')\n $section.classList.remove(this.sectionExpandedClass)\n $showHideIcon.classList.add(this.downChevronIconClass)\n }\n\n // See if \"Show all sections\" button text should be updated\n this.updateShowAllButton(this.areAllSectionsOpen())\n }\n\n /**\n * Get state of section\n *\n * @private\n * @param {Element} $section - Section element\n * @returns {boolean} True if expanded\n */\n isExpanded($section) {\n return $section.classList.contains(this.sectionExpandedClass)\n }\n\n /**\n * Check if all sections are open\n *\n * @private\n * @returns {boolean} True if all sections are open\n */\n areAllSectionsOpen() {\n return Array.from(this.$sections).every(($section) =>\n this.isExpanded($section)\n )\n }\n\n /**\n * Update \"Show all sections\" button\n *\n * @private\n * @param {boolean} expanded - Section expanded\n */\n updateShowAllButton(expanded) {\n if (!this.$showAllButton || !this.$showAllText || !this.$showAllIcon) {\n return\n }\n\n this.$showAllButton.setAttribute('aria-expanded', expanded.toString())\n this.$showAllText.textContent = expanded\n ? this.i18n.t('hideAllSections')\n : this.i18n.t('showAllSections')\n this.$showAllIcon.classList.toggle(this.downChevronIconClass, !expanded)\n }\n\n /**\n * Get the identifier for a section\n *\n * We need a unique way of identifying each content in the Accordion.\n * Since an `#id` should be unique and an `id` is required for `aria-`\n * attributes `id` can be safely used.\n *\n * @param {Element} $section - Section element\n * @returns {string | undefined | null} Identifier for section\n */\n getIdentifier($section) {\n const $button = $section.querySelector(`.${this.sectionButtonClass}`)\n\n return $button?.getAttribute('aria-controls')\n }\n\n /**\n * Set the state of the accordions in sessionStorage\n *\n * @private\n * @param {Element} $section - Section element\n * @param {boolean} isExpanded - Whether the section is expanded\n */\n storeState($section, isExpanded) {\n if (!this.config.rememberExpanded) {\n return\n }\n\n const id = this.getIdentifier($section)\n\n if (id) {\n try {\n window.sessionStorage.setItem(id, isExpanded.toString())\n } catch (exception) {}\n }\n }\n\n /**\n * Read the state of the accordions from sessionStorage\n *\n * @private\n * @param {Element} $section - Section element\n */\n setInitialState($section) {\n if (!this.config.rememberExpanded) {\n return\n }\n\n const id = this.getIdentifier($section)\n\n if (id) {\n try {\n const state = window.sessionStorage.getItem(id)\n\n if (state !== null) {\n this.setExpanded(state === 'true', $section)\n }\n } catch (exception) {}\n }\n }\n\n /**\n * Create an element to improve semantics of the section button with\n * punctuation\n *\n * Adding punctuation to the button can also improve its general semantics by\n * dividing its contents into thematic chunks. See\n * https://github.com/alphagov/govuk-frontend/issues/2327#issuecomment-922957442\n *\n * @private\n * @returns {Element} DOM element\n */\n getButtonPunctuationEl() {\n const $punctuationEl = document.createElement('span')\n $punctuationEl.classList.add(\n 'govuk-visually-hidden',\n this.sectionHeadingDividerClass\n )\n $punctuationEl.textContent = ', '\n return $punctuationEl\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-accordion'\n\n /**\n * Accordion default config\n *\n * @see {@link AccordionConfig}\n * @constant\n * @type {AccordionConfig}\n */\n static defaults = Object.freeze({\n i18n: {\n hideAllSections: 'Hide all sections',\n hideSection: 'Hide',\n hideSectionAriaLabel: 'Hide this section',\n showAllSections: 'Show all sections',\n showSection: 'Show',\n showSectionAriaLabel: 'Show this section'\n },\n rememberExpanded: true\n })\n\n /**\n * Accordion config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' },\n rememberExpanded: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Accordion config\n *\n * @see {@link Accordion.defaults}\n * @typedef {object} AccordionConfig\n * @property {AccordionTranslations} [i18n=Accordion.defaults.i18n] - Accordion translations\n * @property {boolean} [rememberExpanded] - Whether the expanded and collapsed\n * state of each section is remembered and restored when navigating.\n */\n\n/**\n * Accordion translations\n *\n * @see {@link Accordion.defaults.i18n}\n * @typedef {object} AccordionTranslations\n *\n * Messages used by the component for the labels of its buttons. This includes\n * the visible text shown on screen, and text to help assistive technology users\n * for the buttons toggling each section.\n * @property {string} [hideAllSections] - The text content for the 'Hide all\n * sections' button, used when at least one section is expanded.\n * @property {string} [hideSection] - The text content for the 'Hide'\n * button, used when a section is expanded.\n * @property {string} [hideSectionAriaLabel] - The text content appended to the\n * 'Hide' button's accessible name when a section is expanded.\n * @property {string} [showAllSections] - The text content for the 'Show all\n * sections' button, used when all sections are collapsed.\n * @property {string} [showSection] - The text content for the 'Show'\n * button, used when a section is collapsed.\n * @property {string} [showSectionAriaLabel] - The text content appended to the\n * 'Show' button's accessible name when a section is expanded.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\nconst DEBOUNCE_TIMEOUT_IN_SECONDS = 1\n\n/**\n * JavaScript enhancements for the Button component\n *\n * @preserve\n */\nexport class Button extends GOVUKFrontendComponent {\n /**\n * @private\n * @type {ButtonConfig}\n */\n config\n\n /**\n * @private\n * @type {number | null}\n */\n debounceFormSubmitTimer = null\n\n /**\n * @param {Element | null} $root - HTML element to use for button\n * @param {ButtonConfig} [config] - Button config\n */\n constructor($root, config = {}) {\n super($root)\n\n this.config = mergeConfigs(\n Button.defaults,\n config,\n normaliseDataset(Button, this.$root.dataset)\n )\n\n this.$root.addEventListener('keydown', (event) => this.handleKeyDown(event))\n this.$root.addEventListener('click', (event) => this.debounce(event))\n }\n\n /**\n * Trigger a click event when the space key is pressed\n *\n * Some screen readers tell users they can use the space bar to activate\n * things with the 'button' role, so we need to match the functionality of\n * native HTML buttons.\n *\n * See https://github.com/alphagov/govuk_elements/pull/272#issuecomment-233028270\n *\n * @private\n * @param {KeyboardEvent} event - Keydown event\n */\n handleKeyDown(event) {\n const $target = event.target\n\n // Handle space bar only\n if (event.key !== ' ') {\n return\n }\n\n // Handle elements with [role=\"button\"] only\n if (\n $target instanceof HTMLElement &&\n $target.getAttribute('role') === 'button'\n ) {\n event.preventDefault() // prevent the page from scrolling\n $target.click()\n }\n }\n\n /**\n * Debounce double-clicks\n *\n * If the click quickly succeeds a previous click then nothing will happen.\n * This stops people accidentally causing multiple form submissions by double\n * clicking buttons.\n *\n * @private\n * @param {MouseEvent} event - Mouse click event\n * @returns {undefined | false} Returns undefined, or false when debounced\n */\n debounce(event) {\n // Check the button that was clicked has preventDoubleClick enabled\n if (!this.config.preventDoubleClick) {\n return\n }\n\n // If the timer is still running, prevent the click from submitting the form\n if (this.debounceFormSubmitTimer) {\n event.preventDefault()\n return false\n }\n\n this.debounceFormSubmitTimer = window.setTimeout(() => {\n this.debounceFormSubmitTimer = null\n }, DEBOUNCE_TIMEOUT_IN_SECONDS * 1000)\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-button'\n\n /**\n * Button default config\n *\n * @see {@link ButtonConfig}\n * @constant\n * @type {ButtonConfig}\n */\n static defaults = Object.freeze({\n preventDoubleClick: false\n })\n\n /**\n * Button config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n preventDoubleClick: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Button config\n *\n * @typedef {object} ButtonConfig\n * @property {boolean} [preventDoubleClick=false] - Prevent accidental double\n * clicks on submit buttons from submitting forms multiple times.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","/**\n * Returns the value of the given attribute closest to the given element (including itself)\n *\n * @internal\n * @param {Element} $element - The element to start walking the DOM tree up\n * @param {string} attributeName - The name of the attribute\n * @returns {string | null} Attribute value\n */\nexport function closestAttributeValue($element, attributeName) {\n const $closestElementWithAttribute = $element.closest(`[${attributeName}]`)\n return $closestElementWithAttribute\n ? $closestElementWithAttribute.getAttribute(attributeName)\n : null\n}\n","import { closestAttributeValue } from '../../common/closest-attribute-value.mjs'\nimport {\n formatErrorMessage,\n mergeConfigs,\n validateConfig\n} from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ConfigError, ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Character count component\n *\n * Tracks the number of characters or words in the `.govuk-js-character-count`\n * `<textarea>` inside the element. Displays a message with the remaining number\n * of characters/words available, or the number of characters/words in excess.\n *\n * You can configure the message to only appear after a certain percentage\n * of the available characters/words has been entered.\n *\n * @preserve\n */\nexport class CharacterCount extends GOVUKFrontendComponent {\n /** @private */\n $textarea\n\n /** @private */\n $visibleCountMessage\n\n /** @private */\n $screenReaderCountMessage\n\n /**\n * @private\n * @type {number | null}\n */\n lastInputTimestamp = null\n\n /** @private */\n lastInputValue = ''\n\n /**\n * @private\n * @type {number | null}\n */\n valueChecker = null\n\n /**\n * @private\n * @type {CharacterCountConfig}\n */\n config\n\n /** @private */\n i18n\n\n /** @private */\n maxLength\n\n /**\n * @param {Element | null} $root - HTML element to use for character count\n * @param {CharacterCountConfig} [config] - Character count config\n */\n constructor($root, config = {}) {\n super($root)\n\n const $textarea = this.$root.querySelector('.govuk-js-character-count')\n if (\n !(\n $textarea instanceof HTMLTextAreaElement ||\n $textarea instanceof HTMLInputElement\n )\n ) {\n throw new ElementError({\n component: CharacterCount,\n element: $textarea,\n expectedType: 'HTMLTextareaElement or HTMLInputElement',\n identifier: 'Form field (`.govuk-js-character-count`)'\n })\n }\n\n // Read config set using dataset ('data-' values)\n const datasetConfig = normaliseDataset(CharacterCount, this.$root.dataset)\n\n // To ensure data-attributes take complete precedence, even if they change\n // the type of count, we need to reset the `maxlength` and `maxwords` from\n // the JavaScript config.\n //\n // We can't mutate `config`, though, as it may be shared across multiple\n // components inside `initAll`.\n /** @type {CharacterCountConfig} */\n let configOverrides = {}\n if ('maxwords' in datasetConfig || 'maxlength' in datasetConfig) {\n configOverrides = {\n maxlength: undefined,\n maxwords: undefined\n }\n }\n\n this.config = mergeConfigs(\n CharacterCount.defaults,\n config,\n configOverrides,\n datasetConfig\n )\n\n // Check for valid config\n const errors = validateConfig(CharacterCount.schema, this.config)\n if (errors[0]) {\n throw new ConfigError(formatErrorMessage(CharacterCount, errors[0]))\n }\n\n this.i18n = new I18n(this.config.i18n, {\n // Read the fallback if necessary rather than have it set in the defaults\n locale: closestAttributeValue(this.$root, 'lang')\n })\n\n // Determine the limit attribute (characters or words)\n this.maxLength = this.config.maxwords ?? this.config.maxlength ?? Infinity\n\n this.$textarea = $textarea\n\n const textareaDescriptionId = `${this.$textarea.id}-info`\n const $textareaDescription = document.getElementById(textareaDescriptionId)\n if (!$textareaDescription) {\n throw new ElementError({\n component: CharacterCount,\n element: $textareaDescription,\n identifier: `Count message (\\`id=\"${textareaDescriptionId}\"\\`)`\n })\n }\n\n // Inject a description for the textarea if none is present already\n // for when the component was rendered with no maxlength, maxwords\n // nor custom textareaDescriptionText\n if (`${$textareaDescription.textContent}`.match(/^\\s*$/)) {\n $textareaDescription.textContent = this.i18n.t('textareaDescription', {\n count: this.maxLength\n })\n }\n\n // Move the textarea description to be immediately after the textarea\n // Kept for backwards compatibility\n this.$textarea.insertAdjacentElement('afterend', $textareaDescription)\n\n // Create the *screen reader* specific live-updating counter\n // This doesn't need any styling classes, as it is never visible\n const $screenReaderCountMessage = document.createElement('div')\n $screenReaderCountMessage.className =\n 'govuk-character-count__sr-status govuk-visually-hidden'\n $screenReaderCountMessage.setAttribute('aria-live', 'polite')\n this.$screenReaderCountMessage = $screenReaderCountMessage\n $textareaDescription.insertAdjacentElement(\n 'afterend',\n $screenReaderCountMessage\n )\n\n // Create our live-updating counter element, copying the classes from the\n // textarea description for backwards compatibility as these may have been\n // configured\n const $visibleCountMessage = document.createElement('div')\n $visibleCountMessage.className = $textareaDescription.className\n $visibleCountMessage.classList.add('govuk-character-count__status')\n $visibleCountMessage.setAttribute('aria-hidden', 'true')\n this.$visibleCountMessage = $visibleCountMessage\n $textareaDescription.insertAdjacentElement('afterend', $visibleCountMessage)\n\n // Hide the textarea description\n $textareaDescription.classList.add('govuk-visually-hidden')\n\n // Remove hard limit if set\n this.$textarea.removeAttribute('maxlength')\n\n this.bindChangeEvents()\n\n // When the page is restored after navigating 'back' in some browsers the\n // state of form controls is not restored until *after* the DOMContentLoaded\n // event is fired, so we need to sync after the pageshow event.\n window.addEventListener('pageshow', () => this.updateCountMessage())\n\n // Although we've set up handlers to sync state on the pageshow event, init\n // could be called after those events have fired, for example if they are\n // added to the page dynamically, so update now too.\n this.updateCountMessage()\n }\n\n /**\n * Bind change events\n *\n * Set up event listeners on the $textarea so that the count messages update\n * when the user types.\n *\n * @private\n */\n bindChangeEvents() {\n this.$textarea.addEventListener('keyup', () => this.handleKeyUp())\n\n // Bind focus/blur events to start/stop polling\n this.$textarea.addEventListener('focus', () => this.handleFocus())\n this.$textarea.addEventListener('blur', () => this.handleBlur())\n }\n\n /**\n * Handle key up event\n *\n * Update the visible character counter and keep track of when the last update\n * happened for each keypress\n *\n * @private\n */\n handleKeyUp() {\n this.updateVisibleCountMessage()\n this.lastInputTimestamp = Date.now()\n }\n\n /**\n * Handle focus event\n *\n * Speech recognition software such as Dragon NaturallySpeaking will modify\n * the fields by directly changing its `value`. These changes don't trigger\n * events in JavaScript, so we need to poll to handle when and if they occur.\n *\n * Once the keyup event hasn't been detected for at least 1000 ms (1s), check\n * if the textarea value has changed and update the count message if it has.\n *\n * This is so that the update triggered by the manual comparison doesn't\n * conflict with debounced KeyboardEvent updates.\n *\n * @private\n */\n handleFocus() {\n this.valueChecker = window.setInterval(() => {\n if (\n !this.lastInputTimestamp ||\n Date.now() - 500 >= this.lastInputTimestamp\n ) {\n this.updateIfValueChanged()\n }\n }, 1000)\n }\n\n /**\n * Handle blur event\n *\n * Stop checking the textarea value once the textarea no longer has focus\n *\n * @private\n */\n handleBlur() {\n // Cancel value checking on blur\n if (this.valueChecker) {\n window.clearInterval(this.valueChecker)\n }\n }\n\n /**\n * Update count message if textarea value has changed\n *\n * @private\n */\n updateIfValueChanged() {\n if (this.$textarea.value !== this.lastInputValue) {\n this.lastInputValue = this.$textarea.value\n this.updateCountMessage()\n }\n }\n\n /**\n * Update count message\n *\n * Helper function to update both the visible and screen reader-specific\n * counters simultaneously (e.g. on init)\n *\n * @private\n */\n updateCountMessage() {\n this.updateVisibleCountMessage()\n this.updateScreenReaderCountMessage()\n }\n\n /**\n * Update visible count message\n *\n * @private\n */\n updateVisibleCountMessage() {\n const remainingNumber = this.maxLength - this.count(this.$textarea.value)\n const isError = remainingNumber < 0\n\n // If input is over the threshold, remove the disabled class which renders\n // the counter invisible.\n this.$visibleCountMessage.classList.toggle(\n 'govuk-character-count__message--disabled',\n !this.isOverThreshold()\n )\n\n // Update styles\n this.$textarea.classList.toggle('govuk-textarea--error', isError)\n this.$visibleCountMessage.classList.toggle('govuk-error-message', isError)\n this.$visibleCountMessage.classList.toggle('govuk-hint', !isError)\n\n // Update message\n this.$visibleCountMessage.textContent = this.getCountMessage()\n }\n\n /**\n * Update screen reader count message\n *\n * @private\n */\n updateScreenReaderCountMessage() {\n // If over the threshold, remove the aria-hidden attribute, allowing screen\n // readers to announce the content of the element.\n if (this.isOverThreshold()) {\n this.$screenReaderCountMessage.removeAttribute('aria-hidden')\n } else {\n this.$screenReaderCountMessage.setAttribute('aria-hidden', 'true')\n }\n\n // Update message\n this.$screenReaderCountMessage.textContent = this.getCountMessage()\n }\n\n /**\n * Count the number of characters (or words, if `config.maxwords` is set)\n * in the given text\n *\n * @private\n * @param {string} text - The text to count the characters of\n * @returns {number} the number of characters (or words) in the text\n */\n count(text) {\n if (this.config.maxwords) {\n const tokens = text.match(/\\S+/g) ?? [] // Matches consecutive non-whitespace chars\n return tokens.length\n }\n\n return text.length\n }\n\n /**\n * Get count message\n *\n * @private\n * @returns {string} Status message\n */\n getCountMessage() {\n const remainingNumber = this.maxLength - this.count(this.$textarea.value)\n const countType = this.config.maxwords ? 'words' : 'characters'\n return this.formatCountMessage(remainingNumber, countType)\n }\n\n /**\n * Formats the message shown to users according to what's counted\n * and how many remain\n *\n * @private\n * @param {number} remainingNumber - The number of words/characaters remaining\n * @param {string} countType - \"words\" or \"characters\"\n * @returns {string} Status message\n */\n formatCountMessage(remainingNumber, countType) {\n if (remainingNumber === 0) {\n return this.i18n.t(`${countType}AtLimit`)\n }\n\n const translationKeySuffix =\n remainingNumber < 0 ? 'OverLimit' : 'UnderLimit'\n\n return this.i18n.t(`${countType}${translationKeySuffix}`, {\n count: Math.abs(remainingNumber)\n })\n }\n\n /**\n * Check if count is over threshold\n *\n * Checks whether the value is over the configured threshold for the input.\n * If there is no configured threshold, it is set to 0 and this function will\n * always return true.\n *\n * @private\n * @returns {boolean} true if the current count is over the config.threshold\n * (or no threshold is set)\n */\n isOverThreshold() {\n // No threshold means we're always above threshold so save some computation\n if (!this.config.threshold) {\n return true\n }\n\n // Determine the remaining number of characters/words\n const currentLength = this.count(this.$textarea.value)\n const maxLength = this.maxLength\n\n const thresholdValue = (maxLength * this.config.threshold) / 100\n\n return thresholdValue <= currentLength\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-character-count'\n\n /**\n * Character count default config\n *\n * @see {@link CharacterCountConfig}\n * @constant\n * @type {CharacterCountConfig}\n */\n static defaults = Object.freeze({\n threshold: 0,\n i18n: {\n // Characters\n charactersUnderLimit: {\n one: 'You have %{count} character remaining',\n other: 'You have %{count} characters remaining'\n },\n charactersAtLimit: 'You have 0 characters remaining',\n charactersOverLimit: {\n one: 'You have %{count} character too many',\n other: 'You have %{count} characters too many'\n },\n // Words\n wordsUnderLimit: {\n one: 'You have %{count} word remaining',\n other: 'You have %{count} words remaining'\n },\n wordsAtLimit: 'You have 0 words remaining',\n wordsOverLimit: {\n one: 'You have %{count} word too many',\n other: 'You have %{count} words too many'\n },\n textareaDescription: {\n other: ''\n }\n }\n })\n\n /**\n * Character count config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' },\n maxwords: { type: 'number' },\n maxlength: { type: 'number' },\n threshold: { type: 'number' }\n },\n anyOf: [\n {\n required: ['maxwords'],\n errorMessage: 'Either \"maxlength\" or \"maxwords\" must be provided'\n },\n {\n required: ['maxlength'],\n errorMessage: 'Either \"maxlength\" or \"maxwords\" must be provided'\n }\n ]\n })\n}\n\n/**\n * Character count config\n *\n * @see {@link CharacterCount.defaults}\n * @typedef {object} CharacterCountConfig\n * @property {number} [maxlength] - The maximum number of characters.\n * If maxwords is provided, the maxlength option will be ignored.\n * @property {number} [maxwords] - The maximum number of words. If maxwords is\n * provided, the maxlength option will be ignored.\n * @property {number} [threshold=0] - The percentage value of the limit at\n * which point the count message is displayed. If this attribute is set, the\n * count message will be hidden by default.\n * @property {CharacterCountTranslations} [i18n=CharacterCount.defaults.i18n] - Character count translations\n */\n\n/**\n * Character count translations\n *\n * @see {@link CharacterCount.defaults.i18n}\n * @typedef {object} CharacterCountTranslations\n *\n * Messages shown to users as they type. It provides feedback on how many words\n * or characters they have remaining or if they are over the limit. This also\n * includes a message used as an accessible description for the textarea.\n * @property {TranslationPluralForms} [charactersUnderLimit] - Message displayed\n * when the number of characters is under the configured maximum, `maxlength`.\n * This message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining characters. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {string} [charactersAtLimit] - Message displayed when the number of\n * characters reaches the configured maximum, `maxlength`. This message is\n * displayed visually and through assistive technologies.\n * @property {TranslationPluralForms} [charactersOverLimit] - Message displayed\n * when the number of characters is over the configured maximum, `maxlength`.\n * This message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining characters. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {TranslationPluralForms} [wordsUnderLimit] - Message displayed when\n * the number of words is under the configured maximum, `maxlength`. This\n * message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining words. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {string} [wordsAtLimit] - Message displayed when the number of\n * words reaches the configured maximum, `maxlength`. This message is\n * displayed visually and through assistive technologies.\n * @property {TranslationPluralForms} [wordsOverLimit] - Message displayed when\n * the number of words is over the configured maximum, `maxlength`. This\n * message is displayed visually and through assistive technologies. The\n * component will replace the `%{count}` placeholder with the number of\n * remaining words. This is a [pluralised list of\n * messages](https://frontend.design-system.service.gov.uk/localise-govuk-frontend).\n * @property {TranslationPluralForms} [textareaDescription] - Message made\n * available to assistive technologies, if none is already present in the\n * HTML, to describe that the component accepts only a limited amount of\n * content. It is visible on the page when JavaScript is unavailable. The\n * component will replace the `%{count}` placeholder with the value of the\n * `maxlength` or `maxwords` parameter.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n * @typedef {import('../../i18n.mjs').TranslationPluralForms} TranslationPluralForms\n */\n","import { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Checkboxes component\n *\n * @preserve\n */\nexport class Checkboxes extends GOVUKFrontendComponent {\n /** @private */\n $inputs\n\n /**\n * Checkboxes can be associated with a 'conditionally revealed' content block\n * – for example, a checkbox for 'Phone' could reveal an additional form field\n * for the user to enter their phone number.\n *\n * These associations are made using a `data-aria-controls` attribute, which\n * is promoted to an aria-controls attribute during initialisation.\n *\n * We also need to restore the state of any conditional reveals on the page\n * (for example if the user has navigated back), and set up event handlers to\n * keep the reveal in sync with the checkbox state.\n *\n * @param {Element | null} $root - HTML element to use for checkboxes\n */\n constructor($root) {\n super($root)\n\n const $inputs = this.$root.querySelectorAll('input[type=\"checkbox\"]')\n if (!$inputs.length) {\n throw new ElementError({\n component: Checkboxes,\n identifier: 'Form inputs (`<input type=\"checkbox\">`)'\n })\n }\n\n this.$inputs = $inputs\n\n this.$inputs.forEach(($input) => {\n const targetId = $input.getAttribute('data-aria-controls')\n\n // Skip radios without data-aria-controls attributes\n if (!targetId) {\n return\n }\n\n // Throw if target conditional element does not exist.\n if (!document.getElementById(targetId)) {\n throw new ElementError({\n component: Checkboxes,\n identifier: `Conditional reveal (\\`id=\"${targetId}\"\\`)`\n })\n }\n\n // Promote the data-aria-controls attribute to a aria-controls attribute\n // so that the relationship is exposed in the AOM\n $input.setAttribute('aria-controls', targetId)\n $input.removeAttribute('data-aria-controls')\n })\n\n // When the page is restored after navigating 'back' in some browsers the\n // state of form controls is not restored until *after* the DOMContentLoaded\n // event is fired, so we need to sync after the pageshow event.\n window.addEventListener('pageshow', () => this.syncAllConditionalReveals())\n\n // Although we've set up handlers to sync state on the pageshow event, init\n // could be called after those events have fired, for example if they are\n // added to the page dynamically, so sync now too.\n this.syncAllConditionalReveals()\n\n // Handle events\n this.$root.addEventListener('click', (event) => this.handleClick(event))\n }\n\n /**\n * Sync the conditional reveal states for all checkboxes in this component.\n *\n * @private\n */\n syncAllConditionalReveals() {\n this.$inputs.forEach(($input) =>\n this.syncConditionalRevealWithInputState($input)\n )\n }\n\n /**\n * Sync conditional reveal with the input state\n *\n * Synchronise the visibility of the conditional reveal, and its accessible\n * state, with the input's checked state.\n *\n * @private\n * @param {HTMLInputElement} $input - Checkbox input\n */\n syncConditionalRevealWithInputState($input) {\n const targetId = $input.getAttribute('aria-controls')\n if (!targetId) {\n return\n }\n\n const $target = document.getElementById(targetId)\n if ($target?.classList.contains('govuk-checkboxes__conditional')) {\n const inputIsChecked = $input.checked\n\n $input.setAttribute('aria-expanded', inputIsChecked.toString())\n $target.classList.toggle(\n 'govuk-checkboxes__conditional--hidden',\n !inputIsChecked\n )\n }\n }\n\n /**\n * Uncheck other checkboxes\n *\n * Find any other checkbox inputs with the same name value, and uncheck them.\n * This is useful for when a “None of these\" checkbox is checked.\n *\n * @private\n * @param {HTMLInputElement} $input - Checkbox input\n */\n unCheckAllInputsExcept($input) {\n const allInputsWithSameName = document.querySelectorAll(\n `input[type=\"checkbox\"][name=\"${$input.name}\"]`\n )\n\n allInputsWithSameName.forEach(($inputWithSameName) => {\n const hasSameFormOwner = $input.form === $inputWithSameName.form\n if (hasSameFormOwner && $inputWithSameName !== $input) {\n $inputWithSameName.checked = false\n this.syncConditionalRevealWithInputState($inputWithSameName)\n }\n })\n }\n\n /**\n * Uncheck exclusive checkboxes\n *\n * Find any checkbox inputs with the same name value and the 'exclusive'\n * behaviour, and uncheck them. This helps prevent someone checking both a\n * regular checkbox and a \"None of these\" checkbox in the same fieldset.\n *\n * @private\n * @param {HTMLInputElement} $input - Checkbox input\n */\n unCheckExclusiveInputs($input) {\n const allInputsWithSameNameAndExclusiveBehaviour =\n document.querySelectorAll(\n `input[data-behaviour=\"exclusive\"][type=\"checkbox\"][name=\"${$input.name}\"]`\n )\n\n allInputsWithSameNameAndExclusiveBehaviour.forEach(($exclusiveInput) => {\n const hasSameFormOwner = $input.form === $exclusiveInput.form\n if (hasSameFormOwner) {\n $exclusiveInput.checked = false\n this.syncConditionalRevealWithInputState($exclusiveInput)\n }\n })\n }\n\n /**\n * Click event handler\n *\n * Handle a click within the component root – if the click occurred on a checkbox,\n * sync the state of any associated conditional reveal with the checkbox\n * state.\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n handleClick(event) {\n const $clickedInput = event.target\n\n // Ignore clicks on things that aren't checkbox inputs\n if (\n !($clickedInput instanceof HTMLInputElement) ||\n $clickedInput.type !== 'checkbox'\n ) {\n return\n }\n\n // If the checkbox conditionally-reveals some content, sync the state\n const hasAriaControls = $clickedInput.getAttribute('aria-controls')\n if (hasAriaControls) {\n this.syncConditionalRevealWithInputState($clickedInput)\n }\n\n // No further behaviour needed for unchecking\n if (!$clickedInput.checked) {\n return\n }\n\n // Handle 'exclusive' checkbox behaviour (ie \"None of these\")\n const hasBehaviourExclusive =\n $clickedInput.getAttribute('data-behaviour') === 'exclusive'\n if (hasBehaviourExclusive) {\n this.unCheckAllInputsExcept($clickedInput)\n } else {\n this.unCheckExclusiveInputs($clickedInput)\n }\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-checkboxes'\n}\n","import {\n getFragmentFromUrl,\n mergeConfigs,\n setFocus\n} from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Error summary component\n *\n * Takes focus on initialisation for accessible announcement, unless disabled in\n * configuration.\n *\n * @preserve\n */\nexport class ErrorSummary extends GOVUKFrontendComponent {\n /**\n * @private\n * @type {ErrorSummaryConfig}\n */\n config\n\n /**\n * @param {Element | null} $root - HTML element to use for error summary\n * @param {ErrorSummaryConfig} [config] - Error summary config\n */\n constructor($root, config = {}) {\n super($root)\n\n this.config = mergeConfigs(\n ErrorSummary.defaults,\n config,\n normaliseDataset(ErrorSummary, this.$root.dataset)\n )\n\n /**\n * Focus the error summary\n */\n if (!this.config.disableAutoFocus) {\n setFocus(this.$root)\n }\n\n this.$root.addEventListener('click', (event) => this.handleClick(event))\n }\n\n /**\n * Click event handler\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n handleClick(event) {\n const $target = event.target\n if ($target && this.focusTarget($target)) {\n event.preventDefault()\n }\n }\n\n /**\n * Focus the target element\n *\n * By default, the browser will scroll the target into view. Because our\n * labels or legends appear above the input, this means the user will be\n * presented with an input without any context, as the label or legend will be\n * off the top of the screen.\n *\n * Manually handling the click event, scrolling the question into view and\n * then focussing the element solves this.\n *\n * This also results in the label and/or legend being announced correctly in\n * NVDA (as tested in 2018.3.2) - without this only the field type is\n * announced (e.g. \"Edit, has autocomplete\").\n *\n * @private\n * @param {EventTarget} $target - Event target\n * @returns {boolean} True if the target was able to be focussed\n */\n focusTarget($target) {\n // If the element that was clicked was not a link, return early\n if (!($target instanceof HTMLAnchorElement)) {\n return false\n }\n\n const inputId = getFragmentFromUrl($target.href)\n if (!inputId) {\n return false\n }\n\n const $input = document.getElementById(inputId)\n if (!$input) {\n return false\n }\n\n const $legendOrLabel = this.getAssociatedLegendOrLabel($input)\n if (!$legendOrLabel) {\n return false\n }\n\n // Scroll the legend or label into view *before* calling focus on the input\n // to avoid extra scrolling in browsers that don't support `preventScroll`\n // (which at time of writing is most of them...)\n $legendOrLabel.scrollIntoView()\n $input.focus({ preventScroll: true })\n\n return true\n }\n\n /**\n * Get associated legend or label\n *\n * Returns the first element that exists from this list:\n *\n * - The `<legend>` associated with the closest `<fieldset>` ancestor, as long\n * as the top of it is no more than half a viewport height away from the\n * bottom of the input\n * - The first `<label>` that is associated with the input using for=\"inputId\"\n * - The closest parent `<label>`\n *\n * @private\n * @param {Element} $input - The input\n * @returns {Element | null} Associated legend or label, or null if no\n * associated legend or label can be found\n */\n getAssociatedLegendOrLabel($input) {\n const $fieldset = $input.closest('fieldset')\n\n if ($fieldset) {\n const $legends = $fieldset.getElementsByTagName('legend')\n\n if ($legends.length) {\n const $candidateLegend = $legends[0]\n\n // If the input type is radio or checkbox, always use the legend if\n // there is one.\n if (\n $input instanceof HTMLInputElement &&\n ($input.type === 'checkbox' || $input.type === 'radio')\n ) {\n return $candidateLegend\n }\n\n // For other input types, only scroll to the fieldset’s legend (instead\n // of the label associated with the input) if the input would end up in\n // the top half of the screen.\n //\n // This should avoid situations where the input either ends up off the\n // screen, or obscured by a software keyboard.\n const legendTop = $candidateLegend.getBoundingClientRect().top\n const inputRect = $input.getBoundingClientRect()\n\n // If the browser doesn't support Element.getBoundingClientRect().height\n // or window.innerHeight (like IE8), bail and just link to the label.\n if (inputRect.height && window.innerHeight) {\n const inputBottom = inputRect.top + inputRect.height\n\n if (inputBottom - legendTop < window.innerHeight / 2) {\n return $candidateLegend\n }\n }\n }\n }\n\n return (\n document.querySelector(`label[for='${$input.getAttribute('id')}']`) ??\n $input.closest('label')\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-error-summary'\n\n /**\n * Error summary default config\n *\n * @see {@link ErrorSummaryConfig}\n * @constant\n * @type {ErrorSummaryConfig}\n */\n static defaults = Object.freeze({\n disableAutoFocus: false\n })\n\n /**\n * Error summary config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n disableAutoFocus: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Error summary config\n *\n * @typedef {object} ErrorSummaryConfig\n * @property {boolean} [disableAutoFocus=false] - If set to `true` the error\n * summary will not be focussed when the page loads.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Exit this page component\n *\n * @preserve\n */\nexport class ExitThisPage extends GOVUKFrontendComponent {\n /**\n * @private\n * @type {ExitThisPageConfig}\n */\n config\n\n /** @private */\n i18n\n\n /** @private */\n $button\n\n /**\n * @private\n * @type {HTMLAnchorElement | null}\n */\n $skiplinkButton = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $updateSpan = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $indicatorContainer = null\n\n /**\n * @private\n * @type {HTMLElement | null}\n */\n $overlay = null\n\n /** @private */\n keypressCounter = 0\n\n /** @private */\n lastKeyWasModified = false\n\n /** @private */\n timeoutTime = 5000 // milliseconds\n\n // Store the timeout events so that we can clear them to avoid user keypresses overlapping\n // setTimeout returns an id that we can use to clear it with clearTimeout,\n // hence the 'Id' suffix\n\n /**\n * @private\n * @type {number | null}\n */\n keypressTimeoutId = null\n\n /**\n * @private\n * @type {number | null}\n */\n timeoutMessageId = null\n\n /**\n * @param {Element | null} $root - HTML element that wraps the Exit This Page button\n * @param {ExitThisPageConfig} [config] - Exit This Page config\n */\n constructor($root, config = {}) {\n super($root)\n\n const $button = this.$root.querySelector('.govuk-exit-this-page__button')\n if (!($button instanceof HTMLAnchorElement)) {\n throw new ElementError({\n component: ExitThisPage,\n element: $button,\n expectedType: 'HTMLAnchorElement',\n identifier: 'Button (`.govuk-exit-this-page__button`)'\n })\n }\n\n this.config = mergeConfigs(\n ExitThisPage.defaults,\n config,\n normaliseDataset(ExitThisPage, this.$root.dataset)\n )\n\n this.i18n = new I18n(this.config.i18n)\n this.$button = $button\n\n const $skiplinkButton = document.querySelector(\n '.govuk-js-exit-this-page-skiplink'\n )\n if ($skiplinkButton instanceof HTMLAnchorElement) {\n this.$skiplinkButton = $skiplinkButton\n }\n\n this.buildIndicator()\n this.initUpdateSpan()\n this.initButtonClickHandler()\n\n // Check to see if this has already been done by a previous initialisation of ExitThisPage\n if (!('govukFrontendExitThisPageKeypress' in document.body.dataset)) {\n document.addEventListener('keyup', this.handleKeypress.bind(this), true)\n document.body.dataset.govukFrontendExitThisPageKeypress = 'true'\n }\n\n // When the page is restored after navigating 'back' in some browsers the\n // blank overlay remains present, rendering the page unusable. Here, we check\n // to see if it's present on page (re)load, and remove it if so.\n window.addEventListener('pageshow', this.resetPage.bind(this))\n }\n\n /**\n * Create the <span> we use for screen reader announcements.\n *\n * @private\n */\n initUpdateSpan() {\n this.$updateSpan = document.createElement('span')\n this.$updateSpan.setAttribute('role', 'status')\n this.$updateSpan.className = 'govuk-visually-hidden'\n\n this.$root.appendChild(this.$updateSpan)\n }\n\n /**\n * Create button click handlers.\n *\n * @private\n */\n initButtonClickHandler() {\n // Main EtP button\n this.$button.addEventListener('click', this.handleClick.bind(this))\n\n // EtP secondary link\n if (this.$skiplinkButton) {\n this.$skiplinkButton.addEventListener(\n 'click',\n this.handleClick.bind(this)\n )\n }\n }\n\n /**\n * Create the HTML for the 'three lights' indicator on the button.\n *\n * @private\n */\n buildIndicator() {\n // Build container\n // Putting `aria-hidden` on it as it won't contain any readable information\n this.$indicatorContainer = document.createElement('div')\n this.$indicatorContainer.className = 'govuk-exit-this-page__indicator'\n this.$indicatorContainer.setAttribute('aria-hidden', 'true')\n\n // Create three 'lights' and place them within the container\n for (let i = 0; i < 3; i++) {\n const $indicator = document.createElement('div')\n $indicator.className = 'govuk-exit-this-page__indicator-light'\n this.$indicatorContainer.appendChild($indicator)\n }\n\n // Append it all to the module\n this.$button.appendChild(this.$indicatorContainer)\n }\n\n /**\n * Update whether the lights are visible and which ones are lit up depending on\n * the value of `keypressCounter`.\n *\n * @private\n */\n updateIndicator() {\n if (!this.$indicatorContainer) {\n return\n }\n\n // Show or hide the indicator container depending on keypressCounter value\n this.$indicatorContainer.classList.toggle(\n 'govuk-exit-this-page__indicator--visible',\n this.keypressCounter > 0\n )\n\n // Turn on only the indicators we want on\n const $indicators = this.$indicatorContainer.querySelectorAll(\n '.govuk-exit-this-page__indicator-light'\n )\n $indicators.forEach(($indicator, index) => {\n $indicator.classList.toggle(\n 'govuk-exit-this-page__indicator-light--on',\n index < this.keypressCounter\n )\n })\n }\n\n /**\n * Initiates the redirection away from the current page.\n * Includes the loading overlay functionality, which covers the current page with a\n * white overlay so that the contents are not visible during the loading\n * process. This is particularly important on slow network connections.\n *\n * @private\n */\n exitPage() {\n if (!this.$updateSpan) {\n return\n }\n\n this.$updateSpan.textContent = ''\n\n // Blank the page\n // As well as creating an overlay with text, we also set the body to hidden\n // to prevent screen reader and sequential navigation users potentially\n // navigating through the page behind the overlay during loading\n document.body.classList.add('govuk-exit-this-page-hide-content')\n this.$overlay = document.createElement('div')\n this.$overlay.className = 'govuk-exit-this-page-overlay'\n this.$overlay.setAttribute('role', 'alert')\n\n // we do these this way round, thus incurring a second paint, because changing\n // the element text after adding it means that screen readers pick up the\n // announcement more reliably.\n document.body.appendChild(this.$overlay)\n this.$overlay.textContent = this.i18n.t('activated')\n\n window.location.href = this.$button.href\n }\n\n /**\n * Pre-activation logic for when the button is clicked/activated via mouse or\n * pointer.\n *\n * We do this to differentiate it from the keyboard activation event because we\n * need to run `e.preventDefault` as the button or skiplink are both links and we\n * want to apply some additional logic in `exitPage` before navigating.\n *\n * @private\n * @param {MouseEvent} event - mouse click event\n */\n handleClick(event) {\n event.preventDefault()\n this.exitPage()\n }\n\n /**\n * Logic for the 'quick escape' keyboard sequence functionality (pressing the\n * Shift key three times without interruption, within a time limit).\n *\n * @private\n * @param {KeyboardEvent} event - keyup event\n */\n handleKeypress(event) {\n if (!this.$updateSpan) {\n return\n }\n\n // Detect if the 'Shift' key has been pressed. We want to only do things if it\n // was pressed by itself and not in a combination with another key—so we keep\n // track of whether the preceding keyup had shiftKey: true on it, and if it\n // did, we ignore the next Shift keyup event.\n //\n // This works because using Shift as a modifier key (e.g. pressing Shift + A)\n // will fire TWO keyup events, one for A (with e.shiftKey: true) and the other\n // for Shift (with e.shiftKey: false).\n if (event.key === 'Shift' && !this.lastKeyWasModified) {\n this.keypressCounter += 1\n\n // Update the indicator before the below if statement can reset it back to 0\n this.updateIndicator()\n\n // Clear the timeout for the keypress timeout message clearing itself\n if (this.timeoutMessageId) {\n window.clearTimeout(this.timeoutMessageId)\n this.timeoutMessageId = null\n }\n\n if (this.keypressCounter >= 3) {\n this.keypressCounter = 0\n\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n this.keypressTimeoutId = null\n }\n\n this.exitPage()\n } else {\n if (this.keypressCounter === 1) {\n this.$updateSpan.textContent = this.i18n.t('pressTwoMoreTimes')\n } else {\n this.$updateSpan.textContent = this.i18n.t('pressOneMoreTime')\n }\n }\n\n this.setKeypressTimer()\n } else if (this.keypressTimeoutId) {\n // If the user pressed any key other than 'Shift', after having pressed\n // 'Shift' and activating the timer, stop and reset the timer.\n this.resetKeypressTimer()\n }\n\n // Keep track of whether the Shift modifier key was held during this keypress\n this.lastKeyWasModified = event.shiftKey\n }\n\n /**\n * Starts the 'quick escape' keyboard sequence timer.\n *\n * This can be invoked several times. We want this to be possible so that the\n * timer is restarted each time the shortcut key is pressed (e.g. the user has\n * up to n seconds between each keypress, rather than n seconds to invoke the\n * entire sequence.)\n *\n * @private\n */\n setKeypressTimer() {\n // Clear any existing timeout. This is so only one timer is running even if\n // there are multiple keypresses in quick succession.\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n }\n\n // Set a fresh timeout\n this.keypressTimeoutId = window.setTimeout(\n this.resetKeypressTimer.bind(this),\n this.timeoutTime\n )\n }\n\n /**\n * Stops and resets the 'quick escape' keyboard sequence timer.\n *\n * @private\n */\n resetKeypressTimer() {\n if (!this.$updateSpan) {\n return\n }\n\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n this.keypressTimeoutId = null\n }\n\n const $updateSpan = this.$updateSpan\n\n this.keypressCounter = 0\n $updateSpan.textContent = this.i18n.t('timedOut')\n\n this.timeoutMessageId = window.setTimeout(() => {\n $updateSpan.textContent = ''\n }, this.timeoutTime)\n\n this.updateIndicator()\n }\n\n /**\n * Reset the page using the EtP button\n *\n * We use this in situations where a user may re-enter a page using the browser\n * back button. In these cases, the browser can choose to restore the state of\n * the page as it was previously, including restoring the 'ghost page' overlay,\n * the announcement span having it's role set to \"alert\" and the keypress\n * indicator still active, leaving the page in an unusable state.\n *\n * By running this check when the page is shown, we can programatically restore\n * the page and the component to a \"default\" state\n *\n * @private\n */\n resetPage() {\n // If an overlay is set, remove it and reset the value\n document.body.classList.remove('govuk-exit-this-page-hide-content')\n\n if (this.$overlay) {\n this.$overlay.remove()\n this.$overlay = null\n }\n\n // Ensure the announcement span's role is status, not alert and clear any text\n if (this.$updateSpan) {\n this.$updateSpan.setAttribute('role', 'status')\n this.$updateSpan.textContent = ''\n }\n\n // Sync the keypress indicator lights\n this.updateIndicator()\n\n // If the timeouts are active, clear them\n if (this.keypressTimeoutId) {\n window.clearTimeout(this.keypressTimeoutId)\n }\n\n if (this.timeoutMessageId) {\n window.clearTimeout(this.timeoutMessageId)\n }\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-exit-this-page'\n\n /**\n * Exit this page default config\n *\n * @see {@link ExitThisPageConfig}\n * @constant\n * @type {ExitThisPageConfig}\n */\n static defaults = Object.freeze({\n i18n: {\n activated: 'Loading.',\n timedOut: 'Exit this page expired.',\n pressTwoMoreTimes: 'Shift, press 2 more times to exit.',\n pressOneMoreTime: 'Shift, press 1 more time to exit.'\n }\n })\n\n /**\n * Exit this page config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' }\n }\n })\n}\n\n/**\n * Exit this Page config\n *\n * @see {@link ExitThisPage.defaults}\n * @typedef {object} ExitThisPageConfig\n * @property {ExitThisPageTranslations} [i18n=ExitThisPage.defaults.i18n] - Exit this page translations\n */\n\n/**\n * Exit this Page translations\n *\n * @see {@link ExitThisPage.defaults.i18n}\n * @typedef {object} ExitThisPageTranslations\n *\n * Messages used by the component programatically inserted text, including\n * overlay text and screen reader announcements.\n * @property {string} [activated] - Screen reader announcement for when EtP\n * keypress functionality has been successfully activated.\n * @property {string} [timedOut] - Screen reader announcement for when the EtP\n * keypress functionality has timed out.\n * @property {string} [pressTwoMoreTimes] - Screen reader announcement informing\n * the user they must press the activation key two more times.\n * @property {string} [pressOneMoreTime] - Screen reader announcement informing\n * the user they must press the activation key one more time.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { getBreakpoint } from '../../common/index.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Header component\n *\n * @preserve\n */\nexport class Header extends GOVUKFrontendComponent {\n /** @private */\n $menuButton\n\n /** @private */\n $menu\n\n /**\n * Save the opened/closed state for the nav in memory so that we can\n * accurately maintain state when the screen is changed from small to big and\n * back to small\n *\n * @private\n */\n menuIsOpen = false\n\n /**\n * A global const for storing a matchMedia instance which we'll use to detect\n * when a screen size change happens. We rely on it being null if the feature\n * isn't available to initially apply hidden attributes\n *\n * @private\n * @type {MediaQueryList | null}\n */\n mql = null\n\n /**\n * Apply a matchMedia for desktop which will trigger a state sync if the\n * browser viewport moves between states.\n *\n * @param {Element | null} $root - HTML element to use for header\n */\n constructor($root) {\n super($root)\n\n const $menuButton = this.$root.querySelector('.govuk-js-header-toggle')\n\n // Headers don't necessarily have a navigation. When they don't, the menu\n // toggle won't be rendered by our macro (or may be omitted when writing\n // plain HTML)\n if (!$menuButton) {\n return this\n }\n\n const menuId = $menuButton.getAttribute('aria-controls')\n if (!menuId) {\n throw new ElementError({\n component: Header,\n identifier:\n 'Navigation button (`<button class=\"govuk-js-header-toggle\">`) attribute (`aria-controls`)'\n })\n }\n\n const $menu = document.getElementById(menuId)\n if (!$menu) {\n throw new ElementError({\n component: Header,\n element: $menu,\n identifier: `Navigation (\\`<ul id=\"${menuId}\">\\`)`\n })\n }\n\n this.$menu = $menu\n this.$menuButton = $menuButton\n\n this.setupResponsiveChecks()\n\n this.$menuButton.addEventListener('click', () =>\n this.handleMenuButtonClick()\n )\n }\n\n /**\n * Setup viewport resize check\n *\n * @private\n */\n setupResponsiveChecks() {\n const breakpoint = getBreakpoint('desktop')\n\n if (!breakpoint.value) {\n throw new ElementError({\n component: Header,\n identifier: `CSS custom property (\\`${breakpoint.property}\\`) on pseudo-class \\`:root\\``\n })\n }\n\n // Media query list for GOV.UK Frontend desktop breakpoint\n this.mql = window.matchMedia(`(min-width: ${breakpoint.value})`)\n\n // MediaQueryList.addEventListener isn't supported by Safari < 14 so we need\n // to be able to fall back to the deprecated MediaQueryList.addListener\n if ('addEventListener' in this.mql) {\n this.mql.addEventListener('change', () => this.checkMode())\n } else {\n // @ts-expect-error Property 'addListener' does not exist\n // eslint-disable-next-line @typescript-eslint/no-unsafe-call\n this.mql.addListener(() => this.checkMode())\n }\n\n this.checkMode()\n }\n\n /**\n * Sync menu state\n *\n * Uses the global variable menuIsOpen to correctly set the accessible and\n * visual states of the menu and the menu button.\n * Additionally will force the menu to be visible and the menu button to be\n * hidden if the matchMedia is triggered to desktop.\n *\n * @private\n */\n checkMode() {\n if (!this.mql || !this.$menu || !this.$menuButton) {\n return\n }\n\n if (this.mql.matches) {\n this.$menu.removeAttribute('hidden')\n this.$menuButton.setAttribute('hidden', '')\n } else {\n this.$menuButton.removeAttribute('hidden')\n this.$menuButton.setAttribute('aria-expanded', this.menuIsOpen.toString())\n\n if (this.menuIsOpen) {\n this.$menu.removeAttribute('hidden')\n } else {\n this.$menu.setAttribute('hidden', '')\n }\n }\n }\n\n /**\n * Handle menu button click\n *\n * When the menu button is clicked, change the visibility of the menu and then\n * sync the accessibility state and menu button state\n *\n * @private\n */\n handleMenuButtonClick() {\n this.menuIsOpen = !this.menuIsOpen\n this.checkMode()\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-header'\n}\n","import { mergeConfigs, setFocus } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Notification Banner component\n *\n * @preserve\n */\nexport class NotificationBanner extends GOVUKFrontendComponent {\n /**\n * @private\n * @type {NotificationBannerConfig}\n */\n config\n\n /**\n * @param {Element | null} $root - HTML element to use for notification banner\n * @param {NotificationBannerConfig} [config] - Notification banner config\n */\n constructor($root, config = {}) {\n super($root)\n\n this.config = mergeConfigs(\n NotificationBanner.defaults,\n config,\n normaliseDataset(NotificationBanner, this.$root.dataset)\n )\n\n /**\n * Focus the notification banner\n *\n * If `role=\"alert\"` is set, focus the element to help some assistive\n * technologies prioritise announcing it.\n *\n * You can turn off the auto-focus functionality by setting\n * `data-disable-auto-focus=\"true\"` in the component HTML. You might wish to\n * do this based on user research findings, or to avoid a clash with another\n * element which should be focused when the page loads.\n */\n if (\n this.$root.getAttribute('role') === 'alert' &&\n !this.config.disableAutoFocus\n ) {\n setFocus(this.$root)\n }\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-notification-banner'\n\n /**\n * Notification banner default config\n *\n * @see {@link NotificationBannerConfig}\n * @constant\n * @type {NotificationBannerConfig}\n */\n static defaults = Object.freeze({\n disableAutoFocus: false\n })\n\n /**\n * Notification banner config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n disableAutoFocus: { type: 'boolean' }\n }\n })\n}\n\n/**\n * Notification banner config\n *\n * @typedef {object} NotificationBannerConfig\n * @property {boolean} [disableAutoFocus=false] - If set to `true` the\n * notification banner will not be focussed when the page loads. This only\n * applies if the component has a `role` of `alert` – in other cases the\n * component will not be focused on page load, regardless of this option.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n */\n","import { closestAttributeValue } from '../../common/closest-attribute-value.mjs'\nimport { mergeConfigs } from '../../common/index.mjs'\nimport { normaliseDataset } from '../../common/normalise-dataset.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\nimport { I18n } from '../../i18n.mjs'\n\n/**\n * Password input component\n *\n * @preserve\n */\nexport class PasswordInput extends GOVUKFrontendComponent {\n /**\n * @private\n * @type {PasswordInputConfig}\n */\n config\n\n /** @private */\n i18n\n\n /**\n * @private\n * @type {HTMLInputElement}\n */\n $input\n\n /**\n * @private\n * @type {HTMLButtonElement}\n */\n $showHideButton\n\n /** @private */\n $screenReaderStatusMessage\n\n /**\n * @param {Element | null} $root - HTML element to use for password input\n * @param {PasswordInputConfig} [config] - Password input config\n */\n constructor($root, config = {}) {\n super($root)\n\n const $input = this.$root.querySelector('.govuk-js-password-input-input')\n if (!($input instanceof HTMLInputElement)) {\n throw new ElementError({\n component: PasswordInput,\n element: $input,\n expectedType: 'HTMLInputElement',\n identifier: 'Form field (`.govuk-js-password-input-input`)'\n })\n }\n\n if ($input.type !== 'password') {\n throw new ElementError(\n 'Password input: Form field (`.govuk-js-password-input-input`) must be of type `password`.'\n )\n }\n\n const $showHideButton = this.$root.querySelector(\n '.govuk-js-password-input-toggle'\n )\n if (!($showHideButton instanceof HTMLButtonElement)) {\n throw new ElementError({\n component: PasswordInput,\n element: $showHideButton,\n expectedType: 'HTMLButtonElement',\n identifier: 'Button (`.govuk-js-password-input-toggle`)'\n })\n }\n\n if ($showHideButton.type !== 'button') {\n throw new ElementError(\n 'Password input: Button (`.govuk-js-password-input-toggle`) must be of type `button`.'\n )\n }\n\n this.$input = $input\n this.$showHideButton = $showHideButton\n\n this.config = mergeConfigs(\n PasswordInput.defaults,\n config,\n normaliseDataset(PasswordInput, this.$root.dataset)\n )\n\n this.i18n = new I18n(this.config.i18n, {\n // Read the fallback if necessary rather than have it set in the defaults\n locale: closestAttributeValue(this.$root, 'lang')\n })\n\n // Show the toggle button element\n this.$showHideButton.removeAttribute('hidden')\n\n // Create and append the status text for screen readers.\n // This is injected between the input and button so that users get a sensible reading order if\n // moving through the page content linearly:\n // [password input] -> [your password is visible/hidden] -> [show/hide password]\n const $screenReaderStatusMessage = document.createElement('div')\n $screenReaderStatusMessage.className =\n 'govuk-password-input__sr-status govuk-visually-hidden'\n $screenReaderStatusMessage.setAttribute('aria-live', 'polite')\n this.$screenReaderStatusMessage = $screenReaderStatusMessage\n this.$input.insertAdjacentElement('afterend', $screenReaderStatusMessage)\n\n // Bind toggle button\n this.$showHideButton.addEventListener('click', this.toggle.bind(this))\n\n // Bind event to revert the password visibility to hidden\n if (this.$input.form) {\n this.$input.form.addEventListener('submit', () => this.hide())\n }\n\n // If the page is restored from bfcache and the password is visible, hide it again\n window.addEventListener('pageshow', (event) => {\n if (event.persisted && this.$input.type !== 'password') {\n this.hide()\n }\n })\n\n // Default the component to having the password hidden.\n this.hide()\n }\n\n /**\n * Toggle the visibility of the password input\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n toggle(event) {\n event.preventDefault()\n\n // If on this click, the field is type=\"password\", show the value\n if (this.$input.type === 'password') {\n this.show()\n return\n }\n\n // Otherwise, hide it\n // Being defensive - hiding should always be the default\n this.hide()\n }\n\n /**\n * Show the password input value in plain text.\n *\n * @private\n */\n show() {\n this.setType('text')\n }\n\n /**\n * Hide the password input value.\n *\n * @private\n */\n hide() {\n this.setType('password')\n }\n\n /**\n * Set the password input type\n *\n * @param {'text' | 'password'} type - Input type\n * @private\n */\n setType(type) {\n if (type === this.$input.type) {\n return\n }\n\n // Update input type\n this.$input.setAttribute('type', type)\n\n const isHidden = type === 'password'\n const prefixButton = isHidden ? 'show' : 'hide'\n const prefixStatus = isHidden ? 'passwordHidden' : 'passwordShown'\n\n // Update button text\n this.$showHideButton.innerText = this.i18n.t(`${prefixButton}Password`)\n\n // Update button aria-label\n this.$showHideButton.setAttribute(\n 'aria-label',\n this.i18n.t(`${prefixButton}PasswordAriaLabel`)\n )\n\n // Update status change text\n this.$screenReaderStatusMessage.innerText = this.i18n.t(\n `${prefixStatus}Announcement`\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-password-input'\n\n /**\n * Password input default config\n *\n * @see {@link PasswordInputConfig}\n * @constant\n * @default\n * @type {PasswordInputConfig}\n */\n static defaults = Object.freeze({\n i18n: {\n showPassword: 'Show',\n hidePassword: 'Hide',\n showPasswordAriaLabel: 'Show password',\n hidePasswordAriaLabel: 'Hide password',\n passwordShownAnnouncement: 'Your password is visible',\n passwordHiddenAnnouncement: 'Your password is hidden'\n }\n })\n\n /**\n * Password input config schema\n *\n * @constant\n * @satisfies {Schema}\n */\n static schema = Object.freeze({\n properties: {\n i18n: { type: 'object' }\n }\n })\n}\n\n/**\n * Password input config\n *\n * @typedef {object} PasswordInputConfig\n * @property {PasswordInputTranslations} [i18n=PasswordInput.defaults.i18n] - Password input translations\n */\n\n/**\n * Password input translations\n *\n * @see {@link PasswordInput.defaults.i18n}\n * @typedef {object} PasswordInputTranslations\n *\n * Messages displayed to the user indicating the state of the show/hide toggle.\n * @property {string} [showPassword] - Visible text of the button when the\n * password is currently hidden. Plain text only.\n * @property {string} [hidePassword] - Visible text of the button when the\n * password is currently visible. Plain text only.\n * @property {string} [showPasswordAriaLabel] - aria-label of the button when\n * the password is currently hidden. Plain text only.\n * @property {string} [hidePasswordAriaLabel] - aria-label of the button when\n * the password is currently visible. Plain text only.\n * @property {string} [passwordShownAnnouncement] - Screen reader\n * announcement to make when the password has just become visible.\n * Plain text only.\n * @property {string} [passwordHiddenAnnouncement] - Screen reader\n * announcement to make when the password has just been hidden.\n * Plain text only.\n */\n\n/**\n * @typedef {import('../../common/index.mjs').Schema} Schema\n * @typedef {import('../../i18n.mjs').TranslationPluralForms} TranslationPluralForms\n */\n","import { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Radios component\n *\n * @preserve\n */\nexport class Radios extends GOVUKFrontendComponent {\n /** @private */\n $inputs\n\n /**\n * Radios can be associated with a 'conditionally revealed' content block –\n * for example, a radio for 'Phone' could reveal an additional form field for\n * the user to enter their phone number.\n *\n * These associations are made using a `data-aria-controls` attribute, which\n * is promoted to an aria-controls attribute during initialisation.\n *\n * We also need to restore the state of any conditional reveals on the page\n * (for example if the user has navigated back), and set up event handlers to\n * keep the reveal in sync with the radio state.\n *\n * @param {Element | null} $root - HTML element to use for radios\n */\n constructor($root) {\n super($root)\n\n const $inputs = this.$root.querySelectorAll('input[type=\"radio\"]')\n if (!$inputs.length) {\n throw new ElementError({\n component: Radios,\n identifier: 'Form inputs (`<input type=\"radio\">`)'\n })\n }\n\n this.$inputs = $inputs\n\n this.$inputs.forEach(($input) => {\n const targetId = $input.getAttribute('data-aria-controls')\n\n // Skip radios without data-aria-controls attributes\n if (!targetId) {\n return\n }\n\n // Throw if target conditional element does not exist.\n if (!document.getElementById(targetId)) {\n throw new ElementError({\n component: Radios,\n identifier: `Conditional reveal (\\`id=\"${targetId}\"\\`)`\n })\n }\n\n // Promote the data-aria-controls attribute to a aria-controls attribute\n // so that the relationship is exposed in the AOM\n $input.setAttribute('aria-controls', targetId)\n $input.removeAttribute('data-aria-controls')\n })\n\n // When the page is restored after navigating 'back' in some browsers the\n // state of form controls is not restored until *after* the DOMContentLoaded\n // event is fired, so we need to sync after the pageshow event.\n window.addEventListener('pageshow', () => this.syncAllConditionalReveals())\n\n // Although we've set up handlers to sync state on the pageshow event, init\n // could be called after those events have fired, for example if they are\n // added to the page dynamically, so sync now too.\n this.syncAllConditionalReveals()\n\n // Handle events\n this.$root.addEventListener('click', (event) => this.handleClick(event))\n }\n\n /**\n * Sync the conditional reveal states for all radio buttons in this component.\n *\n * @private\n */\n syncAllConditionalReveals() {\n this.$inputs.forEach(($input) =>\n this.syncConditionalRevealWithInputState($input)\n )\n }\n\n /**\n * Sync conditional reveal with the input state\n *\n * Synchronise the visibility of the conditional reveal, and its accessible\n * state, with the input's checked state.\n *\n * @private\n * @param {HTMLInputElement} $input - Radio input\n */\n syncConditionalRevealWithInputState($input) {\n const targetId = $input.getAttribute('aria-controls')\n if (!targetId) {\n return\n }\n\n const $target = document.getElementById(targetId)\n if ($target?.classList.contains('govuk-radios__conditional')) {\n const inputIsChecked = $input.checked\n\n $input.setAttribute('aria-expanded', inputIsChecked.toString())\n $target.classList.toggle(\n 'govuk-radios__conditional--hidden',\n !inputIsChecked\n )\n }\n }\n\n /**\n * Click event handler\n *\n * Handle a click within the component root – if the click occurred on a radio, sync\n * the state of the conditional reveal for all radio buttons in the same form\n * with the same name (because checking one radio could have un-checked a\n * radio under the root of another Radio component)\n *\n * @private\n * @param {MouseEvent} event - Click event\n */\n handleClick(event) {\n const $clickedInput = event.target\n\n // Ignore clicks on things that aren't radio buttons\n if (\n !($clickedInput instanceof HTMLInputElement) ||\n $clickedInput.type !== 'radio'\n ) {\n return\n }\n\n // We only need to consider radios with conditional reveals, which will have\n // aria-controls attributes.\n const $allInputs = document.querySelectorAll(\n 'input[type=\"radio\"][aria-controls]'\n )\n\n const $clickedInputForm = $clickedInput.form\n const $clickedInputName = $clickedInput.name\n\n $allInputs.forEach(($input) => {\n const hasSameFormOwner = $input.form === $clickedInputForm\n const hasSameName = $input.name === $clickedInputName\n\n if (hasSameName && hasSameFormOwner) {\n this.syncConditionalRevealWithInputState($input)\n }\n })\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-radios'\n}\n","import { getBreakpoint } from '../../common/index.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Service Navigation component\n *\n * @preserve\n */\nexport class ServiceNavigation extends GOVUKFrontendComponent {\n /** @private */\n $menuButton\n\n /** @private */\n $menu\n\n /**\n * Remember the open/closed state of the nav so we can maintain it when the\n * screen is resized.\n *\n * @private\n */\n menuIsOpen = false\n\n /**\n * A global const for storing a matchMedia instance which we'll use to detect\n * when a screen size change happens. We rely on it being null if the feature\n * isn't available to initially apply hidden attributes\n *\n * @private\n * @type {MediaQueryList | null}\n */\n mql = null\n\n /**\n * @param {Element | null} $root - HTML element to use for header\n */\n constructor($root) {\n super($root)\n\n const $menuButton = this.$root.querySelector(\n '.govuk-js-service-navigation-toggle'\n )\n\n // Headers don't necessarily have a navigation. When they don't, the menu\n // toggle won't be rendered by our macro (or may be omitted when writing\n // plain HTML)\n if (!$menuButton) {\n return this\n }\n\n const menuId = $menuButton.getAttribute('aria-controls')\n if (!menuId) {\n throw new ElementError({\n component: ServiceNavigation,\n identifier:\n 'Navigation button (`<button class=\"govuk-js-service-navigation-toggle\">`) attribute (`aria-controls`)'\n })\n }\n\n const $menu = document.getElementById(menuId)\n if (!$menu) {\n throw new ElementError({\n component: ServiceNavigation,\n element: $menu,\n identifier: `Navigation (\\`<ul id=\"${menuId}\">\\`)`\n })\n }\n\n this.$menu = $menu\n this.$menuButton = $menuButton\n\n this.setupResponsiveChecks()\n\n this.$menuButton.addEventListener('click', () =>\n this.handleMenuButtonClick()\n )\n }\n\n /**\n * Setup viewport resize check\n *\n * @private\n */\n setupResponsiveChecks() {\n const breakpoint = getBreakpoint('tablet')\n\n if (!breakpoint.value) {\n throw new ElementError({\n component: ServiceNavigation,\n identifier: `CSS custom property (\\`${breakpoint.property}\\`) on pseudo-class \\`:root\\``\n })\n }\n\n // Media query list for GOV.UK Frontend desktop breakpoint\n this.mql = window.matchMedia(`(min-width: ${breakpoint.value})`)\n\n // MediaQueryList.addEventListener isn't supported by Safari < 14 so we need\n // to be able to fall back to the deprecated MediaQueryList.addListener\n if ('addEventListener' in this.mql) {\n this.mql.addEventListener('change', () => this.checkMode())\n } else {\n // @ts-expect-error Property 'addListener' does not exist\n // eslint-disable-next-line @typescript-eslint/no-unsafe-call\n this.mql.addListener(() => this.checkMode())\n }\n\n this.checkMode()\n }\n\n /**\n * Sync menu state\n *\n * Uses the global variable menuIsOpen to correctly set the accessible and\n * visual states of the menu and the menu button.\n * Additionally will force the menu to be visible and the menu button to be\n * hidden if the matchMedia is triggered to desktop.\n *\n * @private\n */\n checkMode() {\n if (!this.mql || !this.$menu || !this.$menuButton) {\n return\n }\n\n if (this.mql.matches) {\n this.$menu.removeAttribute('hidden')\n this.$menuButton.setAttribute('hidden', '')\n } else {\n this.$menuButton.removeAttribute('hidden')\n this.$menuButton.setAttribute('aria-expanded', this.menuIsOpen.toString())\n\n if (this.menuIsOpen) {\n this.$menu.removeAttribute('hidden')\n } else {\n this.$menu.setAttribute('hidden', '')\n }\n }\n }\n\n /**\n * Handle menu button click\n *\n * When the menu button is clicked, change the visibility of the menu and then\n * sync the accessibility state and menu button state\n *\n * @private\n */\n handleMenuButtonClick() {\n this.menuIsOpen = !this.menuIsOpen\n this.checkMode()\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-service-navigation'\n}\n","import { getFragmentFromUrl, setFocus } from '../../common/index.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Skip link component\n *\n * @preserve\n * @augments GOVUKFrontendComponent<HTMLAnchorElement>\n */\nexport class SkipLink extends GOVUKFrontendComponent {\n static elementType = HTMLAnchorElement\n\n /**\n * @param {Element | null} $root - HTML element to use for skip link\n * @throws {ElementError} when $root is not set or the wrong type\n * @throws {ElementError} when $root.hash does not contain a hash\n * @throws {ElementError} when the linked element is missing or the wrong type\n */\n constructor($root) {\n super($root)\n\n const hash = this.$root.hash\n const href = this.$root.getAttribute('href') ?? ''\n\n /** @type {URL | undefined} */\n let url\n\n /**\n * Check for valid link URL\n *\n * {@link https://caniuse.com/url}\n * {@link https://url.spec.whatwg.org}\n *\n */\n try {\n url = new window.URL(this.$root.href)\n } catch (error) {\n throw new ElementError(\n `Skip link: Target link (\\`href=\"${href}\"\\`) is invalid`\n )\n }\n\n // Return early for external URLs or links to other pages\n if (\n url.origin !== window.location.origin ||\n url.pathname !== window.location.pathname\n ) {\n return\n }\n\n const linkedElementId = getFragmentFromUrl(hash)\n\n // Check link path matching current page\n if (!linkedElementId) {\n throw new ElementError(\n `Skip link: Target link (\\`href=\"${href}\"\\`) has no hash fragment`\n )\n }\n\n const $linkedElement = document.getElementById(linkedElementId)\n\n // Check for link target element\n if (!$linkedElement) {\n throw new ElementError({\n component: SkipLink,\n element: $linkedElement,\n identifier: `Target content (\\`id=\"${linkedElementId}\"\\`)`\n })\n }\n\n /**\n * Focus the linked element on click\n *\n * Adds a helper CSS class to hide native focus styles,\n * but removes it on blur to restore native focus styles\n */\n this.$root.addEventListener('click', () =>\n setFocus($linkedElement, {\n onBeforeFocus() {\n $linkedElement.classList.add('govuk-skip-link-focused-element')\n },\n onBlur() {\n $linkedElement.classList.remove('govuk-skip-link-focused-element')\n }\n })\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-skip-link'\n}\n","import { getBreakpoint, getFragmentFromUrl } from '../../common/index.mjs'\nimport { ElementError } from '../../errors/index.mjs'\nimport { GOVUKFrontendComponent } from '../../govuk-frontend-component.mjs'\n\n/**\n * Tabs component\n *\n * @preserve\n */\nexport class Tabs extends GOVUKFrontendComponent {\n /** @private */\n $tabs\n\n /** @private */\n $tabList\n\n /** @private */\n $tabListItems\n\n /** @private */\n jsHiddenClass = 'govuk-tabs__panel--hidden'\n\n /** @private */\n changingHash = false\n\n /** @private */\n boundTabClick\n\n /** @private */\n boundTabKeydown\n\n /** @private */\n boundOnHashChange\n\n /**\n * @private\n * @type {MediaQueryList | null}\n */\n mql = null\n\n /**\n * @param {Element | null} $root - HTML element to use for tabs\n */\n constructor($root) {\n super($root)\n\n const $tabs = this.$root.querySelectorAll('a.govuk-tabs__tab')\n if (!$tabs.length) {\n throw new ElementError({\n component: Tabs,\n identifier: 'Links (`<a class=\"govuk-tabs__tab\">`)'\n })\n }\n\n this.$tabs = $tabs\n\n // Save bound functions so we can remove event listeners during teardown\n this.boundTabClick = this.onTabClick.bind(this)\n this.boundTabKeydown = this.onTabKeydown.bind(this)\n this.boundOnHashChange = this.onHashChange.bind(this)\n\n const $tabList = this.$root.querySelector('.govuk-tabs__list')\n const $tabListItems = this.$root.querySelectorAll(\n 'li.govuk-tabs__list-item'\n )\n\n if (!$tabList) {\n throw new ElementError({\n component: Tabs,\n identifier: 'List (`<ul class=\"govuk-tabs__list\">`)'\n })\n }\n\n if (!$tabListItems.length) {\n throw new ElementError({\n component: Tabs,\n identifier: 'List items (`<li class=\"govuk-tabs__list-item\">`)'\n })\n }\n\n this.$tabList = $tabList\n this.$tabListItems = $tabListItems\n\n this.setupResponsiveChecks()\n }\n\n /**\n * Setup viewport resize check\n *\n * @private\n */\n setupResponsiveChecks() {\n const breakpoint = getBreakpoint('tablet')\n\n if (!breakpoint.value) {\n throw new ElementError({\n component: Tabs,\n identifier: `CSS custom property (\\`${breakpoint.property}\\`) on pseudo-class \\`:root\\``\n })\n }\n\n // Media query list for GOV.UK Frontend tablet breakpoint\n this.mql = window.matchMedia(`(min-width: ${breakpoint.value})`)\n\n // MediaQueryList.addEventListener isn't supported by Safari < 14 so we need\n // to be able to fall back to the deprecated MediaQueryList.addListener\n if ('addEventListener' in this.mql) {\n this.mql.addEventListener('change', () => this.checkMode())\n } else {\n // @ts-expect-error Property 'addListener' does not exist\n // eslint-disable-next-line @typescript-eslint/no-unsafe-call\n this.mql.addListener(() => this.checkMode())\n }\n\n this.checkMode()\n }\n\n /**\n * Setup or teardown handler for viewport resize check\n *\n * @private\n */\n checkMode() {\n if (this.mql?.matches) {\n this.setup()\n } else {\n this.teardown()\n }\n }\n\n /**\n * Setup tab component\n *\n * @private\n */\n setup() {\n this.$tabList.setAttribute('role', 'tablist')\n\n this.$tabListItems.forEach(($item) => {\n $item.setAttribute('role', 'presentation')\n })\n\n this.$tabs.forEach(($tab) => {\n // Set HTML attributes\n this.setAttributes($tab)\n\n // Handle events\n $tab.addEventListener('click', this.boundTabClick, true)\n $tab.addEventListener('keydown', this.boundTabKeydown, true)\n\n // Remove old active panels\n this.hideTab($tab)\n })\n\n // Show either the active tab according to the URL's hash or the first tab\n const $activeTab = this.getTab(window.location.hash) ?? this.$tabs[0]\n\n this.showTab($activeTab)\n\n // Handle hashchange events\n window.addEventListener('hashchange', this.boundOnHashChange, true)\n }\n\n /**\n * Teardown tab component\n *\n * @private\n */\n teardown() {\n this.$tabList.removeAttribute('role')\n\n this.$tabListItems.forEach(($item) => {\n $item.removeAttribute('role')\n })\n\n this.$tabs.forEach(($tab) => {\n // Remove events\n $tab.removeEventListener('click', this.boundTabClick, true)\n $tab.removeEventListener('keydown', this.boundTabKeydown, true)\n\n // Unset HTML attributes\n this.unsetAttributes($tab)\n })\n\n // Remove hashchange event handler\n window.removeEventListener('hashchange', this.boundOnHashChange, true)\n }\n\n /**\n * Handle hashchange event\n *\n * @private\n * @returns {void | undefined} Returns void, or undefined when prevented\n */\n onHashChange() {\n const hash = window.location.hash\n const $tabWithHash = this.getTab(hash)\n if (!$tabWithHash) {\n return\n }\n\n // Prevent changing the hash\n if (this.changingHash) {\n this.changingHash = false\n return\n }\n\n // Show either the active tab according to the URL's hash or the first tab\n const $previousTab = this.getCurrentTab()\n if (!$previousTab) {\n return\n }\n\n this.hideTab($previousTab)\n this.showTab($tabWithHash)\n $tabWithHash.focus()\n }\n\n /**\n * Hide panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n hideTab($tab) {\n this.unhighlightTab($tab)\n this.hidePanel($tab)\n }\n\n /**\n * Show panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n showTab($tab) {\n this.highlightTab($tab)\n this.showPanel($tab)\n }\n\n /**\n * Get tab link by hash\n *\n * @private\n * @param {string} hash - Hash fragment including #\n * @returns {HTMLAnchorElement | null} Tab link\n */\n getTab(hash) {\n return this.$root.querySelector(`a.govuk-tabs__tab[href=\"${hash}\"]`)\n }\n\n /**\n * Set tab link and panel attributes\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n setAttributes($tab) {\n const panelId = getFragmentFromUrl($tab.href)\n if (!panelId) {\n return\n }\n\n // Set tab attributes\n $tab.setAttribute('id', `tab_${panelId}`)\n $tab.setAttribute('role', 'tab')\n $tab.setAttribute('aria-controls', panelId)\n $tab.setAttribute('aria-selected', 'false')\n $tab.setAttribute('tabindex', '-1')\n\n // Set panel attributes\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.setAttribute('role', 'tabpanel')\n $panel.setAttribute('aria-labelledby', $tab.id)\n $panel.classList.add(this.jsHiddenClass)\n }\n\n /**\n * Unset tab link and panel attributes\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n unsetAttributes($tab) {\n // unset tab attributes\n $tab.removeAttribute('id')\n $tab.removeAttribute('role')\n $tab.removeAttribute('aria-controls')\n $tab.removeAttribute('aria-selected')\n $tab.removeAttribute('tabindex')\n\n // unset panel attributes\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.removeAttribute('role')\n $panel.removeAttribute('aria-labelledby')\n $panel.classList.remove(this.jsHiddenClass)\n }\n\n /**\n * Handle tab link clicks\n *\n * @private\n * @param {MouseEvent} event - Mouse click event\n * @returns {void} Returns void\n */\n onTabClick(event) {\n const $currentTab = this.getCurrentTab()\n const $nextTab = event.currentTarget\n\n if (!$currentTab || !($nextTab instanceof HTMLAnchorElement)) {\n return\n }\n\n event.preventDefault()\n\n this.hideTab($currentTab)\n this.showTab($nextTab)\n this.createHistoryEntry($nextTab)\n }\n\n /**\n * Update browser URL hash fragment for tab\n *\n * - Allows back/forward to navigate tabs\n * - Avoids page jump when hash changes\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n createHistoryEntry($tab) {\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n // Save and restore the id so the page doesn't jump when a user clicks a tab\n // (which changes the hash)\n const panelId = $panel.id\n $panel.id = ''\n this.changingHash = true\n window.location.hash = panelId\n $panel.id = panelId\n }\n\n /**\n * Handle tab keydown event\n *\n * - Press right arrow for next tab\n * - Press left arrow for previous tab\n *\n * @private\n * @param {KeyboardEvent} event - Keydown event\n */\n onTabKeydown(event) {\n switch (event.key) {\n // 'Left' and 'Right' required for Edge 16 support.\n case 'ArrowLeft':\n case 'Left':\n this.activatePreviousTab()\n event.preventDefault()\n break\n case 'ArrowRight':\n case 'Right':\n this.activateNextTab()\n event.preventDefault()\n break\n }\n }\n\n /**\n * Activate next tab\n *\n * @private\n */\n activateNextTab() {\n const $currentTab = this.getCurrentTab()\n if (!$currentTab?.parentElement) {\n return\n }\n\n const $nextTabListItem = $currentTab.parentElement.nextElementSibling\n if (!$nextTabListItem) {\n return\n }\n\n const $nextTab = $nextTabListItem.querySelector('a.govuk-tabs__tab')\n if (!$nextTab) {\n return\n }\n\n this.hideTab($currentTab)\n this.showTab($nextTab)\n $nextTab.focus()\n this.createHistoryEntry($nextTab)\n }\n\n /**\n * Activate previous tab\n *\n * @private\n */\n activatePreviousTab() {\n const $currentTab = this.getCurrentTab()\n if (!$currentTab?.parentElement) {\n return\n }\n\n const $previousTabListItem =\n $currentTab.parentElement.previousElementSibling\n if (!$previousTabListItem) {\n return\n }\n\n const $previousTab = $previousTabListItem.querySelector('a.govuk-tabs__tab')\n if (!$previousTab) {\n return\n }\n\n this.hideTab($currentTab)\n this.showTab($previousTab)\n $previousTab.focus()\n this.createHistoryEntry($previousTab)\n }\n\n /**\n * Get tab panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n * @returns {Element | null} Tab panel\n */\n getPanel($tab) {\n const panelId = getFragmentFromUrl($tab.href)\n if (!panelId) {\n return null\n }\n\n return this.$root.querySelector(`#${panelId}`)\n }\n\n /**\n * Show tab panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n showPanel($tab) {\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.classList.remove(this.jsHiddenClass)\n }\n\n /**\n * Hide tab panel for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n hidePanel($tab) {\n const $panel = this.getPanel($tab)\n if (!$panel) {\n return\n }\n\n $panel.classList.add(this.jsHiddenClass)\n }\n\n /**\n * Unset 'selected' state for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n unhighlightTab($tab) {\n if (!$tab.parentElement) {\n return\n }\n\n $tab.setAttribute('aria-selected', 'false')\n $tab.parentElement.classList.remove('govuk-tabs__list-item--selected')\n $tab.setAttribute('tabindex', '-1')\n }\n\n /**\n * Set 'selected' state for tab link\n *\n * @private\n * @param {HTMLAnchorElement} $tab - Tab link\n */\n highlightTab($tab) {\n if (!$tab.parentElement) {\n return\n }\n\n $tab.setAttribute('aria-selected', 'true')\n $tab.parentElement.classList.add('govuk-tabs__list-item--selected')\n $tab.setAttribute('tabindex', '0')\n }\n\n /**\n * Get current tab link\n *\n * @private\n * @returns {HTMLAnchorElement | null} Tab link\n */\n getCurrentTab() {\n return this.$root.querySelector(\n '.govuk-tabs__list-item--selected a.govuk-tabs__tab'\n )\n }\n\n /**\n * Name for the component used when initialising using data-module attributes.\n */\n static moduleName = 'govuk-tabs'\n}\n","import { isSupported } from './common/index.mjs'\nimport { Accordion } from './components/accordion/accordion.mjs'\nimport { Button } from './components/button/button.mjs'\nimport { CharacterCount } from './components/character-count/character-count.mjs'\nimport { Checkboxes } from './components/checkboxes/checkboxes.mjs'\nimport { ErrorSummary } from './components/error-summary/error-summary.mjs'\nimport { ExitThisPage } from './components/exit-this-page/exit-this-page.mjs'\nimport { Header } from './components/header/header.mjs'\nimport { NotificationBanner } from './components/notification-banner/notification-banner.mjs'\nimport { PasswordInput } from './components/password-input/password-input.mjs'\nimport { Radios } from './components/radios/radios.mjs'\nimport { ServiceNavigation } from './components/service-navigation/service-navigation.mjs'\nimport { SkipLink } from './components/skip-link/skip-link.mjs'\nimport { Tabs } from './components/tabs/tabs.mjs'\nimport { SupportError } from './errors/index.mjs'\n\n/**\n * Initialise all components\n *\n * Use the `data-module` attributes to find, instantiate and init all of the\n * components provided as part of GOV.UK Frontend.\n *\n * @param {Config & { scope?: Element, onError?: OnErrorCallback<CompatibleClass> }} [config] - Config for all components (with optional scope)\n */\nfunction initAll(config) {\n config = typeof config !== 'undefined' ? config : {}\n\n // Skip initialisation when GOV.UK Frontend is not supported\n if (!isSupported()) {\n if (config.onError) {\n config.onError(new SupportError(), {\n config\n })\n } else {\n console.log(new SupportError())\n }\n return\n }\n\n const components = /** @type {const} */ ([\n [Accordion, config.accordion],\n [Button, config.button],\n [CharacterCount, config.characterCount],\n [Checkboxes],\n [ErrorSummary, config.errorSummary],\n [ExitThisPage, config.exitThisPage],\n [Header],\n [NotificationBanner, config.notificationBanner],\n [PasswordInput, config.passwordInput],\n [Radios],\n [ServiceNavigation],\n [SkipLink],\n [Tabs]\n ])\n\n // Allow the user to initialise GOV.UK Frontend in only certain sections of the page\n // Defaults to the entire document if nothing is set.\n // const $scope = config.scope ?? document\n\n const options = {\n scope: config.scope ?? document,\n onError: config.onError\n }\n\n components.forEach(([Component, config]) => {\n createAll(Component, config, options)\n })\n}\n\n/**\n * Create all instances of a specific component on the page\n *\n * Uses the `data-module` attribute to find all elements matching the specified\n * component on the page, creating instances of the component object for each\n * of them.\n *\n * Any component errors will be caught and logged to the console.\n *\n * @template {CompatibleClass} T\n * @param {T} Component - class of the component to create\n * @param {T[\"defaults\"]} [config] - Config supplied to component\n * @param {OnErrorCallback<T> | Element | Document | CreateAllOptions<T> } [createAllOptions] - options for createAll including scope of the document to search within and callback function if error throw by component on init\n * @returns {Array<InstanceType<T>>} - array of instantiated components\n */\nfunction createAll(Component, config, createAllOptions) {\n let /** @type {Element | Document} */ $scope = document\n let /** @type {OnErrorCallback<Component> | undefined} */ onError\n\n if (typeof createAllOptions === 'object') {\n createAllOptions = /** @type {CreateAllOptions<Component>} */ (\n // eslint-disable-next-line no-self-assign\n createAllOptions\n )\n\n $scope = createAllOptions.scope ?? $scope\n onError = createAllOptions.onError\n }\n\n if (typeof createAllOptions === 'function') {\n onError = createAllOptions\n }\n\n if (createAllOptions instanceof HTMLElement) {\n $scope = createAllOptions\n }\n\n const $elements = $scope.querySelectorAll(\n `[data-module=\"${Component.moduleName}\"]`\n )\n\n // Skip initialisation when GOV.UK Frontend is not supported\n if (!isSupported()) {\n if (onError) {\n onError(new SupportError(), {\n component: Component,\n config\n })\n } else {\n console.log(new SupportError())\n }\n return []\n }\n\n /* eslint-disable-next-line @typescript-eslint/no-unsafe-return --\n * We can't define CompatibleClass as `{new(): CompatibleClass, moduleName: string}`,\n * as when doing `typeof Accordion` (or any component), TypeScript doesn't seem\n * to acknowledge the static `moduleName` that's set in our component classes.\n * This means we have to set the constructor of `CompatibleClass` as `{new(): any}`,\n * leading to ESLint frowning that we're returning `any[]`.\n */\n return Array.from($elements)\n .map(($element) => {\n try {\n // Only pass config to components that accept it\n // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n return typeof config !== 'undefined'\n ? new Component($element, config)\n : new Component($element)\n } catch (error) {\n if (onError) {\n onError(error, {\n element: $element,\n component: Component,\n config\n })\n } else {\n console.log(error)\n }\n\n return null\n }\n })\n .filter(Boolean) // Exclude components that errored\n}\n\nexport { initAll, createAll }\n\n/* eslint-disable jsdoc/valid-types --\n * `{new(...args: any[] ): object}` is not recognised as valid\n * https://github.com/gajus/eslint-plugin-jsdoc/issues/145#issuecomment-1308722878\n * https://github.com/jsdoc-type-pratt-parser/jsdoc-type-pratt-parser/issues/131\n **/\n\n/**\n * @typedef {{new (...args: any[]): any, defaults?: object, moduleName: string}} CompatibleClass\n */\n\n/* eslint-enable jsdoc/valid-types */\n\n/**\n * Config for all components via `initAll()`\n *\n * @typedef {object} Config\n * @property {AccordionConfig} [accordion] - Accordion config\n * @property {ButtonConfig} [button] - Button config\n * @property {CharacterCountConfig} [characterCount] - Character Count config\n * @property {ErrorSummaryConfig} [errorSummary] - Error Summary config\n * @property {ExitThisPageConfig} [exitThisPage] - Exit This Page config\n * @property {NotificationBannerConfig} [notificationBanner] - Notification Banner config\n * @property {PasswordInputConfig} [passwordInput] - Password input config\n */\n\n/**\n * Config for individual components\n *\n * @typedef {import('./components/accordion/accordion.mjs').AccordionConfig} AccordionConfig\n * @typedef {import('./components/accordion/accordion.mjs').AccordionTranslations} AccordionTranslations\n * @typedef {import('./components/button/button.mjs').ButtonConfig} ButtonConfig\n * @typedef {import('./components/character-count/character-count.mjs').CharacterCountConfig} CharacterCountConfig\n * @typedef {import('./components/character-count/character-count.mjs').CharacterCountTranslations} CharacterCountTranslations\n * @typedef {import('./components/error-summary/error-summary.mjs').ErrorSummaryConfig} ErrorSummaryConfig\n * @typedef {import('./components/exit-this-page/exit-this-page.mjs').ExitThisPageConfig} ExitThisPageConfig\n * @typedef {import('./components/exit-this-page/exit-this-page.mjs').ExitThisPageTranslations} ExitThisPageTranslations\n * @typedef {import('./components/notification-banner/notification-banner.mjs').NotificationBannerConfig} NotificationBannerConfig\n * @typedef {import('./components/password-input/password-input.mjs').PasswordInputConfig} PasswordInputConfig\n */\n\n/**\n * Component config keys, e.g. `accordion` and `characterCount`\n *\n * @typedef {keyof Config} ConfigKey\n */\n\n/**\n * @template {CompatibleClass} T\n * @typedef {object} ErrorContext\n * @property {Element} [element] - Element used for component module initialisation\n * @property {T} [component] - Class of component\n * @property {T[\"defaults\"]} config - Config supplied to component\n */\n\n/**\n * @template {CompatibleClass} T\n * @callback OnErrorCallback\n * @param {unknown} error - Thrown error\n * @param {ErrorContext<T>} context - Object containing the element, component class and configuration\n */\n\n/**\n * @template {CompatibleClass} T\n * @typedef {object} CreateAllOptions\n * @property {Element | Document} [scope] - scope of the document to search within\n * @property {OnErrorCallback<T>} [onError] - callback function if error throw by component on init\n */\n"],"names":["version","normaliseString","value","property","trimmedValue","trim","output","outputType","type","includes","length","isFinite","Number","mergeConfigs","configObjects","formattedConfigObject","configObject","key","Object","keys","option","override","isObject","extractConfigByNamespace","Component","dataset","namespace","schema","properties","newObject","entries","current","keyParts","split","index","name","getFragmentFromUrl","url","pop","getBreakpoint","window","getComputedStyle","document","documentElement","getPropertyValue","undefined","setFocus","$element","options","_options$onBeforeFocu","isFocusable","getAttribute","onBlur","_options$onBlur","call","removeAttribute","setAttribute","addEventListener","once","onBeforeFocus","focus","isSupported","$scope","body","classList","contains","Array","isArray","formatErrorMessage","message","moduleName","normaliseDataset","out","field","GOVUKFrontendError","Error","constructor","args","super","this","SupportError","supportMessage","HTMLScriptElement","prototype","ConfigError","ElementError","messageOrOptions","component","identifier","element","expectedType","InitError","componentOrMessage","GOVUKFrontendComponent","$root","_$root","childConstructor","elementType","checkSupport","checkInitialised","HTMLElement","hasAttribute","isInitialised","I18n","translations","config","_config$locale","locale","lang","t","lookupKey","translation","count","translationPluralForm","getPluralSuffix","match","replacePlaceholders","translationString","formatter","Intl","NumberFormat","supportedLocalesOf","replace","placeholderWithBraces","placeholderKey","hasOwnProperty","placeholderValue","format","hasIntlPluralRulesSupport","Boolean","PluralRules","preferredForm","select","selectPluralFormUsingFallbackRules","console","warn","Math","abs","floor","ruleset","getPluralRulesForLocale","pluralRules","localeShort","pluralRule","pluralRulesMap","languages","arabic","chinese","french","german","irish","russian","scottish","spanish","welsh","n","lastTwo","last","Accordion","i18n","controlsClass","showAllClass","showAllTextClass","sectionClass","sectionExpandedClass","sectionButtonClass","sectionHeaderClass","sectionHeadingClass","sectionHeadingDividerClass","sectionHeadingTextClass","sectionHeadingTextFocusClass","sectionShowHideToggleClass","sectionShowHideToggleFocusClass","sectionShowHideTextClass","upChevronIconClass","downChevronIconClass","sectionSummaryClass","sectionSummaryFocusClass","sectionContentClass","$sections","$showAllButton","$showAllIcon","$showAllText","defaults","querySelectorAll","initControls","initSectionHeaders","updateShowAllButton","areAllSectionsOpen","createElement","add","appendChild","$accordionControls","insertBefore","firstChild","onShowOrHideAllToggle","event","onBeforeMatch","forEach","$section","i","$header","querySelector","constructHeaderMarkup","setExpanded","isExpanded","onSectionToggle","setInitialState","$span","$heading","$summary","$button","id","attr","from","attributes","$headingText","$headingTextFocus","childNodes","$child","$showHideToggle","$showHideToggleFocus","$showHideText","$showHideIcon","getButtonPunctuationEl","$summarySpan","$summarySpanFocus","remove","removeChild","$fragment","target","Element","closest","nowExpanded","storeState","expanded","$content","newButtonText","textContent","ariaLabelParts","push","ariaLabelMessage","join","every","toString","toggle","getIdentifier","rememberExpanded","sessionStorage","setItem","exception","state","getItem","$punctuationEl","freeze","hideAllSections","hideSection","hideSectionAriaLabel","showAllSections","showSection","showSectionAriaLabel","Button","debounceFormSubmitTimer","handleKeyDown","debounce","$target","preventDefault","click","preventDoubleClick","setTimeout","DEBOUNCE_TIMEOUT_IN_SECONDS","closestAttributeValue","attributeName","$closestElementWithAttribute","CharacterCount","_ref","_this$config$maxwords","$textarea","$visibleCountMessage","$screenReaderCountMessage","lastInputTimestamp","lastInputValue","valueChecker","maxLength","HTMLTextAreaElement","HTMLInputElement","datasetConfig","configOverrides","maxlength","maxwords","errors","validationErrors","conditions","required","errorMessage","validateConfig","Infinity","textareaDescriptionId","$textareaDescription","getElementById","insertAdjacentElement","className","bindChangeEvents","updateCountMessage","handleKeyUp","handleFocus","handleBlur","updateVisibleCountMessage","Date","now","setInterval","updateIfValueChanged","clearInterval","updateScreenReaderCountMessage","isError","isOverThreshold","getCountMessage","text","_text$match","remainingNumber","countType","formatCountMessage","translationKeySuffix","threshold","currentLength","charactersUnderLimit","one","other","charactersAtLimit","charactersOverLimit","wordsUnderLimit","wordsAtLimit","wordsOverLimit","textareaDescription","anyOf","Checkboxes","$inputs","$input","targetId","syncAllConditionalReveals","handleClick","syncConditionalRevealWithInputState","inputIsChecked","checked","unCheckAllInputsExcept","$inputWithSameName","form","unCheckExclusiveInputs","$exclusiveInput","$clickedInput","ErrorSummary","disableAutoFocus","focusTarget","HTMLAnchorElement","inputId","href","$legendOrLabel","getAssociatedLegendOrLabel","scrollIntoView","preventScroll","_document$querySelect","$fieldset","$legends","getElementsByTagName","$candidateLegend","legendTop","getBoundingClientRect","top","inputRect","height","innerHeight","ExitThisPage","$skiplinkButton","$updateSpan","$indicatorContainer","$overlay","keypressCounter","lastKeyWasModified","timeoutTime","keypressTimeoutId","timeoutMessageId","buildIndicator","initUpdateSpan","initButtonClickHandler","handleKeypress","bind","govukFrontendExitThisPageKeypress","resetPage","$indicator","updateIndicator","exitPage","location","resetKeypressTimer","clearTimeout","setKeypressTimer","shiftKey","activated","timedOut","pressTwoMoreTimes","pressOneMoreTime","Header","$menuButton","$menu","menuIsOpen","mql","menuId","setupResponsiveChecks","handleMenuButtonClick","breakpoint","matchMedia","checkMode","addListener","matches","NotificationBanner","PasswordInput","$showHideButton","$screenReaderStatusMessage","HTMLButtonElement","hide","persisted","show","setType","isHidden","prefixButton","prefixStatus","innerText","showPassword","hidePassword","showPasswordAriaLabel","hidePasswordAriaLabel","passwordShownAnnouncement","passwordHiddenAnnouncement","Radios","$allInputs","$clickedInputForm","$clickedInputName","hasSameFormOwner","ServiceNavigation","SkipLink","_this$$root$getAttrib","hash","URL","error","origin","pathname","linkedElementId","$linkedElement","Tabs","$tabs","$tabList","$tabListItems","jsHiddenClass","changingHash","boundTabClick","boundTabKeydown","boundOnHashChange","onTabClick","onTabKeydown","onHashChange","_this$mql","setup","teardown","_this$getTab","$item","$tab","setAttributes","hideTab","$activeTab","getTab","showTab","removeEventListener","unsetAttributes","$tabWithHash","$previousTab","getCurrentTab","unhighlightTab","hidePanel","highlightTab","showPanel","panelId","$panel","getPanel","$currentTab","$nextTab","currentTarget","createHistoryEntry","activatePreviousTab","activateNextTab","parentElement","$nextTabListItem","nextElementSibling","$previousTabListItem","previousElementSibling","initAll","_config$scope","onError","log","components","accordion","button","characterCount","errorSummary","exitThisPage","notificationBanner","passwordInput","scope","createAll","createAllOptions","_createAllOptions$sco","$elements","map","filter"],"mappings":"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"}