@aurodesignsystem-dev/auro-formkit 0.0.0-pr1488.2 → 0.0.0-pr1489.0

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 (102) hide show
  1. package/components/bibtemplate/dist/auro-bibtemplate.d.ts +7 -0
  2. package/components/bibtemplate/dist/index.js +9 -1
  3. package/components/bibtemplate/dist/registered.js +9 -1
  4. package/components/checkbox/demo/customize.min.js +1 -1
  5. package/components/checkbox/demo/getting-started.min.js +1 -1
  6. package/components/checkbox/demo/index.min.js +1 -1
  7. package/components/checkbox/demo/pages.json +1 -1
  8. package/components/checkbox/demo/why-checkbox.html +57 -0
  9. package/components/checkbox/demo/why-checkbox.md +86 -0
  10. package/components/checkbox/dist/index.js +1 -1
  11. package/components/checkbox/dist/registered.js +1 -1
  12. package/components/combobox/demo/customize.min.js +234 -16
  13. package/components/combobox/demo/getting-started.min.js +234 -16
  14. package/components/combobox/demo/index.min.js +234 -16
  15. package/components/combobox/demo/pages.json +1 -1
  16. package/components/combobox/demo/why-combobox.html +57 -0
  17. package/components/combobox/demo/why-combobox.md +113 -0
  18. package/components/combobox/dist/index.js +234 -16
  19. package/components/combobox/dist/registered.js +234 -16
  20. package/components/counter/demo/customize.min.js +233 -15
  21. package/components/counter/demo/index.min.js +233 -15
  22. package/components/counter/demo/keyboard-behavior.md +1 -0
  23. package/components/counter/demo/pages.json +1 -1
  24. package/components/counter/demo/why-counter.html +57 -0
  25. package/components/counter/demo/why-counter.md +108 -0
  26. package/components/counter/dist/index.js +10 -2
  27. package/components/counter/dist/registered.js +10 -2
  28. package/components/datepicker/demo/accessibility.md +51 -3
  29. package/components/datepicker/demo/api.md +11 -2
  30. package/components/datepicker/demo/customize.html +2 -0
  31. package/components/datepicker/demo/customize.js +19 -0
  32. package/components/datepicker/demo/customize.md +72 -8
  33. package/components/datepicker/demo/customize.min.js +26029 -0
  34. package/components/datepicker/demo/design.md +3 -1
  35. package/components/datepicker/demo/index.js +5 -1
  36. package/components/datepicker/demo/index.md +83 -2
  37. package/components/datepicker/demo/index.min.js +1564 -96
  38. package/components/datepicker/demo/keyboard-behavior.md +201 -2
  39. package/components/datepicker/demo/pages.json +1 -1
  40. package/components/datepicker/demo/voiceover.md +19 -12
  41. package/components/datepicker/demo/why-datepicker.html +57 -0
  42. package/components/datepicker/demo/why-datepicker.md +133 -0
  43. package/components/datepicker/dist/index.js +1489 -97
  44. package/components/datepicker/dist/registered.js +1489 -97
  45. package/components/datepicker/dist/src/auro-calendar-cell.d.ts +66 -1
  46. package/components/datepicker/dist/src/auro-calendar-month.d.ts +28 -0
  47. package/components/datepicker/dist/src/auro-calendar.d.ts +100 -0
  48. package/components/datepicker/dist/src/auro-datepicker.d.ts +88 -0
  49. package/components/datepicker/dist/src/datepickerKeyboardStrategy.d.ts +5 -3
  50. package/components/dropdown/demo/accessibility.md +11 -0
  51. package/components/dropdown/demo/api.md +1 -0
  52. package/components/dropdown/demo/customize.md +3 -0
  53. package/components/dropdown/demo/customize.min.js +223 -13
  54. package/components/dropdown/demo/getting-started.min.js +223 -13
  55. package/components/dropdown/demo/index.min.js +223 -13
  56. package/components/dropdown/demo/keyboard-behavior.md +1 -0
  57. package/components/dropdown/demo/pages.json +1 -1
  58. package/components/dropdown/demo/why-dropdown.html +57 -0
  59. package/components/dropdown/demo/why-dropdown.md +97 -0
  60. package/components/dropdown/dist/auro-dropdown.d.ts +33 -1
  61. package/components/dropdown/dist/index.js +223 -13
  62. package/components/dropdown/dist/registered.js +223 -13
  63. package/components/form/demo/customize.min.js +2191 -145
  64. package/components/form/demo/getting-started.min.js +2191 -145
  65. package/components/form/demo/index.min.js +2191 -145
  66. package/components/form/demo/pages.json +1 -1
  67. package/components/form/demo/registerDemoDeps.min.js +2191 -145
  68. package/components/form/demo/why-form.html +57 -0
  69. package/components/form/demo/why-form.md +101 -0
  70. package/components/input/demo/customize.min.js +1 -1
  71. package/components/input/demo/getting-started.min.js +1 -1
  72. package/components/input/demo/index.min.js +1 -1
  73. package/components/input/demo/pages.json +1 -1
  74. package/components/input/demo/why-input.html +57 -0
  75. package/components/input/demo/why-input.md +121 -0
  76. package/components/input/dist/index.js +1 -1
  77. package/components/input/dist/registered.js +1 -1
  78. package/components/menu/demo/pages.json +1 -1
  79. package/components/menu/demo/why-menu.html +57 -0
  80. package/components/menu/demo/why-menu.md +104 -0
  81. package/components/radio/demo/customize.min.js +2186 -0
  82. package/components/radio/demo/demo-support.min.js +55807 -0
  83. package/components/radio/demo/getting-started.js +1 -1
  84. package/components/radio/demo/getting-started.md +1 -1
  85. package/components/radio/demo/getting-started.min.js +2205 -0
  86. package/components/radio/demo/index.min.js +1 -1
  87. package/components/radio/demo/pages.json +1 -1
  88. package/components/radio/demo/why-radio.html +57 -0
  89. package/components/radio/demo/why-radio.md +92 -0
  90. package/components/radio/dist/index.js +1 -1
  91. package/components/radio/dist/registered.js +1 -1
  92. package/components/select/demo/customize.min.js +233 -15
  93. package/components/select/demo/getting-started.min.js +233 -15
  94. package/components/select/demo/index.min.js +233 -15
  95. package/components/select/demo/keyboard-behavior.md +1 -0
  96. package/components/select/demo/pages.json +1 -1
  97. package/components/select/demo/why-select.html +57 -0
  98. package/components/select/demo/why-select.md +128 -0
  99. package/components/select/dist/index.js +233 -15
  100. package/components/select/dist/registered.js +233 -15
  101. package/custom-elements.json +745 -28
  102. package/package.json +2 -2
@@ -3243,7 +3243,7 @@ function applyKeyboardStrategy(component, strategy, options = {}) {
3243
3243
  });
3244
3244
  }
3245
3245
 
3246
- var styleCss$2 = css`:host{position:fixed;z-index:var(--depth-tooltip, 400);display:none;isolation:isolate}:host dialog{width:auto;max-width:none;height:auto;max-height:none;padding:0;border:none;margin:0;outline:none;transform:translateZ(0)}:host dialog::backdrop{background:transparent}:host(:not([isfullscreen])) dialog{position:relative;inset:unset}:host(:not([isfullscreen])) .container.shape-box{border-radius:unset}:host(:not([isfullscreen])) .container[class*=shape-pill],:host(:not([isfullscreen])) .container[class*=shape-snowflake]{border-radius:30px}:host(:not([isfullscreen])) .container[class*=shape-rounded]{border-radius:16px}:host(:not([matchWidth])) .container{min-width:fit-content}:host([isfullscreen]){position:fixed;top:0;left:0}:host([isfullscreen]) .container{width:100dvw;max-width:none;height:100dvh;max-height:none;border-radius:unset;margin-top:0;box-shadow:unset;overscroll-behavior:contain}:host([isfullscreen]) .container::backdrop{background:var(--ds-color-background-primary, #fff)}:host(:popover-open){position:fixed;overflow:visible;padding:0;border:none;margin:0;background:transparent;inset:unset;outline:none}:host([data-show]){display:flex}:host([common]:not([isfullscreen])) .container,:host([rounded]:not([isfullscreen])) .container{border-radius:var(--ds-border-radius, 0.375rem)}:host([common][isfullscreen]) .container,:host([rounded][isfullscreen]) .container{border-radius:unset;box-shadow:unset}.container{display:inline-block;overflow:auto;box-sizing:border-box;border-radius:var(--ds-border-radius, 0.375rem);margin:var(--ds-size-50, 0.25rem) 0}.util_displayHiddenVisually{position:absolute;overflow:hidden;width:1px;height:1px;padding:0;border:0;margin:-1px;clip-path:inset(50%);white-space:nowrap}`;
3246
+ var styleCss$2 = css`:host{position:fixed;z-index:var(--depth-tooltip, 400);display:none;isolation:isolate}:host dialog{width:auto;max-width:none;height:auto;max-height:none;padding:0;border:none;margin:0;outline:none;transform:translateZ(0)}:host dialog::backdrop{background:transparent}:host(:not([isfullscreen])) dialog{position:relative;inset:unset}:host(:not([isfullscreen])) .container.shape-box{border-radius:unset}:host(:not([isfullscreen])) .container[class*=shape-pill],:host(:not([isfullscreen])) .container[class*=shape-snowflake]{border-radius:30px}:host(:not([isfullscreen])) .container[class*=shape-rounded]{border-radius:16px}:host([desktopmodal]:popover-open)::backdrop{background:transparent}:host(:not([matchWidth])) .container{min-width:fit-content}:host([isfullscreen]){position:fixed;top:0;left:0}:host([isfullscreen]) .container{width:100dvw;max-width:none;height:100dvh;max-height:none;border-radius:unset;margin-top:0;box-shadow:unset;overscroll-behavior:contain}:host([isfullscreen]) .container::backdrop{background:var(--ds-color-background-primary, #fff)}:host(:popover-open){position:fixed;overflow:visible;padding:0;border:none;margin:0;background:transparent;inset:unset;outline:none}:host([data-show]){display:flex}:host([common]:not([isfullscreen])) .container,:host([rounded]:not([isfullscreen])) .container{border-radius:var(--ds-border-radius, 0.375rem)}:host([common][isfullscreen]) .container,:host([rounded][isfullscreen]) .container{border-radius:unset;box-shadow:unset}.container{display:inline-block;overflow:auto;box-sizing:border-box;border-radius:var(--ds-border-radius, 0.375rem);margin:var(--ds-size-50, 0.25rem) 0}.util_displayHiddenVisually{position:absolute;overflow:hidden;width:1px;height:1px;padding:0;border:0;margin:-1px;clip-path:inset(50%);white-space:nowrap}`;
3247
3247
 
3248
3248
  var colorCss$2 = css`.container{background-color:var(--ds-auro-dropdownbib-container-color);box-shadow:var(--ds-auro-dropdownbib-boxshadow-color);color:var(--ds-auro-dropdownbib-text-color)}`;
3249
3249
 
@@ -3874,7 +3874,7 @@ class AuroHelpText extends LitElement {
3874
3874
  }
3875
3875
  }
3876
3876
 
3877
- var formkitVersion = '202606011708';
3877
+ var formkitVersion = '202606011856';
3878
3878
 
3879
3879
  class AuroElement extends LitElement {
3880
3880
  static get properties() {
@@ -4054,6 +4054,7 @@ class AuroDropdown extends AuroElement {
4054
4054
  _intializeDefaults() {
4055
4055
  this.appearance = 'default';
4056
4056
  this.chevron = false;
4057
+ this.desktopModal = false;
4057
4058
  this.disabled = false;
4058
4059
  this.disableKeyboardHandling = false;
4059
4060
  this.error = false;
@@ -4234,6 +4235,14 @@ class AuroDropdown extends AuroElement {
4234
4235
  reflect: true
4235
4236
  },
4236
4237
 
4238
+ /**
4239
+ * If declared, the dropdown will behave as a modal dialog when in a desktop viewport size.
4240
+ */
4241
+ desktopModal: {
4242
+ type: Boolean,
4243
+ reflect: true
4244
+ },
4245
+
4237
4246
  /**
4238
4247
  * If declared, the dropdown will only show by calling the API .show() public method.
4239
4248
  */
@@ -4521,6 +4530,15 @@ class AuroDropdown extends AuroElement {
4521
4530
 
4522
4531
  disconnectedCallback() {
4523
4532
  super.disconnectedCallback();
4533
+ this._clearPageInert();
4534
+ if (this._bibTabHandler) {
4535
+ this.removeEventListener('keydown', this._bibTabHandler);
4536
+ this._bibTabHandler = undefined;
4537
+ }
4538
+ if (this.focusTrap) {
4539
+ this.focusTrap.disconnect();
4540
+ this.focusTrap = undefined;
4541
+ }
4524
4542
  if (this.floater) {
4525
4543
  this.floater.hideBib('disconnect');
4526
4544
  this.floater.disconnect();
@@ -4548,19 +4566,45 @@ class AuroDropdown extends AuroElement {
4548
4566
  if (this.isPopoverVisible) {
4549
4567
  // Fullscreen: use showModal() for native accessibility (inert outside, focus trap)
4550
4568
  // Desktop: use show() for Floating UI positioning + FocusTrap for focus management
4551
- const useModal = this.isBibFullscreen;
4552
- this.bibElement.value.open(useModal);
4569
+ this.bibElement.value.open(this.isBibFullscreen);
4570
+ this.updateFocusTrap();
4571
+
4572
+ // Desktop modal: make siblings inert so content outside is not interactive
4573
+ if (this.desktopModal && !this.isBibFullscreen) {
4574
+ this._setPageInert();
4575
+ }
4553
4576
  } else {
4554
4577
  this.bibElement.value.close();
4578
+ this._clearPageInert();
4555
4579
  }
4556
4580
  }
4557
4581
 
4558
4582
  // When fullscreen strategy changes while open, re-open dialog with correct mode
4559
4583
  // (e.g. resizing from desktop → mobile while dropdown is open)
4560
4584
  if (changedProperties.has('isBibFullscreen') && this.isPopoverVisible && this.bibElement.value) {
4561
- const useModal = this.isBibFullscreen;
4562
4585
  this.bibElement.value.close();
4563
- this.bibElement.value.open(useModal);
4586
+ this.bibElement.value.open(this.isBibFullscreen);
4587
+
4588
+ // Re-initialize focus management for the new strategy
4589
+ this.updateFocusTrap();
4590
+
4591
+ // Toggle inert: desktop modal needs it, fullscreen showModal() handles it natively
4592
+ if (this.desktopModal && !this.isBibFullscreen) {
4593
+ this._setPageInert();
4594
+ } else {
4595
+ this._clearPageInert();
4596
+ }
4597
+ }
4598
+
4599
+ // Handle desktopModal toggled while the dropdown is already open.
4600
+ // Re-initialize focus trapping and page inert state to match the new mode.
4601
+ if (changedProperties.has('desktopModal') && this.isPopoverVisible && !this.isBibFullscreen) {
4602
+ this.updateFocusTrap();
4603
+ if (this.desktopModal) {
4604
+ this._setPageInert();
4605
+ } else {
4606
+ this._clearPageInert();
4607
+ }
4564
4608
  }
4565
4609
  }
4566
4610
 
@@ -4570,8 +4614,14 @@ class AuroDropdown extends AuroElement {
4570
4614
  * @param {CustomEvent} event - The custom event that contains the dropdown toggle information.
4571
4615
  */
4572
4616
  handleDropdownToggle(event) {
4573
- this.updateFocusTrap();
4574
4617
  this.isPopoverVisible = event.detail.expanded;
4618
+
4619
+ // Tear down FocusTrap when closing. Creation happens in updated()
4620
+ // after the dialog is open so getFocusableElements can find content.
4621
+ if (!this.isPopoverVisible) {
4622
+ this.updateFocusTrap();
4623
+ }
4624
+
4575
4625
  const eventType = event.detail.eventType || "unknown";
4576
4626
  if (!this.isPopoverVisible && this.hasFocus && eventType === "keydown") {
4577
4627
  this.trigger.focus();
@@ -4670,19 +4720,178 @@ class AuroDropdown extends AuroElement {
4670
4720
  * @private
4671
4721
  */
4672
4722
  updateFocusTrap() {
4723
+ // Always clean up existing handlers/traps before setting up new ones
4724
+ // to prevent duplicate listeners on repeated calls.
4725
+ if (this._bibTabHandler) {
4726
+ this.removeEventListener('keydown', this._bibTabHandler);
4727
+ this._bibTabHandler = undefined;
4728
+ }
4729
+
4730
+ if (this.focusTrap) {
4731
+ this.focusTrap.disconnect();
4732
+ this.focusTrap = undefined;
4733
+ }
4734
+
4673
4735
  if (this.isPopoverVisible) {
4674
4736
  if (!this.isBibFullscreen) {
4675
- // Desktop: show() doesn't trap focus, so use FocusTrap
4676
- this.focusTrap = new FocusTrap(this.bibContent);
4677
- this.focusTrap.focusFirstElement();
4737
+ if (this.desktopModal) {
4738
+ // Desktop modal: trap focus only within the bib content.
4739
+ // Can't use FocusTrap on the bib element because keydown events
4740
+ // from slotted content bubble through the dropdown host (light DOM),
4741
+ // not through the bib (shadow projection target). Using FocusTrap
4742
+ // on the dropdown would include the trigger in the tab cycle.
4743
+ // Instead, listen for Tab on the dropdown and manually wrap focus
4744
+ // within the bib's focusable elements.
4745
+ this._bibTabHandler = (event) => {
4746
+ if (event.key !== 'Tab') {
4747
+ return;
4748
+ }
4749
+
4750
+ // Collect focusable elements from the bib content.
4751
+ const focusables = getFocusableElements(this.bibContent);
4752
+
4753
+ // Fallback: try from slotted content directly
4754
+ if (!focusables.length) {
4755
+ const slot = this.shadowRoot.querySelector('.slotContent slot');
4756
+ const assignedNodes = slot ? slot.assignedNodes({ flatten: true }) : [];
4757
+
4758
+ for (const node of assignedNodes) {
4759
+ if (node.nodeType === Node.ELEMENT_NODE) {
4760
+ focusables.push(...getFocusableElements(node));
4761
+ }
4762
+ }
4763
+ }
4764
+
4765
+ if (!focusables.length) {
4766
+ return;
4767
+ }
4768
+
4769
+ event.preventDefault();
4770
+
4771
+ const direction = event.shiftKey ? -1 : 1; // eslint-disable-line no-magic-numbers
4772
+
4773
+ // Walk the active element chain through shadow roots
4774
+ const actives = this._getActiveElements();
4775
+
4776
+ let idx = focusables.findIndex((el) => actives.includes(el));
4777
+
4778
+ if (idx === -1) { // eslint-disable-line no-magic-numbers
4779
+ // Focus is not on a known element — move to first/last
4780
+ idx = direction === 1 ? -1 : focusables.length; // eslint-disable-line no-magic-numbers
4781
+ }
4782
+
4783
+ // Try each element in order, skipping any that can't receive focus
4784
+ // (e.g. hidden elements, elements in collapsed sections)
4785
+ for (let index = 0; index < focusables.length; index++) { // eslint-disable-line no-plusplus
4786
+ let nextIdx = idx + direction;
4787
+
4788
+ // Wrap around
4789
+ if (nextIdx < 0) {
4790
+ nextIdx = focusables.length - 1;
4791
+ } else if (nextIdx >= focusables.length) {
4792
+ nextIdx = 0;
4793
+ }
4794
+
4795
+ focusables[nextIdx].focus();
4796
+
4797
+ // Verify focus actually moved to the target
4798
+ const newActives = this._getActiveElements();
4799
+
4800
+ if (newActives.includes(focusables[nextIdx])) {
4801
+ return;
4802
+ }
4803
+
4804
+ // Focus didn't stick — skip this element and try the next
4805
+ idx = nextIdx;
4806
+ }
4807
+ };
4808
+ this.addEventListener('keydown', this._bibTabHandler);
4809
+
4810
+ // Move initial focus into the bib content, matching FocusTrap behavior
4811
+ requestAnimationFrame(() => {
4812
+ const focusables = getFocusableElements(this.bibContent);
4813
+ if (focusables.length) {
4814
+ focusables[0].focus();
4815
+ }
4816
+ });
4817
+ } else {
4818
+ // Normal desktop: use FocusTrap on the bib element
4819
+ this.focusTrap = new FocusTrap(this.bibContent);
4820
+ this.focusTrap.focusFirstElement();
4821
+ }
4678
4822
  }
4679
4823
  // Fullscreen: showModal() provides native focus trapping
4824
+ }
4825
+ }
4826
+
4827
+ /**
4828
+ * Returns the chain of active (focused) elements through shadow roots.
4829
+ * @private
4830
+ * @returns {Array<HTMLElement>}
4831
+ */
4832
+ _getActiveElements() {
4833
+ let { activeElement } = document;
4834
+ const actives = [activeElement];
4835
+
4836
+ while (activeElement?.shadowRoot?.activeElement) {
4837
+ activeElement = activeElement.shadowRoot.activeElement;
4838
+ actives.push(activeElement);
4839
+ }
4840
+
4841
+ return actives;
4842
+ }
4843
+
4844
+ /**
4845
+ * Sets `inert` on sibling elements of the dropdown's top-level host
4846
+ * so that content outside the dropdown is not interactive while the modal is open.
4847
+ * Walks up through shadow DOM boundaries to find the outermost host element
4848
+ * in the light DOM, then sets `inert` on siblings at each ancestor level
4849
+ * to ensure all page content outside the host subtree is inert.
4850
+ * @private
4851
+ */
4852
+ _setPageInert() {
4853
+ if (this._inertSiblings) {
4680
4854
  return;
4681
4855
  }
4682
4856
 
4683
- if (this.focusTrap) {
4684
- this.focusTrap.disconnect();
4685
- this.focusTrap = undefined;
4857
+ this._inertSiblings = [];
4858
+
4859
+ // Walk up through shadow DOM boundaries to find the topmost host
4860
+ // element in the light DOM. For example, if this dropdown is inside
4861
+ // auro-datepicker's shadow DOM, we walk up to the datepicker element
4862
+ // so we set inert on its siblings — not on the datepicker itself.
4863
+ let host = this;
4864
+ while (host.getRootNode() instanceof ShadowRoot) {
4865
+ host = host.getRootNode().host;
4866
+ }
4867
+
4868
+ // Walk up the ancestor chain, inerting siblings at each level
4869
+ // to ensure the entire page outside the host subtree is inert.
4870
+ let current = host;
4871
+ while (current.parentElement) {
4872
+ const parent = current.parentElement;
4873
+ for (const sibling of parent.children) {
4874
+ if (sibling !== current) {
4875
+ this._inertSiblings.push({ element: sibling, wasInert: sibling.inert });
4876
+ sibling.inert = true;
4877
+ }
4878
+ }
4879
+ current = parent;
4880
+ }
4881
+ }
4882
+
4883
+ /**
4884
+ * Restores `inert` state on siblings that were tracked by `_setPageInert`.
4885
+ * Preserves the previous inert state so externally-inerted elements are
4886
+ * not inadvertently re-enabled.
4887
+ * @private
4888
+ */
4889
+ _clearPageInert() {
4890
+ if (this._inertSiblings) {
4891
+ for (const entry of this._inertSiblings) {
4892
+ entry.element.inert = entry.wasInert;
4893
+ }
4894
+ this._inertSiblings = undefined;
4686
4895
  }
4687
4896
  }
4688
4897
 
@@ -4921,6 +5130,7 @@ class AuroDropdown extends AuroElement {
4921
5130
  shape="${this.shape}"
4922
5131
  ?data-show="${this.isPopoverVisible}"
4923
5132
  ?isfullscreen="${this.isBibFullscreen}"
5133
+ ?desktopmodal="${this.desktopModal}"
4924
5134
  .dialogLabel="${this.bibDialogLabel}"
4925
5135
  ${ref(this.bibElement)}
4926
5136
  >
@@ -3243,7 +3243,7 @@ function applyKeyboardStrategy(component, strategy, options = {}) {
3243
3243
  });
3244
3244
  }
3245
3245
 
3246
- var styleCss$2 = css`:host{position:fixed;z-index:var(--depth-tooltip, 400);display:none;isolation:isolate}:host dialog{width:auto;max-width:none;height:auto;max-height:none;padding:0;border:none;margin:0;outline:none;transform:translateZ(0)}:host dialog::backdrop{background:transparent}:host(:not([isfullscreen])) dialog{position:relative;inset:unset}:host(:not([isfullscreen])) .container.shape-box{border-radius:unset}:host(:not([isfullscreen])) .container[class*=shape-pill],:host(:not([isfullscreen])) .container[class*=shape-snowflake]{border-radius:30px}:host(:not([isfullscreen])) .container[class*=shape-rounded]{border-radius:16px}:host(:not([matchWidth])) .container{min-width:fit-content}:host([isfullscreen]){position:fixed;top:0;left:0}:host([isfullscreen]) .container{width:100dvw;max-width:none;height:100dvh;max-height:none;border-radius:unset;margin-top:0;box-shadow:unset;overscroll-behavior:contain}:host([isfullscreen]) .container::backdrop{background:var(--ds-color-background-primary, #fff)}:host(:popover-open){position:fixed;overflow:visible;padding:0;border:none;margin:0;background:transparent;inset:unset;outline:none}:host([data-show]){display:flex}:host([common]:not([isfullscreen])) .container,:host([rounded]:not([isfullscreen])) .container{border-radius:var(--ds-border-radius, 0.375rem)}:host([common][isfullscreen]) .container,:host([rounded][isfullscreen]) .container{border-radius:unset;box-shadow:unset}.container{display:inline-block;overflow:auto;box-sizing:border-box;border-radius:var(--ds-border-radius, 0.375rem);margin:var(--ds-size-50, 0.25rem) 0}.util_displayHiddenVisually{position:absolute;overflow:hidden;width:1px;height:1px;padding:0;border:0;margin:-1px;clip-path:inset(50%);white-space:nowrap}`;
3246
+ var styleCss$2 = css`:host{position:fixed;z-index:var(--depth-tooltip, 400);display:none;isolation:isolate}:host dialog{width:auto;max-width:none;height:auto;max-height:none;padding:0;border:none;margin:0;outline:none;transform:translateZ(0)}:host dialog::backdrop{background:transparent}:host(:not([isfullscreen])) dialog{position:relative;inset:unset}:host(:not([isfullscreen])) .container.shape-box{border-radius:unset}:host(:not([isfullscreen])) .container[class*=shape-pill],:host(:not([isfullscreen])) .container[class*=shape-snowflake]{border-radius:30px}:host(:not([isfullscreen])) .container[class*=shape-rounded]{border-radius:16px}:host([desktopmodal]:popover-open)::backdrop{background:transparent}:host(:not([matchWidth])) .container{min-width:fit-content}:host([isfullscreen]){position:fixed;top:0;left:0}:host([isfullscreen]) .container{width:100dvw;max-width:none;height:100dvh;max-height:none;border-radius:unset;margin-top:0;box-shadow:unset;overscroll-behavior:contain}:host([isfullscreen]) .container::backdrop{background:var(--ds-color-background-primary, #fff)}:host(:popover-open){position:fixed;overflow:visible;padding:0;border:none;margin:0;background:transparent;inset:unset;outline:none}:host([data-show]){display:flex}:host([common]:not([isfullscreen])) .container,:host([rounded]:not([isfullscreen])) .container{border-radius:var(--ds-border-radius, 0.375rem)}:host([common][isfullscreen]) .container,:host([rounded][isfullscreen]) .container{border-radius:unset;box-shadow:unset}.container{display:inline-block;overflow:auto;box-sizing:border-box;border-radius:var(--ds-border-radius, 0.375rem);margin:var(--ds-size-50, 0.25rem) 0}.util_displayHiddenVisually{position:absolute;overflow:hidden;width:1px;height:1px;padding:0;border:0;margin:-1px;clip-path:inset(50%);white-space:nowrap}`;
3247
3247
 
3248
3248
  var colorCss$2 = css`.container{background-color:var(--ds-auro-dropdownbib-container-color);box-shadow:var(--ds-auro-dropdownbib-boxshadow-color);color:var(--ds-auro-dropdownbib-text-color)}`;
3249
3249
 
@@ -3874,7 +3874,7 @@ class AuroHelpText extends LitElement {
3874
3874
  }
3875
3875
  }
3876
3876
 
3877
- var formkitVersion = '202606011708';
3877
+ var formkitVersion = '202606011856';
3878
3878
 
3879
3879
  class AuroElement extends LitElement {
3880
3880
  static get properties() {
@@ -4054,6 +4054,7 @@ class AuroDropdown extends AuroElement {
4054
4054
  _intializeDefaults() {
4055
4055
  this.appearance = 'default';
4056
4056
  this.chevron = false;
4057
+ this.desktopModal = false;
4057
4058
  this.disabled = false;
4058
4059
  this.disableKeyboardHandling = false;
4059
4060
  this.error = false;
@@ -4234,6 +4235,14 @@ class AuroDropdown extends AuroElement {
4234
4235
  reflect: true
4235
4236
  },
4236
4237
 
4238
+ /**
4239
+ * If declared, the dropdown will behave as a modal dialog when in a desktop viewport size.
4240
+ */
4241
+ desktopModal: {
4242
+ type: Boolean,
4243
+ reflect: true
4244
+ },
4245
+
4237
4246
  /**
4238
4247
  * If declared, the dropdown will only show by calling the API .show() public method.
4239
4248
  */
@@ -4521,6 +4530,15 @@ class AuroDropdown extends AuroElement {
4521
4530
 
4522
4531
  disconnectedCallback() {
4523
4532
  super.disconnectedCallback();
4533
+ this._clearPageInert();
4534
+ if (this._bibTabHandler) {
4535
+ this.removeEventListener('keydown', this._bibTabHandler);
4536
+ this._bibTabHandler = undefined;
4537
+ }
4538
+ if (this.focusTrap) {
4539
+ this.focusTrap.disconnect();
4540
+ this.focusTrap = undefined;
4541
+ }
4524
4542
  if (this.floater) {
4525
4543
  this.floater.hideBib('disconnect');
4526
4544
  this.floater.disconnect();
@@ -4548,19 +4566,45 @@ class AuroDropdown extends AuroElement {
4548
4566
  if (this.isPopoverVisible) {
4549
4567
  // Fullscreen: use showModal() for native accessibility (inert outside, focus trap)
4550
4568
  // Desktop: use show() for Floating UI positioning + FocusTrap for focus management
4551
- const useModal = this.isBibFullscreen;
4552
- this.bibElement.value.open(useModal);
4569
+ this.bibElement.value.open(this.isBibFullscreen);
4570
+ this.updateFocusTrap();
4571
+
4572
+ // Desktop modal: make siblings inert so content outside is not interactive
4573
+ if (this.desktopModal && !this.isBibFullscreen) {
4574
+ this._setPageInert();
4575
+ }
4553
4576
  } else {
4554
4577
  this.bibElement.value.close();
4578
+ this._clearPageInert();
4555
4579
  }
4556
4580
  }
4557
4581
 
4558
4582
  // When fullscreen strategy changes while open, re-open dialog with correct mode
4559
4583
  // (e.g. resizing from desktop → mobile while dropdown is open)
4560
4584
  if (changedProperties.has('isBibFullscreen') && this.isPopoverVisible && this.bibElement.value) {
4561
- const useModal = this.isBibFullscreen;
4562
4585
  this.bibElement.value.close();
4563
- this.bibElement.value.open(useModal);
4586
+ this.bibElement.value.open(this.isBibFullscreen);
4587
+
4588
+ // Re-initialize focus management for the new strategy
4589
+ this.updateFocusTrap();
4590
+
4591
+ // Toggle inert: desktop modal needs it, fullscreen showModal() handles it natively
4592
+ if (this.desktopModal && !this.isBibFullscreen) {
4593
+ this._setPageInert();
4594
+ } else {
4595
+ this._clearPageInert();
4596
+ }
4597
+ }
4598
+
4599
+ // Handle desktopModal toggled while the dropdown is already open.
4600
+ // Re-initialize focus trapping and page inert state to match the new mode.
4601
+ if (changedProperties.has('desktopModal') && this.isPopoverVisible && !this.isBibFullscreen) {
4602
+ this.updateFocusTrap();
4603
+ if (this.desktopModal) {
4604
+ this._setPageInert();
4605
+ } else {
4606
+ this._clearPageInert();
4607
+ }
4564
4608
  }
4565
4609
  }
4566
4610
 
@@ -4570,8 +4614,14 @@ class AuroDropdown extends AuroElement {
4570
4614
  * @param {CustomEvent} event - The custom event that contains the dropdown toggle information.
4571
4615
  */
4572
4616
  handleDropdownToggle(event) {
4573
- this.updateFocusTrap();
4574
4617
  this.isPopoverVisible = event.detail.expanded;
4618
+
4619
+ // Tear down FocusTrap when closing. Creation happens in updated()
4620
+ // after the dialog is open so getFocusableElements can find content.
4621
+ if (!this.isPopoverVisible) {
4622
+ this.updateFocusTrap();
4623
+ }
4624
+
4575
4625
  const eventType = event.detail.eventType || "unknown";
4576
4626
  if (!this.isPopoverVisible && this.hasFocus && eventType === "keydown") {
4577
4627
  this.trigger.focus();
@@ -4670,19 +4720,178 @@ class AuroDropdown extends AuroElement {
4670
4720
  * @private
4671
4721
  */
4672
4722
  updateFocusTrap() {
4723
+ // Always clean up existing handlers/traps before setting up new ones
4724
+ // to prevent duplicate listeners on repeated calls.
4725
+ if (this._bibTabHandler) {
4726
+ this.removeEventListener('keydown', this._bibTabHandler);
4727
+ this._bibTabHandler = undefined;
4728
+ }
4729
+
4730
+ if (this.focusTrap) {
4731
+ this.focusTrap.disconnect();
4732
+ this.focusTrap = undefined;
4733
+ }
4734
+
4673
4735
  if (this.isPopoverVisible) {
4674
4736
  if (!this.isBibFullscreen) {
4675
- // Desktop: show() doesn't trap focus, so use FocusTrap
4676
- this.focusTrap = new FocusTrap(this.bibContent);
4677
- this.focusTrap.focusFirstElement();
4737
+ if (this.desktopModal) {
4738
+ // Desktop modal: trap focus only within the bib content.
4739
+ // Can't use FocusTrap on the bib element because keydown events
4740
+ // from slotted content bubble through the dropdown host (light DOM),
4741
+ // not through the bib (shadow projection target). Using FocusTrap
4742
+ // on the dropdown would include the trigger in the tab cycle.
4743
+ // Instead, listen for Tab on the dropdown and manually wrap focus
4744
+ // within the bib's focusable elements.
4745
+ this._bibTabHandler = (event) => {
4746
+ if (event.key !== 'Tab') {
4747
+ return;
4748
+ }
4749
+
4750
+ // Collect focusable elements from the bib content.
4751
+ const focusables = getFocusableElements(this.bibContent);
4752
+
4753
+ // Fallback: try from slotted content directly
4754
+ if (!focusables.length) {
4755
+ const slot = this.shadowRoot.querySelector('.slotContent slot');
4756
+ const assignedNodes = slot ? slot.assignedNodes({ flatten: true }) : [];
4757
+
4758
+ for (const node of assignedNodes) {
4759
+ if (node.nodeType === Node.ELEMENT_NODE) {
4760
+ focusables.push(...getFocusableElements(node));
4761
+ }
4762
+ }
4763
+ }
4764
+
4765
+ if (!focusables.length) {
4766
+ return;
4767
+ }
4768
+
4769
+ event.preventDefault();
4770
+
4771
+ const direction = event.shiftKey ? -1 : 1; // eslint-disable-line no-magic-numbers
4772
+
4773
+ // Walk the active element chain through shadow roots
4774
+ const actives = this._getActiveElements();
4775
+
4776
+ let idx = focusables.findIndex((el) => actives.includes(el));
4777
+
4778
+ if (idx === -1) { // eslint-disable-line no-magic-numbers
4779
+ // Focus is not on a known element — move to first/last
4780
+ idx = direction === 1 ? -1 : focusables.length; // eslint-disable-line no-magic-numbers
4781
+ }
4782
+
4783
+ // Try each element in order, skipping any that can't receive focus
4784
+ // (e.g. hidden elements, elements in collapsed sections)
4785
+ for (let index = 0; index < focusables.length; index++) { // eslint-disable-line no-plusplus
4786
+ let nextIdx = idx + direction;
4787
+
4788
+ // Wrap around
4789
+ if (nextIdx < 0) {
4790
+ nextIdx = focusables.length - 1;
4791
+ } else if (nextIdx >= focusables.length) {
4792
+ nextIdx = 0;
4793
+ }
4794
+
4795
+ focusables[nextIdx].focus();
4796
+
4797
+ // Verify focus actually moved to the target
4798
+ const newActives = this._getActiveElements();
4799
+
4800
+ if (newActives.includes(focusables[nextIdx])) {
4801
+ return;
4802
+ }
4803
+
4804
+ // Focus didn't stick — skip this element and try the next
4805
+ idx = nextIdx;
4806
+ }
4807
+ };
4808
+ this.addEventListener('keydown', this._bibTabHandler);
4809
+
4810
+ // Move initial focus into the bib content, matching FocusTrap behavior
4811
+ requestAnimationFrame(() => {
4812
+ const focusables = getFocusableElements(this.bibContent);
4813
+ if (focusables.length) {
4814
+ focusables[0].focus();
4815
+ }
4816
+ });
4817
+ } else {
4818
+ // Normal desktop: use FocusTrap on the bib element
4819
+ this.focusTrap = new FocusTrap(this.bibContent);
4820
+ this.focusTrap.focusFirstElement();
4821
+ }
4678
4822
  }
4679
4823
  // Fullscreen: showModal() provides native focus trapping
4824
+ }
4825
+ }
4826
+
4827
+ /**
4828
+ * Returns the chain of active (focused) elements through shadow roots.
4829
+ * @private
4830
+ * @returns {Array<HTMLElement>}
4831
+ */
4832
+ _getActiveElements() {
4833
+ let { activeElement } = document;
4834
+ const actives = [activeElement];
4835
+
4836
+ while (activeElement?.shadowRoot?.activeElement) {
4837
+ activeElement = activeElement.shadowRoot.activeElement;
4838
+ actives.push(activeElement);
4839
+ }
4840
+
4841
+ return actives;
4842
+ }
4843
+
4844
+ /**
4845
+ * Sets `inert` on sibling elements of the dropdown's top-level host
4846
+ * so that content outside the dropdown is not interactive while the modal is open.
4847
+ * Walks up through shadow DOM boundaries to find the outermost host element
4848
+ * in the light DOM, then sets `inert` on siblings at each ancestor level
4849
+ * to ensure all page content outside the host subtree is inert.
4850
+ * @private
4851
+ */
4852
+ _setPageInert() {
4853
+ if (this._inertSiblings) {
4680
4854
  return;
4681
4855
  }
4682
4856
 
4683
- if (this.focusTrap) {
4684
- this.focusTrap.disconnect();
4685
- this.focusTrap = undefined;
4857
+ this._inertSiblings = [];
4858
+
4859
+ // Walk up through shadow DOM boundaries to find the topmost host
4860
+ // element in the light DOM. For example, if this dropdown is inside
4861
+ // auro-datepicker's shadow DOM, we walk up to the datepicker element
4862
+ // so we set inert on its siblings — not on the datepicker itself.
4863
+ let host = this;
4864
+ while (host.getRootNode() instanceof ShadowRoot) {
4865
+ host = host.getRootNode().host;
4866
+ }
4867
+
4868
+ // Walk up the ancestor chain, inerting siblings at each level
4869
+ // to ensure the entire page outside the host subtree is inert.
4870
+ let current = host;
4871
+ while (current.parentElement) {
4872
+ const parent = current.parentElement;
4873
+ for (const sibling of parent.children) {
4874
+ if (sibling !== current) {
4875
+ this._inertSiblings.push({ element: sibling, wasInert: sibling.inert });
4876
+ sibling.inert = true;
4877
+ }
4878
+ }
4879
+ current = parent;
4880
+ }
4881
+ }
4882
+
4883
+ /**
4884
+ * Restores `inert` state on siblings that were tracked by `_setPageInert`.
4885
+ * Preserves the previous inert state so externally-inerted elements are
4886
+ * not inadvertently re-enabled.
4887
+ * @private
4888
+ */
4889
+ _clearPageInert() {
4890
+ if (this._inertSiblings) {
4891
+ for (const entry of this._inertSiblings) {
4892
+ entry.element.inert = entry.wasInert;
4893
+ }
4894
+ this._inertSiblings = undefined;
4686
4895
  }
4687
4896
  }
4688
4897
 
@@ -4921,6 +5130,7 @@ class AuroDropdown extends AuroElement {
4921
5130
  shape="${this.shape}"
4922
5131
  ?data-show="${this.isPopoverVisible}"
4923
5132
  ?isfullscreen="${this.isBibFullscreen}"
5133
+ ?desktopmodal="${this.desktopModal}"
4924
5134
  .dialogLabel="${this.bibDialogLabel}"
4925
5135
  ${ref(this.bibElement)}
4926
5136
  >