govuk_tech_docs 3.3.1 → 3.4.5

Sign up to get free protection for your applications and to get access to all the features.
Files changed (91) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/publish.yaml +10 -7
  3. data/.github/workflows/test.yaml +8 -3
  4. data/.nvmrc +1 -1
  5. data/.rubocop.yml +3 -0
  6. data/CHANGELOG.md +21 -1
  7. data/Gemfile +1 -1
  8. data/govuk_tech_docs.gemspec +13 -9
  9. data/lib/assets/stylesheets/_govuk_tech_docs.scss +13 -0
  10. data/lib/govuk_tech_docs/api_reference/api_reference_extension.rb +1 -1
  11. data/lib/govuk_tech_docs/api_reference/api_reference_renderer.rb +10 -10
  12. data/lib/govuk_tech_docs/path_helpers.rb +40 -10
  13. data/lib/govuk_tech_docs/redirects.rb +2 -2
  14. data/lib/govuk_tech_docs/table_of_contents/heading.rb +2 -2
  15. data/lib/govuk_tech_docs/table_of_contents/heading_tree_renderer.rb +5 -5
  16. data/lib/govuk_tech_docs/table_of_contents/helpers.rb +3 -3
  17. data/lib/govuk_tech_docs/version.rb +1 -1
  18. data/lib/govuk_tech_docs.rb +4 -4
  19. data/lib/source/favicon.ico +0 -0
  20. data/lib/source/layouts/_footer.erb +1 -1
  21. data/lib/source/layouts/_header.erb +15 -17
  22. data/node_modules/govuk-frontend/govuk/all-ie8.scss +8 -0
  23. data/node_modules/govuk-frontend/govuk/all.js +4918 -3796
  24. data/node_modules/govuk-frontend/govuk/common/closest-attribute-value.js +54 -49
  25. data/node_modules/govuk-frontend/govuk/common/govuk-frontend-version.js +17 -0
  26. data/node_modules/govuk-frontend/govuk/common/index.js +172 -152
  27. data/node_modules/govuk-frontend/govuk/common/normalise-dataset.js +334 -321
  28. data/node_modules/govuk-frontend/govuk/common.js +171 -151
  29. data/node_modules/govuk-frontend/govuk/components/_all.scss +3 -2
  30. data/node_modules/govuk-frontend/govuk/components/accordion/_index.scss +26 -7
  31. data/node_modules/govuk-frontend/govuk/components/accordion/accordion.js +2203 -1650
  32. data/node_modules/govuk-frontend/govuk/components/back-link/_index.scss +24 -16
  33. data/node_modules/govuk-frontend/govuk/components/breadcrumbs/_index.scss +34 -11
  34. data/node_modules/govuk-frontend/govuk/components/button/_index.scss +49 -9
  35. data/node_modules/govuk-frontend/govuk/components/button/button.js +961 -916
  36. data/node_modules/govuk-frontend/govuk/components/character-count/character-count.js +2142 -2038
  37. data/node_modules/govuk-frontend/govuk/components/checkboxes/_index.scss +6 -6
  38. data/node_modules/govuk-frontend/govuk/components/checkboxes/checkboxes.js +1204 -1145
  39. data/node_modules/govuk-frontend/govuk/components/details/details.js +826 -799
  40. data/node_modules/govuk-frontend/govuk/components/error-summary/error-summary.js +1097 -1044
  41. data/node_modules/govuk-frontend/govuk/components/exit-this-page/_exit-this-page.scss +2 -0
  42. data/node_modules/govuk-frontend/govuk/components/exit-this-page/_index.scss +97 -0
  43. data/node_modules/govuk-frontend/govuk/components/exit-this-page/exit-this-page.js +2120 -0
  44. data/node_modules/govuk-frontend/govuk/components/file-upload/_index.scss +6 -1
  45. data/node_modules/govuk-frontend/govuk/components/footer/_index.scss +0 -7
  46. data/node_modules/govuk-frontend/govuk/components/header/_index.scss +6 -0
  47. data/node_modules/govuk-frontend/govuk/components/header/header.js +683 -1003
  48. data/node_modules/govuk-frontend/govuk/components/input/_index.scss +15 -3
  49. data/node_modules/govuk-frontend/govuk/components/notification-banner/notification-banner.js +786 -751
  50. data/node_modules/govuk-frontend/govuk/components/radios/_index.scss +5 -5
  51. data/node_modules/govuk-frontend/govuk/components/radios/radios.js +1151 -1105
  52. data/node_modules/govuk-frontend/govuk/components/select/_index.scss +7 -1
  53. data/node_modules/govuk-frontend/govuk/components/skip-link/skip-link.js +1045 -1014
  54. data/node_modules/govuk-frontend/govuk/components/summary-list/_index.scss +107 -0
  55. data/node_modules/govuk-frontend/govuk/components/tabs/tabs.js +1514 -1268
  56. data/node_modules/govuk-frontend/govuk/components/tag/_index.scss +18 -18
  57. data/node_modules/govuk-frontend/govuk/components/textarea/_index.scss +8 -1
  58. data/node_modules/govuk-frontend/govuk/core/_all.scss +1 -0
  59. data/node_modules/govuk-frontend/govuk/core/_govuk-frontend-version.scss +5 -0
  60. data/node_modules/govuk-frontend/govuk/helpers/_colour.scss +5 -2
  61. data/node_modules/govuk-frontend/govuk/helpers/_focused.scss +1 -1
  62. data/node_modules/govuk-frontend/govuk/helpers/_font-faces.scss +1 -1
  63. data/node_modules/govuk-frontend/govuk/helpers/_visually-hidden.scss +12 -0
  64. data/node_modules/govuk-frontend/govuk/i18n.js +371 -364
  65. data/node_modules/govuk-frontend/govuk/objects/_template.scss +20 -0
  66. data/node_modules/govuk-frontend/govuk/objects/_width-container.scss +1 -1
  67. data/node_modules/govuk-frontend/govuk/settings/_colours-organisations.scss +4 -0
  68. data/node_modules/govuk-frontend/govuk/settings/_ie8.scss +16 -0
  69. data/node_modules/govuk-frontend/govuk/settings/_links.scss +5 -1
  70. data/node_modules/govuk-frontend/govuk/settings/_measurements.scss +5 -5
  71. data/node_modules/govuk-frontend/govuk/tools/_ie8.scss +38 -2
  72. data/node_modules/govuk-frontend/govuk/vendor/polyfills/DOMTokenList.js +243 -241
  73. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Date/now.js +14 -12
  74. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Document.js +18 -16
  75. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Element/prototype/classList.js +553 -545
  76. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Element/prototype/closest.js +40 -36
  77. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Element/prototype/dataset.js +257 -250
  78. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Element/prototype/matches.js +22 -20
  79. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Element/prototype/nextElementSibling.js +204 -197
  80. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Element/prototype/previousElementSibling.js +204 -197
  81. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Element.js +109 -105
  82. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Event.js +407 -399
  83. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Function/prototype/bind.js +242 -238
  84. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Object/defineProperty.js +73 -71
  85. data/node_modules/govuk-frontend/govuk/vendor/polyfills/String/prototype/trim.js +15 -13
  86. data/node_modules/govuk-frontend/govuk/vendor/polyfills/Window.js +18 -16
  87. data/node_modules/govuk-frontend/govuk-prototype-kit/init.js +1 -0
  88. data/package-lock.json +2708 -8
  89. data/package.json +1 -1
  90. metadata +25 -7
  91. data/.ruby-version +0 -1
@@ -1,808 +1,843 @@
1
1
  (function (global, factory) {
2
- typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
3
- typeof define === 'function' && define.amd ? define('GOVUKFrontend.NotificationBanner', factory) :
4
- (global.GOVUKFrontend = global.GOVUKFrontend || {}, global.GOVUKFrontend.NotificationBanner = factory());
2
+ typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
3
+ typeof define === 'function' && define.amd ? define('GOVUKFrontend.NotificationBanner', factory) :
4
+ (global.GOVUKFrontend = global.GOVUKFrontend || {}, global.GOVUKFrontend.NotificationBanner = factory());
5
5
  }(this, (function () { 'use strict';
6
6
 
7
- (function(undefined) {
7
+ /**
8
+ * Common helpers which do not require polyfill.
9
+ *
10
+ * IMPORTANT: If a helper require a polyfill, please isolate it in its own module
11
+ * so that the polyfill can be properly tree-shaken and does not burden
12
+ * the components that do not need that helper
13
+ *
14
+ * @module common/index
15
+ */
16
+
17
+ /**
18
+ * Config flattening function
19
+ *
20
+ * Takes any number of objects, flattens them into namespaced key-value pairs,
21
+ * (e.g. {'i18n.showSection': 'Show section'}) and combines them together, with
22
+ * greatest priority on the LAST item passed in.
23
+ *
24
+ * @deprecated Will be made private in v5.0
25
+ * @returns {Object<string, unknown>} A flattened object of key-value pairs.
26
+ */
27
+ function mergeConfigs (/* configObject1, configObject2, ...configObjects */) {
28
+ /**
29
+ * Function to take nested objects and flatten them to a dot-separated keyed
30
+ * object. Doing this means we don't need to do any deep/recursive merging of
31
+ * each of our objects, nor transform our dataset from a flat list into a
32
+ * nested object.
33
+ *
34
+ * @param {Object<string, unknown>} configObject - Deeply nested object
35
+ * @returns {Object<string, unknown>} Flattened object with dot-separated keys
36
+ */
37
+ var flattenObject = function (configObject) {
38
+ // Prepare an empty return object
39
+ /** @type {Object<string, unknown>} */
40
+ var flattenedObject = {};
41
+
42
+ /**
43
+ * Our flattening function, this is called recursively for each level of
44
+ * depth in the object. At each level we prepend the previous level names to
45
+ * the key using `prefix`.
46
+ *
47
+ * @param {Partial<Object<string, unknown>>} obj - Object to flatten
48
+ * @param {string} [prefix] - Optional dot-separated prefix
49
+ */
50
+ var flattenLoop = function (obj, prefix) {
51
+ // Loop through keys...
52
+ for (var key in obj) {
53
+ // Check to see if this is a prototypical key/value,
54
+ // if it is, skip it.
55
+ if (!Object.prototype.hasOwnProperty.call(obj, key)) {
56
+ continue
57
+ }
58
+ var value = obj[key];
59
+ var prefixedKey = prefix ? prefix + '.' + key : key;
60
+ if (typeof value === 'object') {
61
+ // If the value is a nested object, recurse over that too
62
+ flattenLoop(value, prefixedKey);
63
+ } else {
64
+ // Otherwise, add this value to our return object
65
+ flattenedObject[prefixedKey] = value;
66
+ }
67
+ }
68
+ };
69
+
70
+ // Kick off the recursive loop
71
+ flattenLoop(configObject);
72
+ return flattenedObject
73
+ };
74
+
75
+ // Start with an empty object as our base
76
+ /** @type {Object<string, unknown>} */
77
+ var formattedConfigObject = {};
78
+
79
+ // Loop through each of the remaining passed objects and push their keys
80
+ // one-by-one into configObject. Any duplicate keys will override the existing
81
+ // key with the new value.
82
+ for (var i = 0; i < arguments.length; i++) {
83
+ var obj = flattenObject(arguments[i]);
84
+ for (var key in obj) {
85
+ if (Object.prototype.hasOwnProperty.call(obj, key)) {
86
+ formattedConfigObject[key] = obj[key];
87
+ }
88
+ }
89
+ }
90
+
91
+ return formattedConfigObject
92
+ }
93
+
94
+ /**
95
+ * @template {Node} ElementType
96
+ * @callback nodeListIterator
97
+ * @param {ElementType} value - The current node being iterated on
98
+ * @param {number} index - The current index in the iteration
99
+ * @param {NodeListOf<ElementType>} nodes - NodeList from querySelectorAll()
100
+ * @returns {void}
101
+ */
102
+
103
+ // @ts-nocheck
104
+ (function (undefined) {
105
+
106
+ // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Object/defineProperty/detect.js
107
+ var detect = (
108
+ // In IE8, defineProperty could only act on DOM elements, so full support
109
+ // for the feature requires the ability to set a property on an arbitrary object
110
+ 'defineProperty' in Object && (function() {
111
+ try {
112
+ var a = {};
113
+ Object.defineProperty(a, 'test', {value:42});
114
+ return true;
115
+ } catch(e) {
116
+ return false
117
+ }
118
+ }())
119
+ );
120
+
121
+ if (detect) return
8
122
 
9
- // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Window/detect.js
10
- var detect = ('Window' in this);
123
+ // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Object.defineProperty&flags=always
124
+ (function (nativeDefineProperty) {
125
+
126
+ var supportsAccessors = Object.prototype.hasOwnProperty('__defineGetter__');
127
+ var ERR_ACCESSORS_NOT_SUPPORTED = 'Getters & setters cannot be defined on this javascript engine';
128
+ var ERR_VALUE_ACCESSORS = 'A property cannot both have accessors and be writable or have a value';
129
+
130
+ Object.defineProperty = function defineProperty(object, property, descriptor) {
131
+
132
+ // Where native support exists, assume it
133
+ if (nativeDefineProperty && (object === window || object === document || object === Element.prototype || object instanceof Element)) {
134
+ return nativeDefineProperty(object, property, descriptor);
135
+ }
136
+
137
+ if (object === null || !(object instanceof Object || typeof object === 'object')) {
138
+ throw new TypeError('Object.defineProperty called on non-object');
139
+ }
140
+
141
+ if (!(descriptor instanceof Object)) {
142
+ throw new TypeError('Property description must be an object');
143
+ }
144
+
145
+ var propertyString = String(property);
146
+ var hasValueOrWritable = 'value' in descriptor || 'writable' in descriptor;
147
+ var getterType = 'get' in descriptor && typeof descriptor.get;
148
+ var setterType = 'set' in descriptor && typeof descriptor.set;
149
+
150
+ // handle descriptor.get
151
+ if (getterType) {
152
+ if (getterType !== 'function') {
153
+ throw new TypeError('Getter must be a function');
154
+ }
155
+ if (!supportsAccessors) {
156
+ throw new TypeError(ERR_ACCESSORS_NOT_SUPPORTED);
157
+ }
158
+ if (hasValueOrWritable) {
159
+ throw new TypeError(ERR_VALUE_ACCESSORS);
160
+ }
161
+ Object.__defineGetter__.call(object, propertyString, descriptor.get);
162
+ } else {
163
+ object[propertyString] = descriptor.value;
164
+ }
165
+
166
+ // handle descriptor.set
167
+ if (setterType) {
168
+ if (setterType !== 'function') {
169
+ throw new TypeError('Setter must be a function');
170
+ }
171
+ if (!supportsAccessors) {
172
+ throw new TypeError(ERR_ACCESSORS_NOT_SUPPORTED);
173
+ }
174
+ if (hasValueOrWritable) {
175
+ throw new TypeError(ERR_VALUE_ACCESSORS);
176
+ }
177
+ Object.__defineSetter__.call(object, propertyString, descriptor.set);
178
+ }
179
+
180
+ // OK to define value unconditionally - if a getter has been specified as well, an error would be thrown above
181
+ if ('value' in descriptor) {
182
+ object[propertyString] = descriptor.value;
183
+ }
184
+
185
+ return object;
186
+ };
187
+ }(Object.defineProperty));
188
+ })
189
+ .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
190
+
191
+ // @ts-nocheck
192
+ (function (undefined) {
193
+
194
+ // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Document/detect.js
195
+ var detect = ("Document" in this);
196
+
197
+ if (detect) return
11
198
 
12
- if (detect) return
199
+ // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Document&flags=always
200
+ if ((typeof WorkerGlobalScope === "undefined") && (typeof importScripts !== "function")) {
13
201
 
14
- // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Window&flags=always
15
- if ((typeof WorkerGlobalScope === "undefined") && (typeof importScripts !== "function")) {
16
- (function (global) {
17
- if (global.constructor) {
18
- global.Window = global.constructor;
19
- } else {
20
- (global.Window = global.constructor = new Function('return function Window() {}')()).prototype = this;
21
- }
22
- }(this));
23
- }
202
+ if (this.HTMLDocument) { // IE8
203
+
204
+ // HTMLDocument is an extension of Document. If the browser has HTMLDocument but not Document, the former will suffice as an alias for the latter.
205
+ this.Document = this.HTMLDocument;
206
+
207
+ } else {
208
+
209
+ // Create an empty function to act as the missing constructor for the document object, attach the document object as its prototype. The function needs to be anonymous else it is hoisted and causes the feature detect to prematurely pass, preventing the assignments below being made.
210
+ this.Document = this.HTMLDocument = document.constructor = (new Function('return function Document() {}')());
211
+ this.Document.prototype = document;
212
+ }
213
+ }
24
214
 
25
- })
26
- .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
27
215
 
28
- (function(undefined) {
216
+ })
217
+ .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
29
218
 
30
- // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Document/detect.js
31
- var detect = ("Document" in this);
219
+ // @ts-nocheck
32
220
 
33
- if (detect) return
221
+ (function(undefined) {
34
222
 
35
- // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Document&flags=always
36
- if ((typeof WorkerGlobalScope === "undefined") && (typeof importScripts !== "function")) {
223
+ // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Element/detect.js
224
+ var detect = ('Element' in this && 'HTMLElement' in this);
37
225
 
38
- if (this.HTMLDocument) { // IE8
226
+ if (detect) return
39
227
 
40
- // HTMLDocument is an extension of Document. If the browser has HTMLDocument but not Document, the former will suffice as an alias for the latter.
41
- this.Document = this.HTMLDocument;
228
+ // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Element&flags=always
229
+ (function () {
42
230
 
43
- } else {
231
+ // IE8
232
+ if (window.Element && !window.HTMLElement) {
233
+ window.HTMLElement = window.Element;
234
+ return;
235
+ }
44
236
 
45
- // Create an empty function to act as the missing constructor for the document object, attach the document object as its prototype. The function needs to be anonymous else it is hoisted and causes the feature detect to prematurely pass, preventing the assignments below being made.
46
- this.Document = this.HTMLDocument = document.constructor = (new Function('return function Document() {}')());
47
- this.Document.prototype = document;
48
- }
49
- }
50
-
51
-
52
- })
53
- .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
54
-
55
- (function(undefined) {
56
-
57
- // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Element/detect.js
58
- var detect = ('Element' in this && 'HTMLElement' in this);
59
-
60
- if (detect) return
61
-
62
- // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Element&flags=always
63
- (function () {
64
-
65
- // IE8
66
- if (window.Element && !window.HTMLElement) {
67
- window.HTMLElement = window.Element;
68
- return;
69
- }
70
-
71
- // create Element constructor
72
- window.Element = window.HTMLElement = new Function('return function Element() {}')();
73
-
74
- // generate sandboxed iframe
75
- var vbody = document.appendChild(document.createElement('body'));
76
- var frame = vbody.appendChild(document.createElement('iframe'));
77
-
78
- // use sandboxed iframe to replicate Element functionality
79
- var frameDocument = frame.contentWindow.document;
80
- var prototype = Element.prototype = frameDocument.appendChild(frameDocument.createElement('*'));
81
- var cache = {};
82
-
83
- // polyfill Element.prototype on an element
84
- var shiv = function (element, deep) {
85
- var
86
- childNodes = element.childNodes || [],
87
- index = -1,
88
- key, value, childNode;
89
-
90
- if (element.nodeType === 1 && element.constructor !== Element) {
91
- element.constructor = Element;
92
-
93
- for (key in cache) {
94
- value = cache[key];
95
- element[key] = value;
96
- }
97
- }
98
-
99
- while (childNode = deep && childNodes[++index]) {
100
- shiv(childNode, deep);
101
- }
102
-
103
- return element;
104
- };
105
-
106
- var elements = document.getElementsByTagName('*');
107
- var nativeCreateElement = document.createElement;
108
- var interval;
109
- var loopLimit = 100;
110
-
111
- prototype.attachEvent('onpropertychange', function (event) {
112
- var
113
- propertyName = event.propertyName,
114
- nonValue = !cache.hasOwnProperty(propertyName),
115
- newValue = prototype[propertyName],
116
- oldValue = cache[propertyName],
117
- index = -1,
118
- element;
119
-
120
- while (element = elements[++index]) {
121
- if (element.nodeType === 1) {
122
- if (nonValue || element[propertyName] === oldValue) {
123
- element[propertyName] = newValue;
124
- }
125
- }
126
- }
127
-
128
- cache[propertyName] = newValue;
129
- });
130
-
131
- prototype.constructor = Element;
132
-
133
- if (!prototype.hasAttribute) {
134
- // <Element>.hasAttribute
135
- prototype.hasAttribute = function hasAttribute(name) {
136
- return this.getAttribute(name) !== null;
137
- };
138
- }
139
-
140
- // Apply Element prototype to the pre-existing DOM as soon as the body element appears.
141
- function bodyCheck() {
142
- if (!(loopLimit--)) clearTimeout(interval);
143
- if (document.body && !document.body.prototype && /(complete|interactive)/.test(document.readyState)) {
144
- shiv(document, true);
145
- if (interval && document.body.prototype) clearTimeout(interval);
146
- return (!!document.body.prototype);
147
- }
148
- return false;
149
- }
150
- if (!bodyCheck()) {
151
- document.onreadystatechange = bodyCheck;
152
- interval = setInterval(bodyCheck, 25);
153
- }
154
-
155
- // Apply to any new elements created after load
156
- document.createElement = function createElement(nodeName) {
157
- var element = nativeCreateElement(String(nodeName).toLowerCase());
158
- return shiv(element);
159
- };
160
-
161
- // remove sandboxed iframe
162
- document.removeChild(vbody);
163
- }());
164
-
165
- })
166
- .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
167
-
168
- (function(undefined) {
169
-
170
- // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Object/defineProperty/detect.js
171
- var detect = (
172
- // In IE8, defineProperty could only act on DOM elements, so full support
173
- // for the feature requires the ability to set a property on an arbitrary object
174
- 'defineProperty' in Object && (function() {
175
- try {
176
- var a = {};
177
- Object.defineProperty(a, 'test', {value:42});
178
- return true;
179
- } catch(e) {
180
- return false
237
+ // create Element constructor
238
+ window.Element = window.HTMLElement = new Function('return function Element() {}')();
239
+
240
+ // generate sandboxed iframe
241
+ var vbody = document.appendChild(document.createElement('body'));
242
+ var frame = vbody.appendChild(document.createElement('iframe'));
243
+
244
+ // use sandboxed iframe to replicate Element functionality
245
+ var frameDocument = frame.contentWindow.document;
246
+ var prototype = Element.prototype = frameDocument.appendChild(frameDocument.createElement('*'));
247
+ var cache = {};
248
+
249
+ // polyfill Element.prototype on an element
250
+ var shiv = function (element, deep) {
251
+ var
252
+ childNodes = element.childNodes || [],
253
+ index = -1,
254
+ key, value, childNode;
255
+
256
+ if (element.nodeType === 1 && element.constructor !== Element) {
257
+ element.constructor = Element;
258
+
259
+ for (key in cache) {
260
+ value = cache[key];
261
+ element[key] = value;
262
+ }
263
+ }
264
+
265
+ while (childNode = deep && childNodes[++index]) {
266
+ shiv(childNode, deep);
267
+ }
268
+
269
+ return element;
270
+ };
271
+
272
+ var elements = document.getElementsByTagName('*');
273
+ var nativeCreateElement = document.createElement;
274
+ var interval;
275
+ var loopLimit = 100;
276
+
277
+ prototype.attachEvent('onpropertychange', function (event) {
278
+ var
279
+ propertyName = event.propertyName,
280
+ nonValue = !cache.hasOwnProperty(propertyName),
281
+ newValue = prototype[propertyName],
282
+ oldValue = cache[propertyName],
283
+ index = -1,
284
+ element;
285
+
286
+ while (element = elements[++index]) {
287
+ if (element.nodeType === 1) {
288
+ if (nonValue || element[propertyName] === oldValue) {
289
+ element[propertyName] = newValue;
290
+ }
291
+ }
292
+ }
293
+
294
+ cache[propertyName] = newValue;
295
+ });
296
+
297
+ prototype.constructor = Element;
298
+
299
+ if (!prototype.hasAttribute) {
300
+ // <Element>.hasAttribute
301
+ prototype.hasAttribute = function hasAttribute(name) {
302
+ return this.getAttribute(name) !== null;
303
+ };
181
304
  }
182
- }())
183
- );
184
-
185
- if (detect) return
186
-
187
- // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Object.defineProperty&flags=always
188
- (function (nativeDefineProperty) {
189
-
190
- var supportsAccessors = Object.prototype.hasOwnProperty('__defineGetter__');
191
- var ERR_ACCESSORS_NOT_SUPPORTED = 'Getters & setters cannot be defined on this javascript engine';
192
- var ERR_VALUE_ACCESSORS = 'A property cannot both have accessors and be writable or have a value';
193
-
194
- Object.defineProperty = function defineProperty(object, property, descriptor) {
195
-
196
- // Where native support exists, assume it
197
- if (nativeDefineProperty && (object === window || object === document || object === Element.prototype || object instanceof Element)) {
198
- return nativeDefineProperty(object, property, descriptor);
199
- }
200
-
201
- if (object === null || !(object instanceof Object || typeof object === 'object')) {
202
- throw new TypeError('Object.defineProperty called on non-object');
203
- }
204
-
205
- if (!(descriptor instanceof Object)) {
206
- throw new TypeError('Property description must be an object');
207
- }
208
-
209
- var propertyString = String(property);
210
- var hasValueOrWritable = 'value' in descriptor || 'writable' in descriptor;
211
- var getterType = 'get' in descriptor && typeof descriptor.get;
212
- var setterType = 'set' in descriptor && typeof descriptor.set;
213
-
214
- // handle descriptor.get
215
- if (getterType) {
216
- if (getterType !== 'function') {
217
- throw new TypeError('Getter must be a function');
218
- }
219
- if (!supportsAccessors) {
220
- throw new TypeError(ERR_ACCESSORS_NOT_SUPPORTED);
221
- }
222
- if (hasValueOrWritable) {
223
- throw new TypeError(ERR_VALUE_ACCESSORS);
224
- }
225
- Object.__defineGetter__.call(object, propertyString, descriptor.get);
226
- } else {
227
- object[propertyString] = descriptor.value;
228
- }
229
-
230
- // handle descriptor.set
231
- if (setterType) {
232
- if (setterType !== 'function') {
233
- throw new TypeError('Setter must be a function');
234
- }
235
- if (!supportsAccessors) {
236
- throw new TypeError(ERR_ACCESSORS_NOT_SUPPORTED);
237
- }
238
- if (hasValueOrWritable) {
239
- throw new TypeError(ERR_VALUE_ACCESSORS);
240
- }
241
- Object.__defineSetter__.call(object, propertyString, descriptor.set);
242
- }
243
-
244
- // OK to define value unconditionally - if a getter has been specified as well, an error would be thrown above
245
- if ('value' in descriptor) {
246
- object[propertyString] = descriptor.value;
247
- }
248
-
249
- return object;
250
- };
251
- }(Object.defineProperty));
252
- })
253
- .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
254
-
255
- (function(undefined) {
256
-
257
- // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Event/detect.js
258
- var detect = (
259
- (function(global) {
260
-
261
- if (!('Event' in global)) return false;
262
- if (typeof global.Event === 'function') return true;
263
-
264
- try {
265
-
266
- // In IE 9-11, the Event object exists but cannot be instantiated
267
- new Event('click');
268
- return true;
269
- } catch(e) {
305
+
306
+ // Apply Element prototype to the pre-existing DOM as soon as the body element appears.
307
+ function bodyCheck() {
308
+ if (!(loopLimit--)) clearTimeout(interval);
309
+ if (document.body && !document.body.prototype && /(complete|interactive)/.test(document.readyState)) {
310
+ shiv(document, true);
311
+ if (interval && document.body.prototype) clearTimeout(interval);
312
+ return (!!document.body.prototype);
313
+ }
270
314
  return false;
271
315
  }
272
- }(this))
273
- );
274
-
275
- if (detect) return
276
-
277
- // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Event&flags=always
278
- (function () {
279
- var unlistenableWindowEvents = {
280
- click: 1,
281
- dblclick: 1,
282
- keyup: 1,
283
- keypress: 1,
284
- keydown: 1,
285
- mousedown: 1,
286
- mouseup: 1,
287
- mousemove: 1,
288
- mouseover: 1,
289
- mouseenter: 1,
290
- mouseleave: 1,
291
- mouseout: 1,
292
- storage: 1,
293
- storagecommit: 1,
294
- textinput: 1
295
- };
296
-
297
- // This polyfill depends on availability of `document` so will not run in a worker
298
- // However, we asssume there are no browsers with worker support that lack proper
299
- // support for `Event` within the worker
300
- if (typeof document === 'undefined' || typeof window === 'undefined') return;
301
-
302
- function indexOf(array, element) {
303
- var
304
- index = -1,
305
- length = array.length;
306
-
307
- while (++index < length) {
308
- if (index in array && array[index] === element) {
309
- return index;
310
- }
311
- }
312
-
313
- return -1;
314
- }
315
-
316
- var existingProto = (window.Event && window.Event.prototype) || null;
317
- window.Event = Window.prototype.Event = function Event(type, eventInitDict) {
318
- if (!type) {
319
- throw new Error('Not enough arguments');
320
- }
321
-
322
- var event;
323
- // Shortcut if browser supports createEvent
324
- if ('createEvent' in document) {
325
- event = document.createEvent('Event');
326
- var bubbles = eventInitDict && eventInitDict.bubbles !== undefined ? eventInitDict.bubbles : false;
327
- var cancelable = eventInitDict && eventInitDict.cancelable !== undefined ? eventInitDict.cancelable : false;
328
-
329
- event.initEvent(type, bubbles, cancelable);
330
-
331
- return event;
332
- }
333
-
334
- event = document.createEventObject();
335
-
336
- event.type = type;
337
- event.bubbles = eventInitDict && eventInitDict.bubbles !== undefined ? eventInitDict.bubbles : false;
338
- event.cancelable = eventInitDict && eventInitDict.cancelable !== undefined ? eventInitDict.cancelable : false;
339
-
340
- return event;
341
- };
342
- if (existingProto) {
343
- Object.defineProperty(window.Event, 'prototype', {
344
- configurable: false,
345
- enumerable: false,
346
- writable: true,
347
- value: existingProto
348
- });
349
- }
350
-
351
- if (!('createEvent' in document)) {
352
- window.addEventListener = Window.prototype.addEventListener = Document.prototype.addEventListener = Element.prototype.addEventListener = function addEventListener() {
353
- var
354
- element = this,
355
- type = arguments[0],
356
- listener = arguments[1];
357
-
358
- if (element === window && type in unlistenableWindowEvents) {
359
- throw new Error('In IE8 the event: ' + type + ' is not available on the window object. Please see https://github.com/Financial-Times/polyfill-service/issues/317 for more information.');
360
- }
361
-
362
- if (!element._events) {
363
- element._events = {};
364
- }
365
-
366
- if (!element._events[type]) {
367
- element._events[type] = function (event) {
368
- var
369
- list = element._events[event.type].list,
370
- events = list.slice(),
371
- index = -1,
372
- length = events.length,
373
- eventElement;
374
-
375
- event.preventDefault = function preventDefault() {
376
- if (event.cancelable !== false) {
377
- event.returnValue = false;
378
- }
379
- };
380
-
381
- event.stopPropagation = function stopPropagation() {
382
- event.cancelBubble = true;
383
- };
384
-
385
- event.stopImmediatePropagation = function stopImmediatePropagation() {
386
- event.cancelBubble = true;
387
- event.cancelImmediate = true;
388
- };
389
-
390
- event.currentTarget = element;
391
- event.relatedTarget = event.fromElement || null;
392
- event.target = event.target || event.srcElement || element;
393
- event.timeStamp = new Date().getTime();
394
-
395
- if (event.clientX) {
396
- event.pageX = event.clientX + document.documentElement.scrollLeft;
397
- event.pageY = event.clientY + document.documentElement.scrollTop;
398
- }
399
-
400
- while (++index < length && !event.cancelImmediate) {
401
- if (index in events) {
402
- eventElement = events[index];
403
-
404
- if (indexOf(list, eventElement) !== -1 && typeof eventElement === 'function') {
405
- eventElement.call(element, event);
406
- }
407
- }
408
- }
409
- };
410
-
411
- element._events[type].list = [];
412
-
413
- if (element.attachEvent) {
414
- element.attachEvent('on' + type, element._events[type]);
415
- }
416
- }
417
-
418
- element._events[type].list.push(listener);
419
- };
420
-
421
- window.removeEventListener = Window.prototype.removeEventListener = Document.prototype.removeEventListener = Element.prototype.removeEventListener = function removeEventListener() {
422
- var
423
- element = this,
424
- type = arguments[0],
425
- listener = arguments[1],
426
- index;
427
-
428
- if (element._events && element._events[type] && element._events[type].list) {
429
- index = indexOf(element._events[type].list, listener);
430
-
431
- if (index !== -1) {
432
- element._events[type].list.splice(index, 1);
433
-
434
- if (!element._events[type].list.length) {
435
- if (element.detachEvent) {
436
- element.detachEvent('on' + type, element._events[type]);
437
- }
438
- delete element._events[type];
439
- }
440
- }
441
- }
442
- };
443
-
444
- window.dispatchEvent = Window.prototype.dispatchEvent = Document.prototype.dispatchEvent = Element.prototype.dispatchEvent = function dispatchEvent(event) {
445
- if (!arguments.length) {
446
- throw new Error('Not enough arguments');
447
- }
448
-
449
- if (!event || typeof event.type !== 'string') {
450
- throw new Error('DOM Events Exception 0');
451
- }
452
-
453
- var element = this, type = event.type;
454
-
455
- try {
456
- if (!event.bubbles) {
457
- event.cancelBubble = true;
458
-
459
- var cancelBubbleEvent = function (event) {
460
- event.cancelBubble = true;
461
-
462
- (element || window).detachEvent('on' + type, cancelBubbleEvent);
463
- };
464
-
465
- this.attachEvent('on' + type, cancelBubbleEvent);
466
- }
467
-
468
- this.fireEvent('on' + type, event);
469
- } catch (error) {
470
- event.target = element;
471
-
472
- do {
473
- event.currentTarget = element;
474
-
475
- if ('_events' in element && typeof element._events[type] === 'function') {
476
- element._events[type].call(element, event);
477
- }
478
-
479
- if (typeof element['on' + type] === 'function') {
480
- element['on' + type].call(element, event);
481
- }
482
-
483
- element = element.nodeType === 9 ? element.parentWindow : element.parentNode;
484
- } while (element && !event.cancelBubble);
485
- }
486
-
487
- return true;
488
- };
489
-
490
- // Add the DOMContentLoaded Event
491
- document.attachEvent('onreadystatechange', function() {
492
- if (document.readyState === 'complete') {
493
- document.dispatchEvent(new Event('DOMContentLoaded', {
494
- bubbles: true
495
- }));
496
- }
497
- });
498
- }
499
- }());
500
-
501
- })
502
- .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
503
-
504
- /**
505
- * Common helpers which do not require polyfill.
506
- *
507
- * IMPORTANT: If a helper require a polyfill, please isolate it in its own module
508
- * so that the polyfill can be properly tree-shaken and does not burden
509
- * the components that do not need that helper
510
- *
511
- * @module common/index
512
- */
513
-
514
- /**
515
- * Config flattening function
516
- *
517
- * Takes any number of objects, flattens them into namespaced key-value pairs,
518
- * (e.g. {'i18n.showSection': 'Show section'}) and combines them together, with
519
- * greatest priority on the LAST item passed in.
520
- *
521
- * @returns {object} A flattened object of key-value pairs.
522
- */
523
- function mergeConfigs (/* configObject1, configObject2, ...configObjects */) {
524
- /**
525
- * Function to take nested objects and flatten them to a dot-separated keyed
526
- * object. Doing this means we don't need to do any deep/recursive merging of
527
- * each of our objects, nor transform our dataset from a flat list into a
528
- * nested object.
529
- *
530
- * @param {object} configObject - Deeply nested object
531
- * @returns {object} Flattened object with dot-separated keys
532
- */
533
- var flattenObject = function (configObject) {
534
- // Prepare an empty return object
535
- var flattenedObject = {};
536
-
537
- // Our flattening function, this is called recursively for each level of
538
- // depth in the object. At each level we prepend the previous level names to
539
- // the key using `prefix`.
540
- var flattenLoop = function (obj, prefix) {
541
- // Loop through keys...
542
- for (var key in obj) {
543
- // Check to see if this is a prototypical key/value,
544
- // if it is, skip it.
545
- if (!Object.prototype.hasOwnProperty.call(obj, key)) {
546
- continue
547
- }
548
- var value = obj[key];
549
- var prefixedKey = prefix ? prefix + '.' + key : key;
550
- if (typeof value === 'object') {
551
- // If the value is a nested object, recurse over that too
552
- flattenLoop(value, prefixedKey);
553
- } else {
554
- // Otherwise, add this value to our return object
555
- flattenedObject[prefixedKey] = value;
556
- }
557
- }
558
- };
316
+ if (!bodyCheck()) {
317
+ document.onreadystatechange = bodyCheck;
318
+ interval = setInterval(bodyCheck, 25);
319
+ }
559
320
 
560
- // Kick off the recursive loop
561
- flattenLoop(configObject);
562
- return flattenedObject
563
- };
321
+ // Apply to any new elements created after load
322
+ document.createElement = function createElement(nodeName) {
323
+ var element = nativeCreateElement(String(nodeName).toLowerCase());
324
+ return shiv(element);
325
+ };
564
326
 
565
- // Start with an empty object as our base
566
- var formattedConfigObject = {};
567
-
568
- // Loop through each of the remaining passed objects and push their keys
569
- // one-by-one into configObject. Any duplicate keys will override the existing
570
- // key with the new value.
571
- for (var i = 0; i < arguments.length; i++) {
572
- var obj = flattenObject(arguments[i]);
573
- for (var key in obj) {
574
- if (Object.prototype.hasOwnProperty.call(obj, key)) {
575
- formattedConfigObject[key] = obj[key];
576
- }
577
- }
578
- }
327
+ // remove sandboxed iframe
328
+ document.removeChild(vbody);
329
+ }());
579
330
 
580
- return formattedConfigObject
581
- }
331
+ })
332
+ .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
582
333
 
583
- /**
584
- * @callback nodeListIterator
585
- * @param {Element} value - The current node being iterated on
586
- * @param {number} index - The current index in the iteration
587
- * @param {NodeListOf<Element>} nodes - NodeList from querySelectorAll()
588
- * @returns {undefined}
589
- */
334
+ // @ts-nocheck
590
335
 
591
- (function(undefined) {
336
+ (function(undefined) {
592
337
 
593
- // Detection from https://raw.githubusercontent.com/Financial-Times/polyfill-library/13cf7c340974d128d557580b5e2dafcd1b1192d1/polyfills/Element/prototype/dataset/detect.js
594
- var detect = (function(){
595
- if (!document.documentElement.dataset) {
596
- return false;
597
- }
598
- var el = document.createElement('div');
599
- el.setAttribute("data-a-b", "c");
600
- return el.dataset && el.dataset.aB == "c";
601
- }());
338
+ // Detection from https://raw.githubusercontent.com/Financial-Times/polyfill-library/13cf7c340974d128d557580b5e2dafcd1b1192d1/polyfills/Element/prototype/dataset/detect.js
339
+ var detect = (function(){
340
+ if (!document.documentElement.dataset) {
341
+ return false;
342
+ }
343
+ var el = document.createElement('div');
344
+ el.setAttribute("data-a-b", "c");
345
+ return el.dataset && el.dataset.aB == "c";
346
+ }());
602
347
 
603
- if (detect) return
348
+ if (detect) return
604
349
 
605
- // Polyfill derived from https://raw.githubusercontent.com/Financial-Times/polyfill-library/13cf7c340974d128d557580b5e2dafcd1b1192d1/polyfills/Element/prototype/dataset/polyfill.js
606
- Object.defineProperty(Element.prototype, 'dataset', {
607
- get: function() {
608
- var element = this;
609
- var attributes = this.attributes;
610
- var map = {};
611
-
612
- for (var i = 0; i < attributes.length; i++) {
613
- var attribute = attributes[i];
614
-
615
- // This regex has been edited from the original polyfill, to add
616
- // support for period (.) separators in data-* attribute names. These
617
- // are allowed in the HTML spec, but were not covered by the original
618
- // polyfill's regex. We use periods in our i18n implementation.
619
- if (attribute && attribute.name && (/^data-\w[.\w-]*$/).test(attribute.name)) {
620
- var name = attribute.name;
621
- var value = attribute.value;
622
-
623
- var propName = name.substr(5).replace(/-./g, function (prop) {
624
- return prop.charAt(1).toUpperCase();
625
- });
626
-
627
- // If this browser supports __defineGetter__ and __defineSetter__,
628
- // continue using defineProperty. If not (like IE 8 and below), we use
629
- // a hacky fallback which at least gives an object in the right format
630
- if ('__defineGetter__' in Object.prototype && '__defineSetter__' in Object.prototype) {
631
- Object.defineProperty(map, propName, {
632
- enumerable: true,
633
- get: function() {
634
- return this.value;
635
- }.bind({value: value || ''}),
636
- set: function setter(name, value) {
637
- if (typeof value !== 'undefined') {
638
- this.setAttribute(name, value);
639
- } else {
640
- this.removeAttribute(name);
641
- }
642
- }.bind(element, name)
350
+ // Polyfill derived from https://raw.githubusercontent.com/Financial-Times/polyfill-library/13cf7c340974d128d557580b5e2dafcd1b1192d1/polyfills/Element/prototype/dataset/polyfill.js
351
+ Object.defineProperty(Element.prototype, 'dataset', {
352
+ get: function() {
353
+ var element = this;
354
+ var attributes = this.attributes;
355
+ var map = {};
356
+
357
+ for (var i = 0; i < attributes.length; i++) {
358
+ var attribute = attributes[i];
359
+
360
+ // This regex has been edited from the original polyfill, to add
361
+ // support for period (.) separators in data-* attribute names. These
362
+ // are allowed in the HTML spec, but were not covered by the original
363
+ // polyfill's regex. We use periods in our i18n implementation.
364
+ if (attribute && attribute.name && (/^data-\w[.\w-]*$/).test(attribute.name)) {
365
+ var name = attribute.name;
366
+ var value = attribute.value;
367
+
368
+ var propName = name.substr(5).replace(/-./g, function (prop) {
369
+ return prop.charAt(1).toUpperCase();
643
370
  });
644
- } else {
645
- map[propName] = value;
646
- }
647
371
 
372
+ // If this browser supports __defineGetter__ and __defineSetter__,
373
+ // continue using defineProperty. If not (like IE 8 and below), we use
374
+ // a hacky fallback which at least gives an object in the right format
375
+ if ('__defineGetter__' in Object.prototype && '__defineSetter__' in Object.prototype) {
376
+ Object.defineProperty(map, propName, {
377
+ enumerable: true,
378
+ get: function() {
379
+ return this.value;
380
+ }.bind({value: value || ''}),
381
+ set: function setter(name, value) {
382
+ if (typeof value !== 'undefined') {
383
+ this.setAttribute(name, value);
384
+ } else {
385
+ this.removeAttribute(name);
386
+ }
387
+ }.bind(element, name)
388
+ });
389
+ } else {
390
+ map[propName] = value;
391
+ }
392
+
393
+ }
648
394
  }
395
+
396
+ return map;
649
397
  }
650
-
651
- return map;
398
+ });
399
+
400
+ }).call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
401
+
402
+ // @ts-nocheck
403
+ (function (undefined) {
404
+
405
+ // Detection from https://github.com/mdn/content/blob/cf607d68522cd35ee7670782d3ee3a361eaef2e4/files/en-us/web/javascript/reference/global_objects/string/trim/index.md#polyfill
406
+ var detect = ('trim' in String.prototype);
407
+
408
+ if (detect) return
409
+
410
+ // Polyfill from https://github.com/mdn/content/blob/cf607d68522cd35ee7670782d3ee3a361eaef2e4/files/en-us/web/javascript/reference/global_objects/string/trim/index.md#polyfill
411
+ String.prototype.trim = function () {
412
+ return this.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, '');
413
+ };
414
+
415
+ }).call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
416
+
417
+ /* eslint-disable es-x/no-string-prototype-trim -- Polyfill imported */
418
+
419
+ /**
420
+ * Normalise string
421
+ *
422
+ * 'If it looks like a duck, and it quacks like a duck…' 🦆
423
+ *
424
+ * If the passed value looks like a boolean or a number, convert it to a boolean
425
+ * or number.
426
+ *
427
+ * Designed to be used to convert config passed via data attributes (which are
428
+ * always strings) into something sensible.
429
+ *
430
+ * @deprecated Will be made private in v5.0
431
+ * @param {string} value - The value to normalise
432
+ * @returns {string | boolean | number | undefined} Normalised data
433
+ */
434
+ function normaliseString (value) {
435
+ if (typeof value !== 'string') {
436
+ return value
652
437
  }
653
- });
654
438
 
655
- }).call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
439
+ var trimmedValue = value.trim();
656
440
 
657
- (function(undefined) {
441
+ if (trimmedValue === 'true') {
442
+ return true
443
+ }
658
444
 
659
- // Detection from https://github.com/mdn/content/blob/cf607d68522cd35ee7670782d3ee3a361eaef2e4/files/en-us/web/javascript/reference/global_objects/string/trim/index.md#polyfill
660
- var detect = ('trim' in String.prototype);
661
-
662
- if (detect) return
445
+ if (trimmedValue === 'false') {
446
+ return false
447
+ }
663
448
 
664
- // Polyfill from https://github.com/mdn/content/blob/cf607d68522cd35ee7670782d3ee3a361eaef2e4/files/en-us/web/javascript/reference/global_objects/string/trim/index.md#polyfill
665
- String.prototype.trim = function () {
666
- return this.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, '');
667
- };
449
+ // Empty / whitespace-only strings are considered finite so we need to check
450
+ // the length of the trimmed string as well
451
+ if (trimmedValue.length > 0 && isFinite(Number(trimmedValue))) {
452
+ return Number(trimmedValue)
453
+ }
668
454
 
669
- }).call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
670
-
671
- /**
672
- * Normalise string
673
- *
674
- * 'If it looks like a duck, and it quacks like a duck…' 🦆
675
- *
676
- * If the passed value looks like a boolean or a number, convert it to a boolean
677
- * or number.
678
- *
679
- * Designed to be used to convert config passed via data attributes (which are
680
- * always strings) into something sensible.
681
- *
682
- * @param {string} value - The value to normalise
683
- * @returns {string | boolean | number | undefined} Normalised data
684
- */
685
- function normaliseString (value) {
686
- if (typeof value !== 'string') {
687
455
  return value
688
456
  }
689
457
 
690
- var trimmedValue = value.trim();
458
+ /**
459
+ * Normalise dataset
460
+ *
461
+ * Loop over an object and normalise each value using normaliseData function
462
+ *
463
+ * @deprecated Will be made private in v5.0
464
+ * @param {DOMStringMap} dataset - HTML element dataset
465
+ * @returns {Object<string, unknown>} Normalised dataset
466
+ */
467
+ function normaliseDataset (dataset) {
468
+ /** @type {Object<string, unknown>} */
469
+ var out = {};
691
470
 
692
- if (trimmedValue === 'true') {
693
- return true
694
- }
471
+ for (var key in dataset) {
472
+ out[key] = normaliseString(dataset[key]);
473
+ }
695
474
 
696
- if (trimmedValue === 'false') {
697
- return false
475
+ return out
698
476
  }
699
477
 
700
- // Empty / whitespace-only strings are considered finite so we need to check
701
- // the length of the trimmed string as well
702
- if (trimmedValue.length > 0 && isFinite(trimmedValue)) {
703
- return Number(trimmedValue)
704
- }
478
+ // @ts-nocheck
479
+ (function (undefined) {
480
+
481
+ // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Window/detect.js
482
+ var detect = ('Window' in this);
705
483
 
706
- return value
707
- }
708
-
709
- /**
710
- * Normalise dataset
711
- *
712
- * Loop over an object and normalise each value using normaliseData function
713
- *
714
- * @param {DOMStringMap} dataset - HTML element dataset
715
- * @returns {Object<string, string | boolean | number | undefined>} Normalised dataset
716
- */
717
- function normaliseDataset (dataset) {
718
- var out = {};
719
-
720
- for (var key in dataset) {
721
- out[key] = normaliseString(dataset[key]);
484
+ if (detect) return
485
+
486
+ // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Window&flags=always
487
+ if ((typeof WorkerGlobalScope === "undefined") && (typeof importScripts !== "function")) {
488
+ (function (global) {
489
+ if (global.constructor) {
490
+ global.Window = global.constructor;
491
+ } else {
492
+ (global.Window = global.constructor = new Function('return function Window() {}')()).prototype = this;
493
+ }
494
+ }(this));
722
495
  }
723
496
 
724
- return out
725
- }
726
-
727
- /**
728
- * Notification Banner component
729
- *
730
- * @class
731
- * @param {HTMLElement} $module - HTML element to use for notification banner
732
- * @param {NotificationBannerConfig} config - Notification banner config
733
- */
734
- function NotificationBanner ($module, config) {
735
- this.$module = $module;
736
-
737
- var defaultConfig = {
738
- disableAutoFocus: false
739
- };
740
- this.config = mergeConfigs(
741
- defaultConfig,
742
- config || {},
743
- normaliseDataset($module.dataset)
497
+ })
498
+ .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
499
+
500
+ // @ts-nocheck
501
+
502
+ (function(undefined) {
503
+
504
+ // Detection from https://github.com/Financial-Times/polyfill-service/blob/master/packages/polyfill-library/polyfills/Event/detect.js
505
+ var detect = (
506
+ (function(global) {
507
+
508
+ if (!('Event' in global)) return false;
509
+ if (typeof global.Event === 'function') return true;
510
+
511
+ try {
512
+
513
+ // In IE 9-11, the Event object exists but cannot be instantiated
514
+ new Event('click');
515
+ return true;
516
+ } catch(e) {
517
+ return false;
518
+ }
519
+ }(this))
744
520
  );
745
- }
746
-
747
- /**
748
- * Initialise the component
749
- */
750
- NotificationBanner.prototype.init = function () {
751
- var $module = this.$module;
752
- // Check for module
753
- if (!$module) {
754
- return
755
- }
756
521
 
757
- this.setFocus();
758
- };
759
-
760
- /**
761
- * Focus the element
762
- *
763
- * If `role="alert"` is set, focus the element to help some assistive technologies
764
- * prioritise announcing it.
765
- *
766
- * You can turn off the auto-focus functionality by setting `data-disable-auto-focus="true"` in the
767
- * component HTML. You might wish to do this based on user research findings, or to avoid a clash
768
- * with another element which should be focused when the page loads.
769
- */
770
- NotificationBanner.prototype.setFocus = function () {
771
- var $module = this.$module;
772
-
773
- if (this.config.disableAutoFocus) {
774
- return
775
- }
522
+ if (detect) return
776
523
 
777
- if ($module.getAttribute('role') !== 'alert') {
778
- return
779
- }
524
+ // Polyfill from https://cdn.polyfill.io/v2/polyfill.js?features=Event&flags=always
525
+ (function () {
526
+ var unlistenableWindowEvents = {
527
+ click: 1,
528
+ dblclick: 1,
529
+ keyup: 1,
530
+ keypress: 1,
531
+ keydown: 1,
532
+ mousedown: 1,
533
+ mouseup: 1,
534
+ mousemove: 1,
535
+ mouseover: 1,
536
+ mouseenter: 1,
537
+ mouseleave: 1,
538
+ mouseout: 1,
539
+ storage: 1,
540
+ storagecommit: 1,
541
+ textinput: 1
542
+ };
543
+
544
+ // This polyfill depends on availability of `document` so will not run in a worker
545
+ // However, we asssume there are no browsers with worker support that lack proper
546
+ // support for `Event` within the worker
547
+ if (typeof document === 'undefined' || typeof window === 'undefined') return;
548
+
549
+ function indexOf(array, element) {
550
+ var
551
+ index = -1,
552
+ length = array.length;
553
+
554
+ while (++index < length) {
555
+ if (index in array && array[index] === element) {
556
+ return index;
557
+ }
558
+ }
559
+
560
+ return -1;
561
+ }
780
562
 
781
- // Set tabindex to -1 to make the element focusable with JavaScript.
782
- // Remove the tabindex on blur as the component doesn't need to be focusable after the page has
783
- // loaded.
784
- if (!$module.getAttribute('tabindex')) {
785
- $module.setAttribute('tabindex', '-1');
563
+ var existingProto = (window.Event && window.Event.prototype) || null;
564
+ window.Event = Window.prototype.Event = function Event(type, eventInitDict) {
565
+ if (!type) {
566
+ throw new Error('Not enough arguments');
567
+ }
568
+
569
+ var event;
570
+ // Shortcut if browser supports createEvent
571
+ if ('createEvent' in document) {
572
+ event = document.createEvent('Event');
573
+ var bubbles = eventInitDict && eventInitDict.bubbles !== undefined ? eventInitDict.bubbles : false;
574
+ var cancelable = eventInitDict && eventInitDict.cancelable !== undefined ? eventInitDict.cancelable : false;
575
+
576
+ event.initEvent(type, bubbles, cancelable);
577
+
578
+ return event;
579
+ }
580
+
581
+ event = document.createEventObject();
582
+
583
+ event.type = type;
584
+ event.bubbles = eventInitDict && eventInitDict.bubbles !== undefined ? eventInitDict.bubbles : false;
585
+ event.cancelable = eventInitDict && eventInitDict.cancelable !== undefined ? eventInitDict.cancelable : false;
586
+
587
+ return event;
588
+ };
589
+ if (existingProto) {
590
+ Object.defineProperty(window.Event, 'prototype', {
591
+ configurable: false,
592
+ enumerable: false,
593
+ writable: true,
594
+ value: existingProto
595
+ });
596
+ }
786
597
 
787
- $module.addEventListener('blur', function () {
788
- $module.removeAttribute('tabindex');
789
- });
598
+ if (!('createEvent' in document)) {
599
+ window.addEventListener = Window.prototype.addEventListener = Document.prototype.addEventListener = Element.prototype.addEventListener = function addEventListener() {
600
+ var
601
+ element = this,
602
+ type = arguments[0],
603
+ listener = arguments[1];
604
+
605
+ if (element === window && type in unlistenableWindowEvents) {
606
+ throw new Error('In IE8 the event: ' + type + ' is not available on the window object. Please see https://github.com/Financial-Times/polyfill-service/issues/317 for more information.');
607
+ }
608
+
609
+ if (!element._events) {
610
+ element._events = {};
611
+ }
612
+
613
+ if (!element._events[type]) {
614
+ element._events[type] = function (event) {
615
+ var
616
+ list = element._events[event.type].list,
617
+ events = list.slice(),
618
+ index = -1,
619
+ length = events.length,
620
+ eventElement;
621
+
622
+ event.preventDefault = function preventDefault() {
623
+ if (event.cancelable !== false) {
624
+ event.returnValue = false;
625
+ }
626
+ };
627
+
628
+ event.stopPropagation = function stopPropagation() {
629
+ event.cancelBubble = true;
630
+ };
631
+
632
+ event.stopImmediatePropagation = function stopImmediatePropagation() {
633
+ event.cancelBubble = true;
634
+ event.cancelImmediate = true;
635
+ };
636
+
637
+ event.currentTarget = element;
638
+ event.relatedTarget = event.fromElement || null;
639
+ event.target = event.target || event.srcElement || element;
640
+ event.timeStamp = new Date().getTime();
641
+
642
+ if (event.clientX) {
643
+ event.pageX = event.clientX + document.documentElement.scrollLeft;
644
+ event.pageY = event.clientY + document.documentElement.scrollTop;
645
+ }
646
+
647
+ while (++index < length && !event.cancelImmediate) {
648
+ if (index in events) {
649
+ eventElement = events[index];
650
+
651
+ if (indexOf(list, eventElement) !== -1 && typeof eventElement === 'function') {
652
+ eventElement.call(element, event);
653
+ }
654
+ }
655
+ }
656
+ };
657
+
658
+ element._events[type].list = [];
659
+
660
+ if (element.attachEvent) {
661
+ element.attachEvent('on' + type, element._events[type]);
662
+ }
663
+ }
664
+
665
+ element._events[type].list.push(listener);
666
+ };
667
+
668
+ window.removeEventListener = Window.prototype.removeEventListener = Document.prototype.removeEventListener = Element.prototype.removeEventListener = function removeEventListener() {
669
+ var
670
+ element = this,
671
+ type = arguments[0],
672
+ listener = arguments[1],
673
+ index;
674
+
675
+ if (element._events && element._events[type] && element._events[type].list) {
676
+ index = indexOf(element._events[type].list, listener);
677
+
678
+ if (index !== -1) {
679
+ element._events[type].list.splice(index, 1);
680
+
681
+ if (!element._events[type].list.length) {
682
+ if (element.detachEvent) {
683
+ element.detachEvent('on' + type, element._events[type]);
684
+ }
685
+ delete element._events[type];
686
+ }
687
+ }
688
+ }
689
+ };
690
+
691
+ window.dispatchEvent = Window.prototype.dispatchEvent = Document.prototype.dispatchEvent = Element.prototype.dispatchEvent = function dispatchEvent(event) {
692
+ if (!arguments.length) {
693
+ throw new Error('Not enough arguments');
694
+ }
695
+
696
+ if (!event || typeof event.type !== 'string') {
697
+ throw new Error('DOM Events Exception 0');
698
+ }
699
+
700
+ var element = this, type = event.type;
701
+
702
+ try {
703
+ if (!event.bubbles) {
704
+ event.cancelBubble = true;
705
+
706
+ var cancelBubbleEvent = function (event) {
707
+ event.cancelBubble = true;
708
+
709
+ (element || window).detachEvent('on' + type, cancelBubbleEvent);
710
+ };
711
+
712
+ this.attachEvent('on' + type, cancelBubbleEvent);
713
+ }
714
+
715
+ this.fireEvent('on' + type, event);
716
+ } catch (error) {
717
+ event.target = element;
718
+
719
+ do {
720
+ event.currentTarget = element;
721
+
722
+ if ('_events' in element && typeof element._events[type] === 'function') {
723
+ element._events[type].call(element, event);
724
+ }
725
+
726
+ if (typeof element['on' + type] === 'function') {
727
+ element['on' + type].call(element, event);
728
+ }
729
+
730
+ element = element.nodeType === 9 ? element.parentWindow : element.parentNode;
731
+ } while (element && !event.cancelBubble);
732
+ }
733
+
734
+ return true;
735
+ };
736
+
737
+ // Add the DOMContentLoaded Event
738
+ document.attachEvent('onreadystatechange', function() {
739
+ if (document.readyState === 'complete') {
740
+ document.dispatchEvent(new Event('DOMContentLoaded', {
741
+ bubbles: true
742
+ }));
743
+ }
744
+ });
745
+ }
746
+ }());
747
+
748
+ })
749
+ .call('object' === typeof window && window || 'object' === typeof self && self || 'object' === typeof global && global || {});
750
+
751
+ /**
752
+ * Notification Banner component
753
+ *
754
+ * @class
755
+ * @param {Element} $module - HTML element to use for notification banner
756
+ * @param {NotificationBannerConfig} [config] - Notification banner config
757
+ */
758
+ function NotificationBanner ($module, config) {
759
+ if (!($module instanceof HTMLElement)) {
760
+ return this
761
+ }
762
+
763
+ /** @deprecated Will be made private in v5.0 */
764
+ this.$module = $module;
765
+
766
+ var defaultConfig = {
767
+ disableAutoFocus: false
768
+ };
769
+
770
+ /**
771
+ * @deprecated Will be made private in v5.0
772
+ * @type {NotificationBannerConfig}
773
+ */
774
+ this.config = mergeConfigs(
775
+ defaultConfig,
776
+ config || {},
777
+ normaliseDataset($module.dataset)
778
+ );
790
779
  }
791
780
 
792
- $module.focus();
793
- };
781
+ /**
782
+ * Initialise component
783
+ */
784
+ NotificationBanner.prototype.init = function () {
785
+ // Check that required elements are present
786
+ if (!this.$module) {
787
+ return
788
+ }
789
+
790
+ this.setFocus();
791
+ };
794
792
 
795
- /**
796
- * Notification banner config
797
- *
798
- * @typedef {object} NotificationBannerConfig
799
- * @property {boolean} [disableAutoFocus = false] -
800
- * If set to `true` the notification banner will not be focussed when the page
801
- * loads. This only applies if the component has a `role` of `alert` in
802
- * other cases the component will not be focused on page load, regardless of
803
- * this option.
804
- */
793
+ /**
794
+ * Focus the element
795
+ *
796
+ * If `role="alert"` is set, focus the element to help some assistive technologies
797
+ * prioritise announcing it.
798
+ *
799
+ * You can turn off the auto-focus functionality by setting `data-disable-auto-focus="true"` in the
800
+ * component HTML. You might wish to do this based on user research findings, or to avoid a clash
801
+ * with another element which should be focused when the page loads.
802
+ *
803
+ * @deprecated Will be made private in v5.0
804
+ */
805
+ NotificationBanner.prototype.setFocus = function () {
806
+ var $module = this.$module;
807
+
808
+ if (this.config.disableAutoFocus) {
809
+ return
810
+ }
811
+
812
+ if ($module.getAttribute('role') !== 'alert') {
813
+ return
814
+ }
815
+
816
+ // Set tabindex to -1 to make the element focusable with JavaScript.
817
+ // Remove the tabindex on blur as the component doesn't need to be focusable after the page has
818
+ // loaded.
819
+ if (!$module.getAttribute('tabindex')) {
820
+ $module.setAttribute('tabindex', '-1');
821
+
822
+ $module.addEventListener('blur', function () {
823
+ $module.removeAttribute('tabindex');
824
+ });
825
+ }
826
+
827
+ $module.focus();
828
+ };
829
+
830
+ /**
831
+ * Notification banner config
832
+ *
833
+ * @typedef {object} NotificationBannerConfig
834
+ * @property {boolean} [disableAutoFocus = false] - If set to `true` the
835
+ * notification banner will not be focussed when the page loads. This only
836
+ * applies if the component has a `role` of `alert` – in other cases the
837
+ * component will not be focused on page load, regardless of this option.
838
+ */
805
839
 
806
- return NotificationBanner;
840
+ return NotificationBanner;
807
841
 
808
842
  })));
843
+ //# sourceMappingURL=notification-banner.js.map