@iamproperty/components 7.8.1 → 7.8.2--beta2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (164) hide show
  1. package/assets/css/components/card.component.css +1 -1
  2. package/assets/css/components/card.component.css.map +1 -1
  3. package/assets/css/components/card.module.css +1 -1
  4. package/assets/css/components/card.module.css.map +1 -1
  5. package/assets/css/components/filter-card.component.css +1 -1
  6. package/assets/css/components/filter-card.component.css.map +1 -1
  7. package/assets/css/components/modal.component.css +1 -1
  8. package/assets/css/components/modal.component.css.map +1 -1
  9. package/assets/css/components/record-card.component.css +1 -1
  10. package/assets/css/components/record-card.component.css.map +1 -1
  11. package/assets/css/components/video-card.component.css +1 -1
  12. package/assets/css/components/video-card.component.css.map +1 -1
  13. package/assets/css/components/video-modal.component.css +1 -1
  14. package/assets/css/components/video-modal.component.css.map +1 -1
  15. package/assets/js/components/accordion/accordion.component.min.js +1 -1
  16. package/assets/js/components/actionbar/actionbar.component.min.js +1 -1
  17. package/assets/js/components/address-lookup/address-lookup.component.min.js +3 -3
  18. package/assets/js/components/address-lookup/address-lookup.component.min.js.map +1 -1
  19. package/assets/js/components/advanced-select/advanced-select.component.min.js +3 -3
  20. package/assets/js/components/advanced-select/advanced-select.component.min.js.map +1 -1
  21. package/assets/js/components/applied-filters/applied-filters.component.min.js +1 -1
  22. package/assets/js/components/banner/banner.component.min.js +1 -1
  23. package/assets/js/components/barchart/barchart.component.min.js +1 -1
  24. package/assets/js/components/bento-grid/bento-grid.component.min.js +1 -1
  25. package/assets/js/components/bone/bone.component.min.js +1 -1
  26. package/assets/js/components/button/button.component.min.js +1 -1
  27. package/assets/js/components/calendar/calendar.component.min.js +1 -1
  28. package/assets/js/components/card/card.component.min.js +7 -7
  29. package/assets/js/components/card/card.component.min.js.map +1 -1
  30. package/assets/js/components/carousel/carousel.component.min.js +1 -1
  31. package/assets/js/components/collapsible-side/collapsible-side.component.min.js +1 -1
  32. package/assets/js/components/config/config.component.min.js +1 -1
  33. package/assets/js/components/content/content.component.min.js +1 -1
  34. package/assets/js/components/darkmode/darkmode.component.min.js +1 -1
  35. package/assets/js/components/doughnutchart/doughnutchart.component.min.js +1 -1
  36. package/assets/js/components/fileupload/fileupload.component.min.js +1 -1
  37. package/assets/js/components/filter-card/filter-card.component.min.js +5 -5
  38. package/assets/js/components/filter-card/filter-card.component.min.js.map +1 -1
  39. package/assets/js/components/filterlist/filterlist.component.min.js +1 -1
  40. package/assets/js/components/form/form.component.min.js +1 -1
  41. package/assets/js/components/header/header.component.min.js +1 -1
  42. package/assets/js/components/inline-edit/inline-edit.component.min.js +1 -1
  43. package/assets/js/components/input/input.component.min.js +1 -1
  44. package/assets/js/components/input-range/input-range.component.min.js +1 -1
  45. package/assets/js/components/marketing/marketing.component.min.js +1 -1
  46. package/assets/js/components/menu/menu.component.min.js +1 -1
  47. package/assets/js/components/milestone/milestone.component.min.js +1 -1
  48. package/assets/js/components/milestone-group/milestone-group.component.min.js +1 -1
  49. package/assets/js/components/modal/modal.component.min.js +2 -2
  50. package/assets/js/components/multi-step/multi-step.component.min.js +1 -1
  51. package/assets/js/components/multi-step-modal/multi-step-modal.component.min.js +1 -1
  52. package/assets/js/components/multiselect/multiselect.component.js +17 -5
  53. package/assets/js/components/multiselect/multiselect.component.min.js +3 -3
  54. package/assets/js/components/multiselect/multiselect.component.min.js.map +1 -1
  55. package/assets/js/components/nav/nav.component.min.js +1 -1
  56. package/assets/js/components/notification/notification.component.min.js +1 -1
  57. package/assets/js/components/pagination/pagination.component.min.js +1 -1
  58. package/assets/js/components/password/password.component.min.js +1 -1
  59. package/assets/js/components/popover/popover.component.min.js +1 -1
  60. package/assets/js/components/rank/rank.component.min.js +1 -1
  61. package/assets/js/components/rankings/rankings.component.min.js +1 -1
  62. package/assets/js/components/rating/rating.component.min.js +1 -1
  63. package/assets/js/components/record-card/record-card.component.min.js +5 -5
  64. package/assets/js/components/record-card/record-card.component.min.js.map +1 -1
  65. package/assets/js/components/search/search.component.js +2 -2
  66. package/assets/js/components/search/search.component.min.js +5 -5
  67. package/assets/js/components/search/search.component.min.js.map +1 -1
  68. package/assets/js/components/skeleton/skeleton.component.min.js +1 -1
  69. package/assets/js/components/slider/slider.component.min.js +1 -1
  70. package/assets/js/components/split-button/split-button.component.min.js +1 -1
  71. package/assets/js/components/std-address-lookup/std-address-lookup.component.min.js +4 -4
  72. package/assets/js/components/std-address-lookup/std-address-lookup.component.min.js.map +1 -1
  73. package/assets/js/components/std-nav/std-nav.component.min.js +1 -1
  74. package/assets/js/components/std-nav-standalone/std-nav-standalone.component.min.js +1 -1
  75. package/assets/js/components/table/table.component.min.js +1 -1
  76. package/assets/js/components/table-ajax/table-ajax.component.min.js +1 -1
  77. package/assets/js/components/table-basic/table-basic.component.min.js +1 -1
  78. package/assets/js/components/table-no-submit/table-no-submit.component.min.js +1 -1
  79. package/assets/js/components/table-submit/table-submit.component.min.js +1 -1
  80. package/assets/js/components/tabs/tabs.component.min.js +1 -1
  81. package/assets/js/components/tag/tag.component.min.js +1 -1
  82. package/assets/js/components/tooltip/tooltip.component.min.js +1 -1
  83. package/assets/js/components/video/video.component.min.js +1 -1
  84. package/assets/js/components/video-card/video-card.component.min.js +6 -6
  85. package/assets/js/components/video-card/video-card.component.min.js.map +1 -1
  86. package/assets/js/components/video-modal/video-modal.component.min.js +2 -2
  87. package/assets/js/components/word-count/word-count.component.min.js +1 -1
  88. package/assets/js/modules/advanced-select.js +6 -3
  89. package/assets/js/modules/advanced-select.test.js +28 -0
  90. package/assets/js/modules/applied-filters.test.js +21 -0
  91. package/assets/js/modules/card.module.js +3 -2
  92. package/assets/js/modules/card.module.test.js +20 -0
  93. package/assets/js/modules/carousel.test.js +18 -0
  94. package/assets/js/modules/chart.module.test.js +22 -0
  95. package/assets/js/modules/chart.test.js +15 -0
  96. package/assets/js/modules/content.test.js +20 -0
  97. package/assets/js/modules/data-layer.test.js +12 -0
  98. package/assets/js/modules/dialogs.test.js +13 -0
  99. package/assets/js/modules/drawer.test.js +16 -0
  100. package/assets/js/modules/dropdown.test.js +31 -0
  101. package/assets/js/modules/dynamicEvents.test.js +19 -0
  102. package/assets/js/modules/fileupload.test.js +17 -0
  103. package/assets/js/modules/filterlist.test.js +18 -0
  104. package/assets/js/modules/{helper.test.js → helpers.test.js} +42 -2
  105. package/assets/js/modules/integration-tests.test.js +12 -0
  106. package/assets/js/modules/milestone-group.test.js +18 -0
  107. package/assets/js/modules/milestone.test.js +31 -0
  108. package/assets/js/modules/modal.test.js +18 -0
  109. package/assets/js/modules/nav.test.js +37 -0
  110. package/assets/js/modules/notification.test.js +11 -0
  111. package/assets/js/modules/orderablelist.test.js +15 -0
  112. package/assets/js/modules/pagination.test.js +21 -0
  113. package/assets/js/modules/password.test.js +33 -0
  114. package/assets/js/modules/table.test.js +42 -0
  115. package/assets/js/modules/tabs.test.js +23 -0
  116. package/assets/js/modules/test-dom.js +725 -0
  117. package/assets/js/modules/test-globals.js +7 -0
  118. package/assets/js/modules/test-utils.js +26 -0
  119. package/assets/js/modules/test.js +20 -1
  120. package/assets/js/modules/testimonial.test.js +14 -0
  121. package/assets/js/modules/videos.test.js +37 -0
  122. package/assets/js/scripts.bundle.js +1 -1
  123. package/assets/js/scripts.bundle.min.js +1 -1
  124. package/assets/sass/components/card.module.scss +2 -0
  125. package/assets/sass/components/modal.component.scss +1 -2
  126. package/assets/ts/components/multiselect/multiselect.component.ts +25 -5
  127. package/assets/ts/components/search/search.component.ts +2 -2
  128. package/assets/ts/modules/advanced-select.test.ts +33 -0
  129. package/assets/ts/modules/advanced-select.ts +6 -2
  130. package/assets/ts/modules/applied-filters.test.ts +25 -0
  131. package/assets/ts/modules/card.module.test.ts +24 -0
  132. package/assets/ts/modules/card.module.ts +2 -1
  133. package/assets/ts/modules/carousel.test.ts +27 -0
  134. package/assets/ts/modules/chart.module.test.ts +26 -0
  135. package/assets/ts/modules/chart.test.ts +19 -0
  136. package/assets/ts/modules/content.test.ts +24 -0
  137. package/assets/ts/modules/data-layer.test.ts +15 -0
  138. package/assets/ts/modules/dialogs.test.ts +17 -0
  139. package/assets/ts/modules/drawer.test.ts +20 -0
  140. package/assets/ts/modules/dropdown.test.ts +38 -0
  141. package/assets/ts/modules/dynamicEvents.test.ts +23 -0
  142. package/assets/ts/modules/fileupload.test.ts +21 -0
  143. package/assets/ts/modules/filterlist.test.ts +22 -0
  144. package/assets/ts/modules/{helper.test.ts → helpers.test.ts} +63 -2
  145. package/assets/ts/modules/integration-tests.test.ts +15 -0
  146. package/assets/ts/modules/milestone-group.test.ts +22 -0
  147. package/assets/ts/modules/milestone.test.ts +35 -0
  148. package/assets/ts/modules/modal.test.ts +22 -0
  149. package/assets/ts/modules/nav.test.ts +41 -0
  150. package/assets/ts/modules/notification.test.ts +15 -0
  151. package/assets/ts/modules/orderablelist.test.ts +19 -0
  152. package/assets/ts/modules/pagination.test.ts +25 -0
  153. package/assets/ts/modules/password.test.ts +40 -0
  154. package/assets/ts/modules/table.test.ts +57 -0
  155. package/assets/ts/modules/tabs.test.ts +27 -0
  156. package/assets/ts/modules/test-dom.ts +798 -0
  157. package/assets/ts/modules/test-globals.ts +7 -0
  158. package/assets/ts/modules/test-utils.ts +29 -0
  159. package/assets/ts/modules/test.ts +28 -1
  160. package/assets/ts/modules/testimonial.test.ts +17 -0
  161. package/assets/ts/modules/videos.test.ts +33 -0
  162. package/dist/components.es.js +24 -24
  163. package/dist/components.umd.js +111 -111
  164. package/package.json +1 -1
@@ -0,0 +1,7 @@
1
+ if (typeof globalThis.window === 'undefined') {
2
+ Object.defineProperty(globalThis, 'window', {
3
+ value: new EventTarget(),
4
+ writable: true,
5
+ configurable: true,
6
+ });
7
+ }
@@ -0,0 +1,26 @@
1
+ /* eslint-disable */
2
+ export const append = (parent, ...children) => {
3
+ children.forEach((child) => parent.appendChild(child));
4
+ return parent;
5
+ };
6
+ export const silenceConsole = (callback) => {
7
+ const originalConsole = {
8
+ groupCollapsed: console.groupCollapsed,
9
+ groupEnd: console.groupEnd,
10
+ log: console.log,
11
+ table: console.table,
12
+ };
13
+ console.groupCollapsed = () => { };
14
+ console.groupEnd = () => { };
15
+ console.log = () => { };
16
+ console.table = () => { };
17
+ try {
18
+ return callback();
19
+ }
20
+ finally {
21
+ console.groupCollapsed = originalConsole.groupCollapsed;
22
+ console.groupEnd = originalConsole.groupEnd;
23
+ console.log = originalConsole.log;
24
+ console.table = originalConsole.table;
25
+ }
26
+ };
@@ -4,12 +4,31 @@ export const describe = (desc, fn) => {
4
4
  };
5
5
  export const it = (desc, fn) => {
6
6
  try {
7
- fn();
7
+ const result = fn();
8
+ if (result && typeof result.then === 'function') {
9
+ const pendingTests = globalThis.__unitTestPromises || [];
10
+ globalThis.__unitTestPromises = pendingTests;
11
+ pendingTests.push(result
12
+ .then(() => {
13
+ console.log(`\x1b[32m ${desc} \x1b[0m`);
14
+ })
15
+ .catch((error) => {
16
+ console.log(`\x1b[31m ${desc} \x1b[0m`);
17
+ console.error(error);
18
+ if (typeof process !== 'undefined') {
19
+ process.exitCode = 1;
20
+ }
21
+ }));
22
+ return;
23
+ }
8
24
  console.log(`\x1b[32m ${desc} \x1b[0m`);
9
25
  }
10
26
  catch (error) {
11
27
  console.log(`\x1b[31m ${desc} \x1b[0m`);
12
28
  console.error(error);
29
+ if (typeof process !== 'undefined') {
30
+ process.exitCode = 1;
31
+ }
13
32
  }
14
33
  };
15
34
  export const expect = (isTrue) => {
@@ -0,0 +1,14 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { append } from './test-utils.ts';
4
+ import testimonial from './testimonial.ts';
5
+ installTestDom();
6
+ describe('Testimonial module', () => {
7
+ it('does not wire testimonial controls for a single image', () => {
8
+ const testimonialElement = createElement('div');
9
+ const images = createElement('div', { class: 'testimonial__images' });
10
+ append(images, createElement('img'));
11
+ append(testimonialElement, images);
12
+ expect(testimonial(testimonialElement) === false);
13
+ });
14
+ });
@@ -0,0 +1,37 @@
1
+ /* eslint-disable */
2
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4
+ return new (P || (P = Promise))(function (resolve, reject) {
5
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
6
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
7
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
8
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
9
+ });
10
+ };
11
+ import { describe, expect, it } from './test.ts';
12
+ import { createElement, installTestDom } from './test-dom.ts';
13
+ import { append } from './test-utils.ts';
14
+ import { createYoutTubeVideo, openVimeoVideo } from './videos.ts';
15
+ const { window } = installTestDom();
16
+ describe('Videos module', () => {
17
+ it('reuses existing YouTube players and renders Vimeo embeds', () => __awaiter(void 0, void 0, void 0, function* () {
18
+ let played = false;
19
+ window.player = {
20
+ abc123: {
21
+ pauseVideo: () => { },
22
+ playVideo: () => {
23
+ played = true;
24
+ },
25
+ },
26
+ };
27
+ window.dataLayer = [];
28
+ const embed = createElement('div', { id: 'abc123' });
29
+ const youtubeResult = yield createYoutTubeVideo(embed, 'abc123');
30
+ const component = createElement('iam-video', { dataVimeo: '98765' });
31
+ append(component, createElement('div', { class: 'embed' }));
32
+ yield openVimeoVideo(component);
33
+ expect(youtubeResult === false);
34
+ expect(played);
35
+ expect(component.querySelector('.embed').innerHTML.includes('player.vimeo.com/video/98765'));
36
+ }));
37
+ });
@@ -1,5 +1,5 @@
1
1
  /*!
2
- * iamKey v7.8.1
2
+ * iamKey v7.8.2--beta2
3
3
  * Copyright 2022-2026 iamproperty
4
4
  */(function(w){typeof define=="function"&&define.amd?define(w):w()})((function(){"use strict";var w=a=>{a.classList.add("js-enabled"),(navigator.userAgent.indexOf("MSIE")!==-1||navigator.appVersion.indexOf("Trident/")>0)&&a.classList.add("ie")},H=a=>{var e=function(n){var r=document.querySelector('label[for="'.concat(n.replace("#",""),'"]')),s=document.querySelector(n+" summary"),u=document.querySelector("dialog".concat(n)),i=document.querySelector("detail".concat(n));r instanceof HTMLElement?r.click():s instanceof HTMLElement?s.click():u instanceof HTMLElement?u.showModal():i instanceof HTMLElement&&i.addAttribute("open")};location.hash&&e(location.hash),window.addEventListener("hashchange",function(){e(location.hash)},!1),addEventListener("popstate",o=>{if(o&&o.state&&o.state.type&&o.state.type=="pagination"){var n=document.querySelector("#".concat(o.state.form)),r=document.querySelector("#".concat(o.state.form," [data-pagination]"));r?r.value=o.state.page:n.innerHTML+='<input name="page" type="hidden" data-pagination="true" value="'.concat(o.state.page,'" />'),n.dispatchEvent(new Event("submit"))}}),Array.from(document.querySelectorAll("form")).forEach(o=>{o?.closest("iam-form")||o.addEventListener("submit",n=>{var r;o.querySelector(":invalid")&&(o.classList.add("was-validated"),(r=o?.querySelector("input:invalid"))===null||r===void 0||r.scrollIntoView(),n.preventDefault())})}),document.addEventListener("click",o=>{var n;if(o&&o.target instanceof HTMLElement&&o.target.matches("form button:not([type=button])")){var r=o.target.closest("form");r?.closest("iam-form")||(Array.from(r.querySelectorAll("[data-password-type]")).forEach(s=>{s.setAttribute("type","password")}),(r.querySelector(":invalid")||r.querySelector('.pwd-checker[data-strength="1"]')||r.querySelector('.pwd-checker[data-strength="2"]'))&&(r.classList.add("was-validated"),(n=r?.querySelector("input:invalid"))===null||n===void 0||n.scrollIntoView(),o.preventDefault()),r.querySelector("iam-multiselect[data-is-required][data-error]")&&(r.classList.add("was-validated"),o.preventDefault()))}}),document.addEventListener("keydown",o=>{o.key==="Escape"&&document.querySelector(".dialog--transactional[open], .dialog--acknowledgement[open]")&&(o.preventDefault(),o.stopPropagation())}),Array.from(document.querySelectorAll("label progress")).forEach(o=>{var n=o.closest("label");n.setAttribute("data-percent",o.getAttribute("value"))})};function N(){window.dataLayer=window.dataLayer||[],window.dataLayer.push({event:"Pageview",pageTitle:document.title}),document.addEventListener("click",a=>{var e=a.target.closest("[open] summary");if(e)window.dataLayer.push({event:"closeDetails",detailsTitle:e.textContent||""});else{var o=a.target.closest("summary"),n=a.target.closest("a"),r=a.target.closest("button");o&&window.dataLayer.push({event:"openDetails",detailsTitle:o.textContent||""}),n&&window.dataLayer.push({event:"linkClicked",linkText:n.hasAttribute("title")?n.getAttribute("title")||"":n.textContent||"",class:n.hasAttribute("class")&&n.getAttribute("class")||"",href:n.getAttribute("href")||""}),r&&window.dataLayer.push({event:"buttonClicked",buttonText:r.textContent||"",class:r.hasAttribute("class")&&r.getAttribute("class")||""})}})}window.triggerDynamicEvent=function(a){var e=arguments.length>1&&arguments[1]!==void 0?arguments[1]:"change";console.log("event"),a.hasAttribute("data-change-events")&&e=="change"&&v(a,a.getAttribute("data-change-events")),a.hasAttribute("data-click-events")&&e=="click"&&v(a,a.getAttribute("data-click-events"))};var M=()=>{document.addEventListener("change",a=>{a&&a.target instanceof HTMLElement&&a.target.closest("[data-change-events]")&&v(a.target,a.target.closest("[data-change-events]").getAttribute("data-change-events"))}),document.addEventListener("keyup",a=>{a&&a.target instanceof HTMLElement&&a.target.closest("[data-change-events]")&&v(a.target,a.target.closest("[data-change-events]").getAttribute("data-change-events"))}),document.addEventListener("click",a=>{a&&a.target instanceof HTMLElement&&a.target.closest("[data-click-events]")&&v(a.target,a.target.closest("[data-click-events]").getAttribute("data-click-events"))})},v=(a,e)=>{if(e||(e=a.parentNode.getAttribute("data-change-events")),!e)return!1;Array.from(JSON.parse(e)).forEach(o=>{C(a,o)})},C=(a,e)=>{if("matches"in e)return e.matches=="any"||a.value==e.matches?f(a,e,"if"):f(a,e,"else"),!1;if("in-list"in e){var o=document.querySelector("".concat(e["in-list"],' option[value="').concat(a.value,'"]'));return document.querySelector("".concat(e["in-list"],' option[value="').concat(a.value,'"]'))?f(o,e,"if"):f(o,e,"else"),!1}else"event"in e&&f(a,e,"event")},f=(a,e,o)=>{if(!(o in e))return!1;switch(e[o]){case"hide":if(document.querySelector(e.target)){var n=document.querySelector(e.target);n.classList.add("js-hide"),Array.from(n.querySelectorAll("[data-required]")).forEach(i=>{i.removeAttribute("required")})}break;case"show":if(document.querySelector(e.target)){var r=document.querySelector(e.target);r.classList.remove("js-hide"),Array.from(r.querySelectorAll("[data-required]")).forEach(i=>{i.closest(".js-hide")||i.setAttribute("required","true")})}break;case"populate-form":D(a,e);break;case"dispatchEvent":var s=new Event(e.value);document.querySelector("".concat(e.target)).dispatchEvent(s);break;case"setAttribute":Array.from(document.querySelectorAll("".concat(e.target))).forEach(function(i){i.setAttribute(e.attribute,e.value)});break;case"focus":document.querySelector("".concat(e.target)).focus();break;case"removeAttribute":Array.from(document.querySelectorAll("".concat(e.target))).forEach(function(i){i.removeAttribute(e.attribute)});break;case"updateValue":document.querySelector("".concat(e.target)).value=e.value?e.value:"";var u=new Event("change");document.querySelector("".concat(e.target)).dispatchEvent(u);break;case"submitForm":document.querySelector("".concat(e.target)).submit();break;case"openLink":document.querySelector("".concat(e.target)).value&&(window.location.href=document.querySelector("".concat(e.target)).value);break}},D=function(e,o){var n=JSON.parse(e.getAttribute("data-values")),r=document.querySelector(o.target);if(!n)return!1;Object.keys(n).forEach(s=>{document.getElementById(s)&&document.getElementById(s).tagName=="SPAN"&&(document.getElementById(s).innerHTML=n[s]),r.querySelector('select[name="'.concat(s,'"] [value="').concat(n[s],'"]'))?(r.querySelector('select[name="'.concat(s,'"]')).value=n[s],e.hasAttribute("data-lock-fields")&&(r.querySelector('select[name="'.concat(s,'"]')).disabled=!0)):r.querySelector('input[name="'.concat(s,'"][type="radio"][value="').concat(n[s],'"]'))?(Array.from(r.querySelectorAll('input[name="'.concat(s,'"][type="radio"]'))).forEach(function(u){u.disabled=!0}),r.querySelector('input[name="'.concat(s,'"][type="radio"][value="').concat(n[s],'"]')).checked=!0,r.querySelector('input[name="'.concat(s,'"][type="radio"][value="').concat(n[s],'"]')).disabled=!1):r.querySelector('input[name="'.concat(s,'"]'))&&(r.querySelector('input[name="'.concat(s,'"]')).value=n[s],e.hasAttribute("data-lock-fields")&&r.querySelector('input[name="'.concat(s,'"]')).setAttribute("readonly","true"))})},F=()=>{var a=[],e={},o="These tests run against mark-up and content meant to test how a component or element has been integrated. Making sure that guidance setout in the design system is being followed.";window.integrationTestsIntro=o,console.log("%cVirgin Money Framework Tests","color: #E10A0A; font-size: 24px; font-weight: bold; margin-top: 20px; margin-bottom: 10px;"),console.log("%c"+o,"font-size: 14px; margin-bottom: 20px;"),console.log("%cKey","font-size: 16px;font-weight:bold;"),console.log("%c %cSuccessfully implemented","background-color: green; margin-right: 10px","background-color: transparent"),console.log("%c %cNOT implemented correctly","background-color: red; margin-right: 10px","background-color: transparent"),console.log("%c %cN/A","background-color: grey; margin-right: 10px","background-color: transparent"),console.log("%c %cWarning - No action needed","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("%c ? %cWarning - Action needed","background-color: orange; margin-right: 10px","background-color: transparent"),e={Test:"Each card with in a card deck has a footer with some CTA text via a button link",Total:0,Passes:0,Fails:0},Array.from(document.querySelectorAll(".card-deck .card")).forEach((t,c)=>{if(t.closest(".navbar")==null){var l=t.querySelector(".btn-link");l!=null&&l.textContent.length?e.Passes++:e.Fails++}}),e.Total=e.Passes+e.Fails,a.push(e),e={Test:"Article Decks need to have a background colour applied",Total:0,Passes:0,Fails:0},Array.from(document.querySelectorAll(".article-deck")).forEach((t,c)=>{var l=t.closest(".container");l!=null&&l.matches('[class*="bg-"]')?e.Passes++:e.Fails++}),e.Total=e.Passes+e.Fails,a.push(e),e={Test:"Tabs Carousel has a h2 title with a display-4 class. It should always have a strapline next to it also.",Total:0,Passes:0,Fails:0,Notes:"The site nav contains cards without a footer by design."},Array.from(document.querySelectorAll(".tabs__container--carousel")).forEach((t,c)=>{var l=t.querySelector("h2.display-4:first-child"),d=t.querySelector("h2.display-4:first-child + .strapline");l!=null&&d!=null?e.Passes++:e.Fails++}),e.Total=e.Passes+e.Fails,a.push(e),e={Test:"Split Containers can only have multiple of two columns. So either 2, 4, 6 and so on.",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll(".container.container--split")).forEach((t,c)=>{t.querySelectorAll(':scope > .row > [class*="col"]').length%2===0?e.Passes++:e.Fails++}),e.Total=e.Passes+e.Fails,a.push(e),e={Test:"Statement Containers with only one paragraph or text element should have the class of .h6",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll(".container.container--statement")).forEach((t,c)=>{t.querySelectorAll(":scope > *").length==1&&t.querySelectorAll(":scope > .h6:first-child").length==1?e.Passes++:t.querySelectorAll(":scope > *").length==1&&e.Fails++}),e.Total=e.Passes+e.Fails,a.push(e),e={Test:"CTA Containers title should always have the class of .h1",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll(".container.container--cta")).forEach((t,c)=>{t.querySelectorAll("h2.h1:first-child").length==1?e.Passes++:e.Fails++}),e.Total=e.Passes+e.Fails,a.push(e),console.log("%cImplementation Tests","font-size: 16px;font-weight:bold; margin-top: 20px;"),a.forEach(function(t){var c="green";t.Total===0?c="grey":t.Fails!=0&&(c="red"),t.Type=="warning"&&(c="orange"),console.groupCollapsed("%c %c"+t.Test,"background-color: "+c+"; margin-right: 10px","background-color: transparent"),console.table({Passed:{value:t.Passes},Failed:{value:t.Fails},Total:{value:t.Total}}),t.Notes!=""&&console.log(t.Notes),console.groupEnd()}),window.integrationTests=a;var n=[];e={Test:"There are XX padding classes being used on this page",Type:"warning",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll('[class*="p-"],[class*="pt-"],[class*="pb-"],[class*="pl-"],[class*="pr-"],[class*="px-"],[class*="py-"]')).forEach((t,c)=>{e.Total++}),e.Test=e.Test.replace("XX",e.Total),n.push(e),e={Test:"There are XX margin classes being used on this page",Type:"warning",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll('[class*="m-"],[class*="mt-"],[class*="mb-"],[class*="ml-"],[class*="mr-"],[class*="mx-"],[class*="my-"]')).forEach((t,c)=>{e.Total++}),e.Test=e.Test.replace("XX",e.Total),n.push(e),e={Test:"There are XX text modifier classes being used on this page",Type:"warning",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll('[class*="text-"]')).forEach((t,c)=>{e.Total++}),e.Test=e.Test.replace("XX",e.Total),n.push(e),console.log("%cUtility classes","font-size: 16px;font-weight:bold;"),console.log("The use of certain utility classes is necessary at times but the over use of these classes will lead to design/layout inconsistencies."),n.forEach(function(t){var c="green";t.Total===0?c="grey":t.Fails!=0&&(c="red"),t.Type=="warning"&&(c="orange"),console.groupCollapsed("%c %c"+t.Test,"background-color: "+c+"; margin-right: 10px","background-color: transparent"),console.table({Total:{value:t.Total}}),t.Notes!=""&&console.log(t.Notes),console.groupEnd()}),console.log("%cAccessibility and Best practice Tests","font-size: 16px;font-weight:bold;");var r="green",s=1,u={};Array.from(document.querySelectorAll("h1,h2,h3,h4,h5,h6")).forEach((t,c)=>{var l=1,d=t.textContent;t.matches("h2")?l=2:t.matches("h3")?l=3:t.matches("h4")?l=4:t.matches("h5")?l=5:t.matches("h6")&&(l=6);var h=l-s<=1;c==0&&l!=1&&(h=!1);var p={"Heading Level":l,"In order":h};h?s=l:r="red",u[d]=p}),console.groupCollapsed("%c %cHeadings are in the correct order","background-color: "+r+"; margin-right: 10px","background-color: transparent"),console.table(u),console.groupEnd(),console.groupCollapsed("%c ? %cDo all of the headings make sense out of context and do they introduce the content that follows?","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("You should be able to read the below table and get an understanding of what the page content consists of. The titles should make sense when read in order and child titles should be relevant to the parent title."),console.table(u),console.groupEnd();var i={};Array.from(document.querySelectorAll("a")).forEach((t,c)=>{var l=t.getAttribute("href"),d=t.textContent,h=t.getAttribute("title"),p={Text:d,Title:h};i[l]=p}),console.groupCollapsed("%c ? %cDo all of the links make sense out of context and do they describe the page/section that it links to?","background-color: orange; margin-right: 10px","background-color: transparent"),console.table(i),console.groupEnd();var A={},m={};Array.from(document.querySelectorAll("img[alt]")).forEach((t,c)=>{var l=t.getAttribute("src"),d=t.getAttribute("alt");if(d!=""){var h={Alt:d};A[l]=h}else{var l=t.getAttribute("src"),p={};m[l]=p}}),console.groupCollapsed("%c ? %cDo all of the images with an alt tag make sense out of context and describes the image appropriately?","background-color: orange; margin-right: 10px","background-color: transparent"),console.table(A),console.log("%cImages with an empty alt tag:","font-weight: bold;"),console.table(m),console.groupEnd(),console.groupCollapsed("%c ? %cCan you navigation the page links by tabbing through the page?","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("Are all of the links, input fields and buttons accessible through using the keyboard only?"),console.groupEnd(),console.log("%cLighthouse","font-size: 16px;font-weight:bold;"),console.groupCollapsed("%c ? %cHave you ran a lighthouse report on this page","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("If you are in a modern version of chrome you can a lighthouse report by opening the lighthouse window within the devtools. This should appear as a tab link above."),console.log("This report will give you more detail about the above the tests aswell as extra tests. "),console.groupEnd(),console.log("%cImage sizes","font-size: 16px;font-weight:bold;"),console.log("The below tests work from the current screen size, so if you have your screen at a mobile viewport the scaling and sizes of images will reflect that. It is worth running the image tests in both mobile and desktop views. Simply change the screen size and refresh the page."),Array.from(document.querySelectorAll("img")).forEach((t,c)=>{if(!t.matches("[src*='http']")&&!t.matches("[src$='.svg']")){var l=t.getAttribute("src"),d=new XMLHttpRequest;d.open("HEAD",l,!0),d.onreadystatechange=function(){if(d.readyState==4)if(d.status==200){var h=d.getResponseHeader("Content-Length");if(h>1024){var p=t.naturalWidth,P=t.naturalHeight,S=t.clientWidth,O=t.clientHeight,k=(S/p).toFixed(2)=="0.00"?"Hidden":(S/p).toFixed(2),B=Math.floor(h/1024),E=(h/(p*P)).toFixed(2),b="green",L="green",q="green",x="",T="";E>.5?(b="red",L="red",x="The image has a VERY large 'Bytes per pixel' score, trying optimising the image using https://tinypng.com/."):E>.2&&(b="orange",L="orange",x="The image has a large 'Bytes per pixel' score, trying optimising the image using https://tinypng.com/."),k>1.2?(b="red",q="red",T="The image size is smaller than the size shown on screen this may cause the image to appear blurry and low quality."):k<.5?(b="red",q="red",T="The image size is VERY large compared to the size shown on screen, this means unnecessary kb's are being downloaded. Try resizing the image in photoshop or other image editing software."):k<.8&&(b="orange",q="orange",T="The image size is large compared to the size shown on screen, this means unnecessary kb's are being downloaded. Try resizing the image in photoshop or other image editing software."),console.log("%c %c "+t.src,"background: url("+t.src+"); border: 3px solid "+b+"; line-height: 30px; width: 30px; background-size: cover;","background-color: transparent"),x!=""&&console.log("%c %c"+x,"background-color: "+L+"; margin-right: 10px","background-color: transparent"),T!=""&&console.log("%c %c"+T,"background-color: "+q+"; margin-right: 10px","background-color: transparent"),console.table({Image:{Dimensions:p+"x"+P,"Scaled to":S+"x"+O,Scale:k,Filesize:B+"Kb","Bytes per pixel":E}}),console.groupEnd()}}else console.log("%c %c"+t.src+" not loaded","background-color: red; margin-right: 10px","background-color: transparent")},d.send(null)}});var g=[];if(typeof window.integrationTests<"u"&&window.integrationTests.forEach(function(t){t.Total!=0&&t.Type!="warning"&&g.push(t.Test)}),g.length!=0){var y=document.querySelector("#vmtests-display");if(y!=null){y.innerHTML=`<div class="container">
5
5
  <h2>Integration notes</h2>
@@ -1,5 +1,5 @@
1
1
  /*!
2
- * iamKey v7.8.1
2
+ * iamKey v7.8.2--beta2
3
3
  * Copyright 2022-2026 iamproperty
4
4
  */
5
5
  !function(e){"function"==typeof define&&define.amd?define(e):e()}(function(){"use strict";window.triggerDynamicEvent=function(t){var a=arguments.length>1&&void 0!==arguments[1]?arguments[1]:"change";console.log("event"),t.hasAttribute("data-change-events")&&"change"==a&&e(t,t.getAttribute("data-change-events")),t.hasAttribute("data-click-events")&&"click"==a&&e(t,t.getAttribute("data-click-events"))};var e=(e,a)=>{if(a||(a=e.parentNode.getAttribute("data-change-events")),!a)return!1;Array.from(JSON.parse(a)).forEach(a=>{t(e,a)})},t=(e,t)=>{if("matches"in t)return"any"==t.matches||e.value==t.matches?a(e,t,"if"):a(e,t,"else"),!1;if("in-list"in t){var o=document.querySelector("".concat(t["in-list"],' option[value="').concat(e.value,'"]'));return document.querySelector("".concat(t["in-list"],' option[value="').concat(e.value,'"]'))?a(o,t,"if"):a(o,t,"else"),!1}"event"in t&&a(e,t,"event")},a=(e,t,a)=>{if(!(a in t))return!1;switch(t[a]){case"hide":if(document.querySelector(t.target)){var n=document.querySelector(t.target);n.classList.add("js-hide"),Array.from(n.querySelectorAll("[data-required]")).forEach(e=>{e.removeAttribute("required")})}break;case"show":if(document.querySelector(t.target)){var r=document.querySelector(t.target);r.classList.remove("js-hide"),Array.from(r.querySelectorAll("[data-required]")).forEach(e=>{e.closest(".js-hide")||e.setAttribute("required","true")})}break;case"populate-form":o(e,t);break;case"dispatchEvent":var s=new Event(t.value);document.querySelector("".concat(t.target)).dispatchEvent(s);break;case"setAttribute":Array.from(document.querySelectorAll("".concat(t.target))).forEach(function(e){e.setAttribute(t.attribute,t.value)});break;case"focus":document.querySelector("".concat(t.target)).focus();break;case"removeAttribute":Array.from(document.querySelectorAll("".concat(t.target))).forEach(function(e){e.removeAttribute(t.attribute)});break;case"updateValue":document.querySelector("".concat(t.target)).value=t.value?t.value:"";var l=new Event("change");document.querySelector("".concat(t.target)).dispatchEvent(l);break;case"submitForm":document.querySelector("".concat(t.target)).submit();break;case"openLink":document.querySelector("".concat(t.target)).value&&(window.location.href=document.querySelector("".concat(t.target)).value)}},o=function(e,t){var a=JSON.parse(e.getAttribute("data-values")),o=document.querySelector(t.target);if(!a)return!1;Object.keys(a).forEach(t=>{document.getElementById(t)&&"SPAN"==document.getElementById(t).tagName&&(document.getElementById(t).innerHTML=a[t]),o.querySelector('select[name="'.concat(t,'"] [value="').concat(a[t],'"]'))?(o.querySelector('select[name="'.concat(t,'"]')).value=a[t],e.hasAttribute("data-lock-fields")&&(o.querySelector('select[name="'.concat(t,'"]')).disabled=!0)):o.querySelector('input[name="'.concat(t,'"][type="radio"][value="').concat(a[t],'"]'))?(Array.from(o.querySelectorAll('input[name="'.concat(t,'"][type="radio"]'))).forEach(function(e){e.disabled=!0}),o.querySelector('input[name="'.concat(t,'"][type="radio"][value="').concat(a[t],'"]')).checked=!0,o.querySelector('input[name="'.concat(t,'"][type="radio"][value="').concat(a[t],'"]')).disabled=!1):o.querySelector('input[name="'.concat(t,'"]'))&&(o.querySelector('input[name="'.concat(t,'"]')).value=a[t],e.hasAttribute("data-lock-fields")&&o.querySelector('input[name="'.concat(t,'"]')).setAttribute("readonly","true"))})},n=()=>{var e=[],t={},a="These tests run against mark-up and content meant to test how a component or element has been integrated. Making sure that guidance setout in the design system is being followed.";window.integrationTestsIntro=a,console.log("%cVirgin Money Framework Tests","color: #E10A0A; font-size: 24px; font-weight: bold; margin-top: 20px; margin-bottom: 10px;"),console.log("%c"+a,"font-size: 14px; margin-bottom: 20px;"),console.log("%cKey","font-size: 16px;font-weight:bold;"),console.log("%c %cSuccessfully implemented","background-color: green; margin-right: 10px","background-color: transparent"),console.log("%c %cNOT implemented correctly","background-color: red; margin-right: 10px","background-color: transparent"),console.log("%c %cN/A","background-color: grey; margin-right: 10px","background-color: transparent"),console.log("%c %cWarning - No action needed","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("%c ? %cWarning - Action needed","background-color: orange; margin-right: 10px","background-color: transparent"),t={Test:"Each card with in a card deck has a footer with some CTA text via a button link",Total:0,Passes:0,Fails:0},Array.from(document.querySelectorAll(".card-deck .card")).forEach((e,a)=>{if(null==e.closest(".navbar")){var o=e.querySelector(".btn-link");null!=o&&o.textContent.length?t.Passes++:t.Fails++}}),t.Total=t.Passes+t.Fails,e.push(t),t={Test:"Article Decks need to have a background colour applied",Total:0,Passes:0,Fails:0},Array.from(document.querySelectorAll(".article-deck")).forEach((e,a)=>{var o=e.closest(".container");null!=o&&o.matches('[class*="bg-"]')?t.Passes++:t.Fails++}),t.Total=t.Passes+t.Fails,e.push(t),t={Test:"Tabs Carousel has a h2 title with a display-4 class. It should always have a strapline next to it also.",Total:0,Passes:0,Fails:0,Notes:"The site nav contains cards without a footer by design."},Array.from(document.querySelectorAll(".tabs__container--carousel")).forEach((e,a)=>{var o=e.querySelector("h2.display-4:first-child"),n=e.querySelector("h2.display-4:first-child + .strapline");null!=o&&null!=n?t.Passes++:t.Fails++}),t.Total=t.Passes+t.Fails,e.push(t),t={Test:"Split Containers can only have multiple of two columns. So either 2, 4, 6 and so on.",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll(".container.container--split")).forEach((e,a)=>{e.querySelectorAll(':scope > .row > [class*="col"]').length%2==0?t.Passes++:t.Fails++}),t.Total=t.Passes+t.Fails,e.push(t),t={Test:"Statement Containers with only one paragraph or text element should have the class of .h6",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll(".container.container--statement")).forEach((e,a)=>{1==e.querySelectorAll(":scope > *").length&&1==e.querySelectorAll(":scope > .h6:first-child").length?t.Passes++:1==e.querySelectorAll(":scope > *").length&&t.Fails++}),t.Total=t.Passes+t.Fails,e.push(t),t={Test:"CTA Containers title should always have the class of .h1",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll(".container.container--cta")).forEach((e,a)=>{1==e.querySelectorAll("h2.h1:first-child").length?t.Passes++:t.Fails++}),t.Total=t.Passes+t.Fails,e.push(t),console.log("%cImplementation Tests","font-size: 16px;font-weight:bold; margin-top: 20px;"),e.forEach(function(e){var t="green";0===e.Total?t="grey":0!=e.Fails&&(t="red"),"warning"==e.Type&&(t="orange"),console.groupCollapsed("%c %c"+e.Test,"background-color: "+t+"; margin-right: 10px","background-color: transparent"),console.table({Passed:{value:e.Passes},Failed:{value:e.Fails},Total:{value:e.Total}}),""!=e.Notes&&console.log(e.Notes),console.groupEnd()}),window.integrationTests=e;var o=[];t={Test:"There are XX padding classes being used on this page",Type:"warning",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll('[class*="p-"],[class*="pt-"],[class*="pb-"],[class*="pl-"],[class*="pr-"],[class*="px-"],[class*="py-"]')).forEach((e,a)=>{t.Total++}),t.Test=t.Test.replace("XX",t.Total),o.push(t),t={Test:"There are XX margin classes being used on this page",Type:"warning",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll('[class*="m-"],[class*="mt-"],[class*="mb-"],[class*="ml-"],[class*="mr-"],[class*="mx-"],[class*="my-"]')).forEach((e,a)=>{t.Total++}),t.Test=t.Test.replace("XX",t.Total),o.push(t),t={Test:"There are XX text modifier classes being used on this page",Type:"warning",Total:0,Passes:0,Fails:0,Notes:""},Array.from(document.querySelectorAll('[class*="text-"]')).forEach((e,a)=>{t.Total++}),t.Test=t.Test.replace("XX",t.Total),o.push(t),console.log("%cUtility classes","font-size: 16px;font-weight:bold;"),console.log("The use of certain utility classes is necessary at times but the over use of these classes will lead to design/layout inconsistencies."),o.forEach(function(e){var t="green";0===e.Total?t="grey":0!=e.Fails&&(t="red"),"warning"==e.Type&&(t="orange"),console.groupCollapsed("%c %c"+e.Test,"background-color: "+t+"; margin-right: 10px","background-color: transparent"),console.table({Total:{value:e.Total}}),""!=e.Notes&&console.log(e.Notes),console.groupEnd()}),console.log("%cAccessibility and Best practice Tests","font-size: 16px;font-weight:bold;");var n="green",r=1,s={};Array.from(document.querySelectorAll("h1,h2,h3,h4,h5,h6")).forEach((e,t)=>{var a=1,o=e.textContent;e.matches("h2")?a=2:e.matches("h3")?a=3:e.matches("h4")?a=4:e.matches("h5")?a=5:e.matches("h6")&&(a=6);var l=a-r<=1;0==t&&1!=a&&(l=!1);var c={"Heading Level":a,"In order":l};l?r=a:n="red",s[o]=c}),console.groupCollapsed("%c %cHeadings are in the correct order","background-color: "+n+"; margin-right: 10px","background-color: transparent"),console.table(s),console.groupEnd(),console.groupCollapsed("%c ? %cDo all of the headings make sense out of context and do they introduce the content that follows?","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("You should be able to read the below table and get an understanding of what the page content consists of. The titles should make sense when read in order and child titles should be relevant to the parent title."),console.table(s),console.groupEnd();var l={};Array.from(document.querySelectorAll("a")).forEach((e,t)=>{var a=e.getAttribute("href"),o={Text:e.textContent,Title:e.getAttribute("title")};l[a]=o}),console.groupCollapsed("%c ? %cDo all of the links make sense out of context and do they describe the page/section that it links to?","background-color: orange; margin-right: 10px","background-color: transparent"),console.table(l),console.groupEnd();var c={},i={};Array.from(document.querySelectorAll("img[alt]")).forEach((e,t)=>{var a=e.getAttribute("src"),o=e.getAttribute("alt");if(""!=o){var n={Alt:o};c[a]=n}else a=e.getAttribute("src"),i[a]={}}),console.groupCollapsed("%c ? %cDo all of the images with an alt tag make sense out of context and describes the image appropriately?","background-color: orange; margin-right: 10px","background-color: transparent"),console.table(c),console.log("%cImages with an empty alt tag:","font-weight: bold;"),console.table(i),console.groupEnd(),console.groupCollapsed("%c ? %cCan you navigation the page links by tabbing through the page?","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("Are all of the links, input fields and buttons accessible through using the keyboard only?"),console.groupEnd(),console.log("%cLighthouse","font-size: 16px;font-weight:bold;"),console.groupCollapsed("%c ? %cHave you ran a lighthouse report on this page","background-color: orange; margin-right: 10px","background-color: transparent"),console.log("If you are in a modern version of chrome you can a lighthouse report by opening the lighthouse window within the devtools. This should appear as a tab link above."),console.log("This report will give you more detail about the above the tests aswell as extra tests. "),console.groupEnd(),console.log("%cImage sizes","font-size: 16px;font-weight:bold;"),console.log("The below tests work from the current screen size, so if you have your screen at a mobile viewport the scaling and sizes of images will reflect that. It is worth running the image tests in both mobile and desktop views. Simply change the screen size and refresh the page."),Array.from(document.querySelectorAll("img")).forEach((e,t)=>{if(!e.matches("[src*='http']")&&!e.matches("[src$='.svg']")){var a=e.getAttribute("src"),o=new XMLHttpRequest;o.open("HEAD",a,!0),o.onreadystatechange=function(){if(4==o.readyState)if(200==o.status){var t=o.getResponseHeader("Content-Length");if(t>1024){var a=e.naturalWidth,n=e.naturalHeight,r=e.clientWidth,s=e.clientHeight,l="0.00"==(r/a).toFixed(2)?"Hidden":(r/a).toFixed(2),c=Math.floor(t/1024),i=(t/(a*n)).toFixed(2),u="green",d="green",g="green",h="",p="";i>.5?(u="red",d="red",h="The image has a VERY large 'Bytes per pixel' score, trying optimising the image using https://tinypng.com/."):i>.2&&(u="orange",d="orange",h="The image has a large 'Bytes per pixel' score, trying optimising the image using https://tinypng.com/."),l>1.2?(u="red",g="red",p="The image size is smaller than the size shown on screen this may cause the image to appear blurry and low quality."):l<.5?(u="red",g="red",p="The image size is VERY large compared to the size shown on screen, this means unnecessary kb's are being downloaded. Try resizing the image in photoshop or other image editing software."):l<.8&&(u="orange",g="orange",p="The image size is large compared to the size shown on screen, this means unnecessary kb's are being downloaded. Try resizing the image in photoshop or other image editing software."),console.log("%c %c "+e.src,"background: url("+e.src+"); border: 3px solid "+u+"; line-height: 30px; width: 30px; background-size: cover;","background-color: transparent"),""!=h&&console.log("%c %c"+h,"background-color: "+d+"; margin-right: 10px","background-color: transparent"),""!=p&&console.log("%c %c"+p,"background-color: "+g+"; margin-right: 10px","background-color: transparent"),console.table({Image:{Dimensions:a+"x"+n,"Scaled to":r+"x"+s,Scale:l,Filesize:c+"Kb","Bytes per pixel":i}}),console.groupEnd()}}else console.log("%c %c"+e.src+" not loaded","background-color: red; margin-right: 10px","background-color: transparent")},o.send(null)}});var u=[];if(typeof window.integrationTests<"u"&&window.integrationTests.forEach(function(e){0!=e.Total&&"warning"!=e.Type&&u.push(e.Test)}),0!=u.length){var d=document.querySelector("#vmtests-display");if(null!=d){d.innerHTML='<div class="container">\n <h2>Integration notes</h2>\n <ul id="integrationTests"></ul>\n </div>';var g=document.getElementById("integrationTests");null!=g&&u.forEach(function(e){g.innerHTML+="<li>"+e+"</li>"})}}return null},r=Object.freeze({__proto__:null,default:n});["localhost"].includes(window.location.hostname)&&Promise.resolve().then(function(){return r}).then(e=>{window.iamTest=()=>{n()}}).catch(e=>{console.log(e.message)}),document.addEventListener("DOMContentLoaded",()=>{return t=void 0,a=void 0,n=function*(){var t,a;window.dataLayer=window.dataLayer||[],window.dataLayer.push({event:"Pageview",pageTitle:document.title}),document.addEventListener("click",e=>{var t=e.target.closest("[open] summary");if(t)window.dataLayer.push({event:"closeDetails",detailsTitle:t.textContent||""});else{var a=e.target.closest("summary"),o=e.target.closest("a"),n=e.target.closest("button");a&&window.dataLayer.push({event:"openDetails",detailsTitle:a.textContent||""}),o&&window.dataLayer.push({event:"linkClicked",linkText:o.hasAttribute("title")?o.getAttribute("title")||"":o.textContent||"",class:o.hasAttribute("class")&&o.getAttribute("class")||"",href:o.getAttribute("href")||""}),n&&window.dataLayer.push({event:"buttonClicked",buttonText:n.textContent||"",class:n.hasAttribute("class")&&n.getAttribute("class")||""})}}),document.addEventListener("change",t=>{t&&t.target instanceof HTMLElement&&t.target.closest("[data-change-events]")&&e(t.target,t.target.closest("[data-change-events]").getAttribute("data-change-events"))}),document.addEventListener("keyup",t=>{t&&t.target instanceof HTMLElement&&t.target.closest("[data-change-events]")&&e(t.target,t.target.closest("[data-change-events]").getAttribute("data-change-events"))}),document.addEventListener("click",t=>{t&&t.target instanceof HTMLElement&&t.target.closest("[data-click-events]")&&e(t.target,t.target.closest("[data-click-events]").getAttribute("data-click-events"))}),(a=document.body).classList.add("js-enabled"),(-1!==navigator.userAgent.indexOf("MSIE")||navigator.appVersion.indexOf("Trident/")>0)&&a.classList.add("ie"),t=function(e){var t=document.querySelector('label[for="'.concat(e.replace("#",""),'"]')),a=document.querySelector(e+" summary"),o=document.querySelector("dialog".concat(e)),n=document.querySelector("detail".concat(e));t instanceof HTMLElement?t.click():a instanceof HTMLElement?a.click():o instanceof HTMLElement?o.showModal():n instanceof HTMLElement&&n.addAttribute("open")},location.hash&&t(location.hash),window.addEventListener("hashchange",function(){t(location.hash)},!1),addEventListener("popstate",e=>{if(e&&e.state&&e.state.type&&"pagination"==e.state.type){var t=document.querySelector("#".concat(e.state.form)),a=document.querySelector("#".concat(e.state.form," [data-pagination]"));a?a.value=e.state.page:t.innerHTML+='<input name="page" type="hidden" data-pagination="true" value="'.concat(e.state.page,'" />'),t.dispatchEvent(new Event("submit"))}}),Array.from(document.querySelectorAll("form")).forEach(e=>{e?.closest("iam-form")||e.addEventListener("submit",t=>{var a;e.querySelector(":invalid")&&(e.classList.add("was-validated"),null===(a=e?.querySelector("input:invalid"))||void 0===a||a.scrollIntoView(),t.preventDefault())})}),document.addEventListener("click",e=>{var t;if(e&&e.target instanceof HTMLElement&&e.target.matches("form button:not([type=button])")){var a=e.target.closest("form");a?.closest("iam-form")||(Array.from(a.querySelectorAll("[data-password-type]")).forEach(e=>{e.setAttribute("type","password")}),(a.querySelector(":invalid")||a.querySelector('.pwd-checker[data-strength="1"]')||a.querySelector('.pwd-checker[data-strength="2"]'))&&(a.classList.add("was-validated"),null===(t=a?.querySelector("input:invalid"))||void 0===t||t.scrollIntoView(),e.preventDefault()),a.querySelector("iam-multiselect[data-is-required][data-error]")&&(a.classList.add("was-validated"),e.preventDefault()))}}),document.addEventListener("keydown",e=>{"Escape"===e.key&&document.querySelector(".dialog--transactional[open], .dialog--acknowledgement[open]")&&(e.preventDefault(),e.stopPropagation())}),Array.from(document.querySelectorAll("label progress")).forEach(e=>{e.closest("label").setAttribute("data-percent",e.getAttribute("value"))})},new((o=void 0)||(o=Promise))(function(e,r){function s(e){try{c(n.next(e))}catch(e){r(e)}}function l(e){try{c(n.throw(e))}catch(e){r(e)}}function c(t){t.done?e(t.value):function(e){return e instanceof o?e:new o(function(t){t(e)})}(t.value).then(s,l)}c((n=n.apply(t,a||[])).next())});var t,a,o,n})});
@@ -107,6 +107,8 @@
107
107
  font-weight: normal !important;
108
108
  font-size: var(--card-body-fs, 1rem) !important;
109
109
  line-height: var(--card-body-lh, inherit) !important;
110
+ overflow-x: clip;
111
+ text-overflow: ellipsis;
110
112
  }
111
113
  ::slotted(small) {
112
114
  padding-top: var(--card-gap, 1rem) !important;
@@ -29,7 +29,6 @@
29
29
  width: 90vw;
30
30
  max-width: 20.25rem; /* col-width*4 324px */
31
31
 
32
- /* height: 90vh; */
33
32
  max-height: min(90vh, 47rem);
34
33
  overscroll-behavior: contain;
35
34
  overflow-y: hidden;
@@ -80,7 +79,7 @@
80
79
  padding-inline: var(--dialog-padding);
81
80
  padding-block: calc(var(--dialog-padding) - 0.75rem);
82
81
  height: 100%;
83
- max-height: 100%;
82
+ max-height: min(90vh, 47rem);
84
83
  overflow: auto;
85
84
  overscroll-behavior: contain;
86
85
  position: relative;
@@ -55,6 +55,8 @@ class iamMultiselect extends HTMLElement {
55
55
  const ajaxURL = this.getAttribute('data-url');
56
56
  innerLabel.innerHTML = multiselect.getAttribute('data-label');
57
57
 
58
+ const minLength = this.hasAttribute('data-min') ? this.getAttribute('data-min') : 3;
59
+
58
60
  // Make sure the dropdown options are set
59
61
  Array.from(this.querySelectorAll(':scope > label')).forEach((label) => {
60
62
  label.classList.add('dropdown__option');
@@ -95,6 +97,7 @@ class iamMultiselect extends HTMLElement {
95
97
 
96
98
  // Set the correct attributes
97
99
  function setItem(inputToSet): void {
100
+
98
101
  if (inputToSet.checked == false) {
99
102
  inputToSet.closest('label').removeAttribute('slot');
100
103
  inputToSet.closest('label').removeAttribute('style');
@@ -127,6 +130,7 @@ class iamMultiselect extends HTMLElement {
127
130
  }
128
131
  }
129
132
 
133
+
130
134
  // Set on load
131
135
  Array.from(multiselect.querySelectorAll(`label input[type="checkbox"]:checked`)).forEach((checkbox) => {
132
136
  setItem(checkbox);
@@ -134,11 +138,9 @@ class iamMultiselect extends HTMLElement {
134
138
 
135
139
  search.addEventListener('input', () => {
136
140
 
137
- if (multiselect.hasAttribute('data-url')) {
141
+ if (multiselect.hasAttribute('data-url') && search.value.length == minLength) {
138
142
 
139
- if (search.value.length == 3) {
140
- searchAjax(multiselect, search, filterList);
141
- }
143
+ searchAjax(multiselect, search, filterList);
142
144
  } else {
143
145
 
144
146
  filterList(multiselect, search);
@@ -278,7 +280,25 @@ class iamMultiselect extends HTMLElement {
278
280
  else activeElement.checked = false;
279
281
  }
280
282
 
281
- setItem(activeElement);
283
+ console.log(search);
284
+
285
+ if (activeElement.getAttribute('type') != 'checkbox') {
286
+ if (multiselect.querySelector(`input[type="checkbox"][value="${search.value}" i]`)) {
287
+ multiselect.querySelector(`input[type="checkbox"][value="${search.value}" i]`).checked = true;
288
+
289
+ setItem(multiselect.querySelector(`input[type="checkbox"][value="${search.value}" i]`));
290
+ }
291
+ search.value = '';
292
+
293
+ Array.from(multiselect.querySelectorAll(`label input[type="checkbox"]`)).forEach((checkbox) => {
294
+ setItem(checkbox);
295
+ });
296
+ }
297
+ else {
298
+
299
+ setItem(activeElement);
300
+ }
301
+
282
302
  search.focus();
283
303
 
284
304
  break;
@@ -51,9 +51,9 @@ class iamSearch extends HTMLElement {
51
51
  const displaySchema = this.hasAttribute('data-display-schema') ? this.getAttribute('data-display-schema') : 'label';
52
52
  const loopSchema = this.hasAttribute('data-schema') ? this.getAttribute('data-schema') : '';
53
53
  let datalist = this.querySelector('datalist');
54
- let minLength = 0;
54
+ let minLength = this.hasAttribute('data-min-length') ? this.getAttribute('data-min-length') : 0;
55
55
 
56
- if (searchWrapper.hasAttribute('data-url')) {
56
+ if (searchWrapper.hasAttribute('data-url') && !this.hasAttribute('data-min-length')) {
57
57
 
58
58
  minLength = 3;
59
59
  }
@@ -0,0 +1,33 @@
1
+ import './test-globals.ts';
2
+ import { describe, expect, it } from './test.ts';
3
+ import { createElement, installTestDom } from './test-dom.ts';
4
+ import { append } from './test-utils.ts';
5
+ import advancedSelect from './advanced-select.ts';
6
+
7
+ installTestDom();
8
+
9
+ describe('Advanced select module', () => {
10
+ it('moves native datalist wiring onto component state', () => {
11
+ const component = createElement('iam-advanced-select');
12
+ component.shadowRoot = createElement('shadow-root');
13
+ const shadowInputWrapper = createElement('div', { class: 'input__wrapper' });
14
+ const shadowClear = createElement('button', { class: 'clear-search' });
15
+ append(shadowInputWrapper, shadowClear);
16
+ append(component.shadowRoot, shadowInputWrapper);
17
+
18
+ const inputWrapper = createElement('div', { class: 'input__wrapper' });
19
+ const input = createElement('input', { list: 'choices', placeholder: 'Pick one' });
20
+ const datalist = createElement('datalist', { id: 'choices' });
21
+ append(datalist, createElement('option', { value: 'A' }));
22
+ append(inputWrapper, input);
23
+ append(component, inputWrapper, datalist);
24
+
25
+ advancedSelect(component, input, datalist);
26
+
27
+ expect(input.getAttribute('data-list') === 'choices');
28
+ expect(input.getAttribute('list') === '');
29
+ expect(input.classList.contains('empty'));
30
+ expect(component.classList.contains('has-empty-input'));
31
+ expect(datalist.getAttribute('slot') === 'datalist');
32
+ });
33
+ });
@@ -7,8 +7,12 @@ function advancedSelect(advancedSelect, displayInputField, datalist, isSearch =
7
7
 
8
8
  if(advancedSelect.querySelector('.suffix')){
9
9
  advancedSelect.querySelector('.suffix')?.setAttribute('slot','suffix');
10
- advancedSelect.shadowRoot.querySelector('.suffix')?.innerHTML = '<slot name="suffix"></slot>';
11
- advancedSelect.shadowRoot.querySelector('.suffix')?.classList = "";
10
+
11
+ const shadowSuffix = advancedSelect.shadowRoot.querySelector('.suffix');
12
+ if (shadowSuffix) {
13
+ shadowSuffix.innerHTML = '<slot name="suffix"></slot>';
14
+ shadowSuffix.className = '';
15
+ }
12
16
  }
13
17
 
14
18
 
@@ -0,0 +1,25 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { append } from './test-utils.ts';
4
+ import createAppliedFilters from './applied-filters.ts';
5
+
6
+ installTestDom();
7
+
8
+ describe('Applied filters module', () => {
9
+ it('creates a filter tag for an initial text input value', () => {
10
+ const container = createElement('iam-applied-filters');
11
+ const filters = createElement('div');
12
+ const input = createElement('input', {
13
+ name: 'search',
14
+ value: 'auction',
15
+ dataFilterText: 'Search $value',
16
+ });
17
+
18
+ append(container, input);
19
+ createAppliedFilters(container, filters);
20
+
21
+ expect(filters.children.length === 1);
22
+ expect(filters.children[0].getAttribute('data-name') === 'search');
23
+ expect(filters.children[0].innerHTML === 'Search auction');
24
+ });
25
+ });
@@ -0,0 +1,24 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { append } from './test-utils.ts';
4
+ import { setupCard } from './card.module.ts';
5
+
6
+ installTestDom();
7
+
8
+ describe('Card module', () => {
9
+ it('hydrates image, total and empty badge state', () => {
10
+ const card = createElement('iam-card', { dataImage: '/test.jpg', dataTotal: '42' });
11
+ card.shadowRoot = createElement('shadow-root');
12
+ const head = createElement('div', { class: 'card__head' });
13
+ const body = createElement('div', { class: 'card__body' });
14
+ const badges = createElement('div', { class: 'card__badges' });
15
+ append(card.shadowRoot, head, badges, body);
16
+
17
+ setupCard(card);
18
+
19
+ expect(card.classList.contains('card'));
20
+ expect(head.innerHTML.includes('/test.jpg'));
21
+ expect(body.innerHTML.includes('42'));
22
+ expect(badges.classList.contains('empty'));
23
+ });
24
+ });
@@ -34,7 +34,8 @@ export const setupCard = (cardComponent: any): void => {
34
34
  `<div class="card__total">${cardComponent.getAttribute('data-total')}</div>`
35
35
  );
36
36
  else {
37
- cardBody?.querySelector('.card__total')?.innerHTML = cardComponent.getAttribute('data-total');
37
+ const cardTotal = cardBody?.querySelector('.card__total');
38
+ if (cardTotal) cardTotal.innerHTML = cardComponent.getAttribute('data-total');
38
39
  }
39
40
  } else if (cardComponent.querySelector('[slot="total-icon"]')) {
40
41
  cardBody.insertAdjacentHTML('beforeend', `<div class="card__total"><slot name="total-icon"></slot></div>`);
@@ -0,0 +1,27 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { append } from './test-utils.ts';
4
+ import { generatePipsHTML, generateThumbnailList, getProgressMax } from './carousel.ts';
5
+
6
+ installTestDom();
7
+
8
+ describe('Carousel module', () => {
9
+ it('builds thumbnail-aware pips and progress max values', () => {
10
+ const carousel = createElement('iam-carousel');
11
+ append(
12
+ carousel,
13
+ createElement('div', { dataThumbnail: '/one.jpg' }),
14
+ createElement('iam-card'),
15
+ createElement('div', { dataThumbnail: '/three.jpg' })
16
+ );
17
+
18
+ const thumbnails = generateThumbnailList(carousel);
19
+ const html = generatePipsHTML(carousel, thumbnails);
20
+
21
+ expect(thumbnails[0] === '/one.jpg');
22
+ expect(thumbnails[2] === '/three.jpg');
23
+ expect(html.includes('control-1 has-thumbnail'));
24
+ expect(html.includes('Slide 2'));
25
+ expect(getProgressMax(7, 3) === 7);
26
+ });
27
+ });
@@ -0,0 +1,26 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { append } from './test-utils.ts';
4
+ import { addClasses } from './chart.module.ts';
5
+
6
+ installTestDom();
7
+
8
+ describe('Chart module', () => {
9
+ it('reads chart data and applies configured colour variables', () => {
10
+ const chartElement = createElement('iam-chart', {
11
+ dataMin: '0',
12
+ dataMax: '100',
13
+ dataYaxis: '0,50,100',
14
+ dataGuidelines: '25,75',
15
+ 'data-colour-1': 'success',
16
+ });
17
+ const chartOuter = createElement('div');
18
+ const coloured = createElement('span', { 'data-colour-2': 'warning' });
19
+ append(chartOuter, coloured);
20
+
21
+ addClasses(chartElement, chartOuter);
22
+
23
+ expect(chartElement.style.getPropertyValue('--chart-colour-1') === 'var(--chart-colour-success)');
24
+ expect(coloured.style.getPropertyValue('--chart-colour-2-hover') === 'var(--chart-colour-warning-hover)');
25
+ });
26
+ });
@@ -0,0 +1,19 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { createTypeSwitcher } from './chart.ts';
4
+
5
+ installTestDom();
6
+
7
+ describe('Legacy chart module', () => {
8
+ it('creates chart option controls for legacy charts', () => {
9
+ const chartElement = createElement('iam-chart', { dataTypes: 'bar,line', dataType: 'line' });
10
+ const chartKey = createElement('div', { class: 'chart__key' });
11
+ const chartOptions = createElement('div', { class: 'chart__options' });
12
+
13
+ createTypeSwitcher(chartElement, chartKey, chartOptions);
14
+
15
+ expect(chartOptions.innerHTML.includes('for="chart-'));
16
+ expect(chartOptions.innerHTML.includes('bar'));
17
+ expect(chartOptions.innerHTML.includes('line'));
18
+ });
19
+ });
@@ -0,0 +1,24 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { append } from './test-utils.ts';
4
+ import { transformButtons } from './content.ts';
5
+
6
+ const { document } = installTestDom();
7
+
8
+ describe('Content module', () => {
9
+ it('transforms WordPress button wrappers into direct links', () => {
10
+ const parent = createElement('div');
11
+ const buttons = createElement('div', { class: 'wp-block-buttons' });
12
+ const button = createElement('div', { class: 'btn btn-primary wp-block-button' });
13
+ const link = createElement('a', { href: '/test' }, 'Open');
14
+ append(button, link);
15
+ append(buttons, button);
16
+ append(parent, buttons);
17
+ append(document.body, parent);
18
+
19
+ transformButtons();
20
+
21
+ expect(link.getAttribute('class') === 'btn btn-primary wp-block-button');
22
+ expect(parent.children[0].localName === 'fragment');
23
+ });
24
+ });
@@ -0,0 +1,15 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { installTestDom } from './test-dom.ts';
3
+ import createDataLayer from './data-layer.ts';
4
+
5
+ const { window } = installTestDom();
6
+
7
+ describe('Data layer module', () => {
8
+ it('pushes the initial pageview event', () => {
9
+ window.dataLayer = [];
10
+ createDataLayer();
11
+
12
+ expect(window.dataLayer[0].event === 'Pageview');
13
+ expect(window.dataLayer[0].pageTitle === 'Test page');
14
+ });
15
+ });
@@ -0,0 +1,17 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { createDialog } from './dialogs.ts';
4
+
5
+ installTestDom();
6
+
7
+ describe('Dialogs module', () => {
8
+ it('wraps dialog content and adds a close button', () => {
9
+ const dialog = createElement('dialog');
10
+ dialog.innerHTML = '<p>Body</p>';
11
+
12
+ createDialog(dialog);
13
+
14
+ expect(dialog.innerHTML.includes('mh-lg'));
15
+ expect(dialog.innerHTML.includes('dialog__close'));
16
+ });
17
+ });
@@ -0,0 +1,20 @@
1
+ import { describe, expect, it } from './test.ts';
2
+ import { createElement, installTestDom } from './test-dom.ts';
3
+ import { append } from './test-utils.ts';
4
+ import drawer from './drawer.ts';
5
+
6
+ const { document } = installTestDom();
7
+
8
+ describe('Drawer module', () => {
9
+ it('marks the drawer end as in-view and closes the drawer toggle', () => {
10
+ const toggle = createElement('input', { id: 'showDrawer' });
11
+ toggle.checked = true;
12
+ const drawerEnd = createElement('div', { id: 'drawer-end' });
13
+ append(document.body, toggle, drawerEnd);
14
+
15
+ drawer();
16
+
17
+ expect(drawerEnd.classList.contains('in-view'));
18
+ expect(toggle.checked === false);
19
+ });
20
+ });