@nvidia-elements/core 0.0.2 → 0.0.3

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 (195) hide show
  1. package/CHANGELOG.md +17 -28
  2. package/README.md +1 -10
  3. package/dist/_virtual/{_@oxc-project_runtime@0.115.0 → _@oxc-project_runtime@0.123.0}/helpers/decorate.js +1 -1
  4. package/dist/accordion/accordion2.js +5 -5
  5. package/dist/alert/alert-banner2.js +1 -1
  6. package/dist/alert/alert-group2.js +2 -2
  7. package/dist/alert/alert2.js +2 -2
  8. package/dist/avatar/avatar-group2.js +1 -1
  9. package/dist/avatar/avatar2.js +2 -2
  10. package/dist/badge/badge2.js +2 -2
  11. package/dist/breadcrumb/breadcrumb2.js +2 -2
  12. package/dist/bundles/index.d.ts +38 -34
  13. package/dist/bundles/index.js +13 -6952
  14. package/dist/button/button2.js +2 -2
  15. package/dist/button-group/button-group2.js +2 -2
  16. package/dist/card/card2.js +5 -5
  17. package/dist/chat-message/chat-message2.js +2 -2
  18. package/dist/checkbox/checkbox-group2.js +2 -2
  19. package/dist/checkbox/checkbox2.js +2 -2
  20. package/dist/checkbox/define.js +6 -3
  21. package/dist/checkbox/define.js.map +1 -1
  22. package/dist/color/color2.js +2 -2
  23. package/dist/color/define.js +5 -3
  24. package/dist/color/define.js.map +1 -1
  25. package/dist/combobox/combobox.d.ts +1 -0
  26. package/dist/combobox/combobox.examples.js.map +1 -1
  27. package/dist/combobox/combobox.examples.json +11 -0
  28. package/dist/combobox/combobox2.js +76 -55
  29. package/dist/combobox/combobox2.js.map +1 -1
  30. package/dist/combobox/define.js +5 -3
  31. package/dist/combobox/define.js.map +1 -1
  32. package/dist/copy-button/copy-button2.js +2 -2
  33. package/dist/custom-elements-jsx.d.ts +9 -62
  34. package/dist/custom-elements-vue.d.ts +9 -62
  35. package/dist/custom-elements.json +61 -733
  36. package/dist/data.html.json +4 -42
  37. package/dist/date/date2.js +2 -2
  38. package/dist/date/define.js +5 -3
  39. package/dist/date/define.js.map +1 -1
  40. package/dist/datetime/datetime2.js +2 -2
  41. package/dist/datetime/define.js +5 -3
  42. package/dist/datetime/define.js.map +1 -1
  43. package/dist/dialog/dialog-footer2.js +2 -2
  44. package/dist/dialog/dialog-header2.js +2 -2
  45. package/dist/dialog/dialog2.js +2 -2
  46. package/dist/divider/divider2.js +2 -2
  47. package/dist/dot/dot2.js +2 -2
  48. package/dist/drawer/drawer-content2.js +2 -2
  49. package/dist/drawer/drawer-footer2.js +2 -2
  50. package/dist/drawer/drawer-header2.js +2 -2
  51. package/dist/drawer/drawer2.js +2 -2
  52. package/dist/dropdown/dropdown-footer2.js +2 -2
  53. package/dist/dropdown/dropdown-header2.js +2 -2
  54. package/dist/dropdown/dropdown2.js +2 -2
  55. package/dist/dropdown-group/dropdown-group.js +2 -2
  56. package/dist/dropzone/dropzone2.js +2 -2
  57. package/dist/file/define.js +5 -3
  58. package/dist/file/define.js.map +1 -1
  59. package/dist/file/file2.js +1 -1
  60. package/dist/forms/control/control2.js +2 -2
  61. package/dist/forms/control-group/control-group2.js +2 -2
  62. package/dist/forms/control-message/control-message2.js +2 -2
  63. package/dist/grid/cell/cell2.js +1 -1
  64. package/dist/grid/column/column.d.ts +1 -0
  65. package/dist/grid/column/column2.js +5 -2
  66. package/dist/grid/column/column2.js.map +1 -1
  67. package/dist/grid/footer/footer2.js +2 -2
  68. package/dist/grid/grid2.js +2 -2
  69. package/dist/grid/header/header2.js +2 -2
  70. package/dist/grid/placeholder/placeholder2.js +1 -1
  71. package/dist/grid/row/row2.js +2 -2
  72. package/dist/icon/icon2.js +3 -3
  73. package/dist/icon-button/icon-button2.js +2 -2
  74. package/dist/index.js +1 -1
  75. package/dist/input/define.js +6 -3
  76. package/dist/input/define.js.map +1 -1
  77. package/dist/input/input-group2.js +1 -1
  78. package/dist/input/input2.js +2 -2
  79. package/dist/internal/base/button.js +1 -1
  80. package/dist/internal/index.js +40 -40
  81. package/dist/internal/services/global.service.js +1 -1
  82. package/dist/internal/utils/dom.d.ts +2 -0
  83. package/dist/internal/utils/dom.js +51 -51
  84. package/dist/internal/utils/dom.js.map +1 -1
  85. package/dist/logo/logo2.js +2 -2
  86. package/dist/menu/menu-item2.js +2 -2
  87. package/dist/menu/menu.d.ts +3 -0
  88. package/dist/menu/menu2.js +23 -2
  89. package/dist/menu/menu2.js.map +1 -1
  90. package/dist/month/define.js +5 -3
  91. package/dist/month/define.js.map +1 -1
  92. package/dist/month/month2.js +2 -2
  93. package/dist/notification/notification-group2.js +2 -2
  94. package/dist/notification/notification2.js +2 -2
  95. package/dist/page/page-panel/page-panel-content2.js +1 -1
  96. package/dist/page/page-panel/page-panel-footer2.js +2 -2
  97. package/dist/page/page-panel/page-panel-header2.js +2 -2
  98. package/dist/page/page-panel/page-panel2.js +2 -2
  99. package/dist/page/page.examples.js.map +1 -1
  100. package/dist/page/page2.js +2 -2
  101. package/dist/page-header/page-header.examples.js.map +1 -1
  102. package/dist/page-header/page-header2.js +1 -1
  103. package/dist/page-loader/page-loader.js +1 -1
  104. package/dist/page-loader/page-loader.js.map +1 -1
  105. package/dist/page-loader/page-loader2.js +2 -2
  106. package/dist/pagination/pagination2.js +2 -2
  107. package/dist/panel/panel2.js +5 -5
  108. package/dist/password/define.js +5 -3
  109. package/dist/password/define.js.map +1 -1
  110. package/dist/password/password2.js +2 -2
  111. package/dist/preferences-input/preferences-input2.js +2 -2
  112. package/dist/progress-bar/progress-bar2.js +2 -2
  113. package/dist/progress-ring/progress-ring2.js +2 -2
  114. package/dist/progressive-filter-chip/progressive-filter-chip2.js +2 -2
  115. package/dist/pulse/pulse2.js +2 -2
  116. package/dist/radio/define.js +6 -3
  117. package/dist/radio/define.js.map +1 -1
  118. package/dist/radio/radio-group2.js +2 -2
  119. package/dist/radio/radio2.js +2 -2
  120. package/dist/range/define.js +5 -3
  121. package/dist/range/define.js.map +1 -1
  122. package/dist/range/range2.js +2 -2
  123. package/dist/resize-handle/resize-handle2.js +2 -2
  124. package/dist/search/define.js +5 -3
  125. package/dist/search/define.js.map +1 -1
  126. package/dist/search/search2.js +2 -2
  127. package/dist/select/define.js +5 -3
  128. package/dist/select/define.js.map +1 -1
  129. package/dist/select/select2.js +2 -2
  130. package/dist/skeleton/skeleton2.js +2 -2
  131. package/dist/sort-button/sort-button2.js +2 -2
  132. package/dist/sparkline/sparkline.utils.js +2 -8
  133. package/dist/sparkline/sparkline.utils.js.map +1 -1
  134. package/dist/sparkline/sparkline2.js +2 -2
  135. package/dist/star-rating/define.js +5 -3
  136. package/dist/star-rating/define.js.map +1 -1
  137. package/dist/star-rating/star-rating2.js +2 -2
  138. package/dist/steps/steps2.js +3 -3
  139. package/dist/switch/define.js +6 -3
  140. package/dist/switch/define.js.map +1 -1
  141. package/dist/switch/switch-group2.js +2 -2
  142. package/dist/switch/switch2.js +2 -2
  143. package/dist/tabs/tabs2.js +3 -3
  144. package/dist/tag/tag2.js +2 -2
  145. package/dist/textarea/define.js +5 -3
  146. package/dist/textarea/define.js.map +1 -1
  147. package/dist/textarea/textarea2.js +1 -1
  148. package/dist/time/define.js +5 -3
  149. package/dist/time/define.js.map +1 -1
  150. package/dist/time/time2.js +2 -2
  151. package/dist/toast/toast2.js +2 -2
  152. package/dist/toggletip/toggletip-footer2.js +2 -2
  153. package/dist/toggletip/toggletip-header2.js +2 -2
  154. package/dist/toggletip/toggletip2.js +2 -2
  155. package/dist/toolbar/toolbar2.js +2 -2
  156. package/dist/tooltip/tooltip2.js +2 -2
  157. package/dist/tree/tree-node2.js +2 -2
  158. package/dist/tree/tree2.js +2 -2
  159. package/dist/week/define.js +5 -3
  160. package/dist/week/define.js.map +1 -1
  161. package/dist/week/week2.js +2 -2
  162. package/package.json +10 -36
  163. package/dist/app-header/app-header.d.ts +0 -39
  164. package/dist/app-header/app-header.js +0 -6
  165. package/dist/app-header/app-header.js.map +0 -1
  166. package/dist/app-header/app-header2.js +0 -54
  167. package/dist/app-header/app-header2.js.map +0 -1
  168. package/dist/app-header/define.d.ts +0 -6
  169. package/dist/app-header/define.js +0 -7
  170. package/dist/app-header/define.js.map +0 -1
  171. package/dist/app-header/index.d.ts +0 -1
  172. package/dist/app-header/index.js +0 -2
  173. package/dist/bundles/audit-logs.js +0 -50
  174. package/dist/bundles/icons.js +0 -285
  175. package/dist/bundles/index.css +0 -2
  176. package/dist/bundles/log.service.js +0 -90
  177. package/dist/bundles/rolldown-runtime.js +0 -11
  178. package/dist/css/module.layout.css +0 -1
  179. package/dist/css/module.typography.css +0 -1
  180. package/dist/index.css +0 -1
  181. package/dist/json-viewer/define.d.ts +0 -8
  182. package/dist/json-viewer/define.js +0 -8
  183. package/dist/json-viewer/define.js.map +0 -1
  184. package/dist/json-viewer/index.d.ts +0 -2
  185. package/dist/json-viewer/index.js +0 -3
  186. package/dist/json-viewer/json-viewer.d.ts +0 -25
  187. package/dist/json-viewer/json-viewer.js +0 -6
  188. package/dist/json-viewer/json-viewer.js.map +0 -1
  189. package/dist/json-viewer/json-viewer2.js +0 -39
  190. package/dist/json-viewer/json-viewer2.js.map +0 -1
  191. package/dist/json-viewer/node/node.d.ts +0 -21
  192. package/dist/json-viewer/node/node.js +0 -6
  193. package/dist/json-viewer/node/node.js.map +0 -1
  194. package/dist/json-viewer/node/node2.js +0 -72
  195. package/dist/json-viewer/node/node2.js.map +0 -1
@@ -1,4 +1,4 @@
1
- import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.js";
1
+ import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.123.0/helpers/decorate.js";
2
2
  import { BaseButton as t } from "../internal/base/button.js";
3
3
  import { useStyles as n } from "../internal/styles/index.js";
4
4
  import r from "./button.js";
@@ -12,7 +12,7 @@ var o = class extends t {
12
12
  static {
13
13
  this.metadata = {
14
14
  tag: "nve-button",
15
- version: "0.0.2"
15
+ version: "0.0.3"
16
16
  };
17
17
  }
18
18
  render() {
@@ -1,6 +1,6 @@
1
1
  import { appendRootNodeStyle as e } from "../internal/utils/dom.js";
2
2
  import { attachInternals as t } from "../internal/utils/a11y.js";
3
- import { __decorate as n } from "../_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.js";
3
+ import { __decorate as n } from "../_virtual/_@oxc-project_runtime@0.123.0/helpers/decorate.js";
4
4
  import { audit as r } from "../internal/controllers/audit.controller.js";
5
5
  import { keyNavigationList as i } from "../internal/controllers/keynav-list.controller.js";
6
6
  import { useStyles as a } from "../internal/styles/index.js";
@@ -26,7 +26,7 @@ var f = class extends c {
26
26
  static {
27
27
  this.metadata = {
28
28
  tag: "nve-button-group",
29
- version: "0.0.2",
29
+ version: "0.0.3",
30
30
  children: [
31
31
  "nve-button",
32
32
  "nve-icon-button",
@@ -1,4 +1,4 @@
1
- import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.js";
1
+ import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.123.0/helpers/decorate.js";
2
2
  import { audit as t } from "../internal/controllers/audit.controller.js";
3
3
  import { hostAttr as n } from "../internal/decorators/host-attr.js";
4
4
  import { useStyles as r } from "../internal/styles/index.js";
@@ -16,7 +16,7 @@ var d = class extends c {
16
16
  static {
17
17
  this.metadata = {
18
18
  tag: "nve-card",
19
- version: "0.0.2"
19
+ version: "0.0.3"
20
20
  };
21
21
  }
22
22
  render() {
@@ -37,7 +37,7 @@ var f = class extends c {
37
37
  static {
38
38
  this.metadata = {
39
39
  tag: "nve-card-header",
40
- version: "0.0.2",
40
+ version: "0.0.3",
41
41
  parents: ["nve-card"]
42
42
  };
43
43
  }
@@ -53,7 +53,7 @@ var p = class extends c {
53
53
  static {
54
54
  this.metadata = {
55
55
  tag: "nve-card-content",
56
- version: "0.0.2",
56
+ version: "0.0.3",
57
57
  parents: ["nve-card"]
58
58
  };
59
59
  }
@@ -72,7 +72,7 @@ var m = class extends c {
72
72
  static {
73
73
  this.metadata = {
74
74
  tag: "nve-card-footer",
75
- version: "0.0.2",
75
+ version: "0.0.3",
76
76
  parents: ["nve-card"]
77
77
  };
78
78
  }
@@ -1,5 +1,5 @@
1
1
  import { appendRootNodeStyle as e } from "../internal/utils/dom.js";
2
- import { __decorate as t } from "../_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.js";
2
+ import { __decorate as t } from "../_virtual/_@oxc-project_runtime@0.123.0/helpers/decorate.js";
3
3
  import { colorStateStyles as n, useStyles as r } from "../internal/styles/index.js";
4
4
  import i from "./chat-message.js";
5
5
  import a from "./chat-message.global.js";
@@ -13,7 +13,7 @@ var l = class extends o {
13
13
  static {
14
14
  this.metadata = {
15
15
  tag: "nve-chat-message",
16
- version: "0.0.2"
16
+ version: "0.0.3"
17
17
  };
18
18
  }
19
19
  render() {
@@ -1,4 +1,4 @@
1
- import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.js";
1
+ import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.123.0/helpers/decorate.js";
2
2
  import { audit as t } from "../internal/controllers/audit.controller.js";
3
3
  import { typeSSR as n } from "../internal/controllers/type-ssr.controller.js";
4
4
  import { useStyles as r } from "../internal/styles/index.js";
@@ -12,7 +12,7 @@ var o = class extends i {
12
12
  static {
13
13
  this.metadata = {
14
14
  tag: "nve-checkbox-group",
15
- version: "0.0.2",
15
+ version: "0.0.3",
16
16
  children: [
17
17
  "label",
18
18
  "nve-control-message",
@@ -1,4 +1,4 @@
1
- import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.js";
1
+ import { __decorate as e } from "../_virtual/_@oxc-project_runtime@0.123.0/helpers/decorate.js";
2
2
  import { audit as t } from "../internal/controllers/audit.controller.js";
3
3
  import { hostAttr as n } from "../internal/decorators/host-attr.js";
4
4
  import { useStyles as r } from "../internal/styles/index.js";
@@ -15,7 +15,7 @@ var o = class extends i {
15
15
  static {
16
16
  this.metadata = {
17
17
  tag: "nve-checkbox",
18
- version: "0.0.2",
18
+ version: "0.0.3",
19
19
  children: [
20
20
  "label",
21
21
  "input",
@@ -1,7 +1,10 @@
1
1
  import { define as e } from "../internal/utils/dom.js";
2
- import { Checkbox as t } from "./checkbox2.js";
3
- import { CheckboxGroup as n } from "./checkbox-group2.js";
4
- e(t), e(n);
2
+ import { ControlMessage as t } from "../forms/control-message/control-message2.js";
3
+ import { Control as n } from "../forms/control/control2.js";
4
+ import { ControlGroup as r } from "../forms/control-group/control-group2.js";
5
+ import { Checkbox as i } from "./checkbox2.js";
6
+ import { CheckboxGroup as a } from "./checkbox-group2.js";
7
+ e(i), e(a), e(n), e(r), e(t);
5
8
  //#endregion
6
9
 
7
10
  //# sourceMappingURL=define.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"define.js","names":[],"sources":["../../src/checkbox/define.ts"],"sourcesContent":["// SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { define } from '@nvidia-elements/core/internal';\nimport { Checkbox, CheckboxGroup } from '@nvidia-elements/core/checkbox';\nimport '@nvidia-elements/core/forms/define.js';\n\ndefine(Checkbox);\ndefine(CheckboxGroup);\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'nve-checkbox': Checkbox;\n 'nve-checkbox-group': CheckboxGroup;\n }\n}\n"],"mappings":";;;AAOA,EAAO,EAAS,EAChB,EAAO,EAAc"}
1
+ {"version":3,"file":"define.js","names":[],"sources":["../../src/checkbox/define.ts"],"sourcesContent":["// SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { define } from '@nvidia-elements/core/internal';\nimport { Checkbox, CheckboxGroup } from '@nvidia-elements/core/checkbox';\nimport { Control, ControlGroup, ControlMessage } from '@nvidia-elements/core/forms';\n\ndefine(Checkbox);\ndefine(CheckboxGroup);\ndefine(Control);\ndefine(ControlGroup);\ndefine(ControlMessage);\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'nve-checkbox': Checkbox;\n 'nve-checkbox-group': CheckboxGroup;\n }\n}\n"],"mappings":";;;;;;AAOA,EAAO,EAAS,EAChB,EAAO,EAAc,EACrB,EAAO,EAAQ,EACf,EAAO,EAAa,EACpB,EAAO,EAAe"}
@@ -1,5 +1,5 @@
1
1
  import { appendRootNodeStyle as e, openEyeDropper as t } from "../internal/utils/dom.js";
2
- import { __decorate as n } from "../_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.js";
2
+ import { __decorate as n } from "../_virtual/_@oxc-project_runtime@0.123.0/helpers/decorate.js";
3
3
  import { scopedRegistry as r } from "../internal/decorators/scoped-registry.js";
4
4
  import { useStyles as i } from "../internal/styles/index.js";
5
5
  import { Control as a } from "../forms/control/control2.js";
@@ -21,7 +21,7 @@ var f = class extends a {
21
21
  static {
22
22
  this.metadata = {
23
23
  tag: "nve-color",
24
- version: "0.0.2"
24
+ version: "0.0.3"
25
25
  };
26
26
  }
27
27
  static {
@@ -1,7 +1,9 @@
1
1
  import { define as e } from "../internal/utils/dom.js";
2
- import { Color as t } from "./color2.js";
3
- //#region src/color/define.ts
4
- e(t);
2
+ import { ControlMessage as t } from "../forms/control-message/control-message2.js";
3
+ import { Control as n } from "../forms/control/control2.js";
4
+ import { ControlGroup as r } from "../forms/control-group/control-group2.js";
5
+ import { Color as i } from "./color2.js";
6
+ e(i), e(n), e(r), e(t);
5
7
  //#endregion
6
8
 
7
9
  //# sourceMappingURL=define.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"define.js","names":[],"sources":["../../src/color/define.ts"],"sourcesContent":["// SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { define } from '@nvidia-elements/core/internal';\nimport { Color } from '@nvidia-elements/core/color';\nimport '@nvidia-elements/core/forms/define.js';\n\ndefine(Color);\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'nve-color': Color;\n }\n}\n"],"mappings":";;;AAOA,EAAO,EAAM"}
1
+ {"version":3,"file":"define.js","names":[],"sources":["../../src/color/define.ts"],"sourcesContent":["// SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { define } from '@nvidia-elements/core/internal';\nimport { Color } from '@nvidia-elements/core/color';\nimport { Control, ControlGroup, ControlMessage } from '@nvidia-elements/core/forms';\n\ndefine(Color);\ndefine(Control);\ndefine(ControlGroup);\ndefine(ControlMessage);\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'nve-color': Color;\n }\n}\n"],"mappings":";;;;;AAOA,EAAO,EAAM,EACb,EAAO,EAAQ,EACf,EAAO,EAAa,EACpB,EAAO,EAAe"}
@@ -34,6 +34,7 @@ import { Checkbox } from '../checkbox';
34
34
  * @csspart menu-item - The menu item elements
35
35
  * @csspart checkbox - The checkbox element
36
36
  * @csspart icon - The icon element
37
+ * @event scroll - Fires when the user scrolls the dropdown option list. Throttled to one dispatch per animation frame. `detail: { scrollTop, scrollHeight, clientHeight }`.
37
38
  * @aria https://www.w3.org/WAI/ARIA/apg/patterns/combobox/examples/combobox-autocomplete-list/
38
39
  */
39
40
  export declare class Combobox extends Control implements ContainerElement {
@@ -1 +1 @@
1
- {"version":3,"file":"combobox.examples.js","names":["#unusedFilters","#removeFilter","#createfilter","#updateFilter","#addFilter","#valueChange"],"sources":["../../src/combobox/combobox.examples.ts"],"sourcesContent":["// SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { html, LitElement, unsafeCSS } from 'lit';\nimport { property } from 'lit/decorators/property.js';\nimport { state } from 'lit/decorators/state.js';\nimport { choose } from 'lit/directives/choose.js';\nimport '@nvidia-elements/core/progressive-filter-chip/define.js';\nimport '@nvidia-elements/core/combobox/define.js';\nimport '@nvidia-elements/core/icon/define.js';\nimport '@nvidia-elements/core/tag/define.js';\nimport '@nvidia-elements/core/date/define.js';\nimport '@nvidia-elements/core/select/define.js';\nimport '@nvidia-elements/core/input/define.js';\nimport '@nvidia-elements/core/dropdown/define.js';\n\nexport default {\n title: 'Elements/Combobox',\n component: 'nve-combobox',\n};\n\n/**\n * @summary Basic combobox with search input and datalist options for filtering and selection.\n */\nexport const Default = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Vertical layout showing different combobox states (normal, disabled, success, error) stacked for comparison.\n * @tags test-case\n */\nexport const Vertical = () => {\n return html`\n<div nve-layout=\"column gap:lg full\">\n <nve-combobox>\n <label>label</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox>\n <label>disabled</label>\n <input type=\"search\" disabled />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox>\n <label>success</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"success\">message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox>\n <label>error</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"error\">message</nve-control-message>\n </nve-combobox>\n</div>`\n};\n\n/**\n * @summary Horizontal layout showing different combobox states (normal, disabled, success, error) for inline form layouts.\n */\nexport const Horizontal = () => {\n return html`\n<div nve-layout=\"column gap:lg full\">\n <nve-combobox layout=\"horizontal\">\n <label>label</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox layout=\"horizontal\">\n <label>disabled</label>\n <input type=\"search\" disabled />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox layout=\"horizontal\">\n <label>success</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"success\">message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox layout=\"horizontal\">\n <label>error</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"error\">message</nve-control-message>\n </nve-combobox>\n</div>`\n};\n\n/**\n * @summary Flat container style with prefix icon for compact inline filtering interfaces.\n */\nexport const Flat = () => {\n return html`\n <nve-combobox container=\"flat\">\n <nve-icon name=\"filter\" slot=\"prefix-icon\"></nve-icon>\n <input type=\"search\" aria-label=\"search\">\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Single select allows users only to select from a predefined list\n * of options. Invalid input is automatically cleared. All options are visible\n * on focus until typing begins for filtering.\n */\nexport const Select = () => {\n return html`\n<nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n</nve-combobox>\n `\n};\n\n/**\n * @summary Multi select allows users to select many options from a\n * predefined list. The select `value` will only reflect the first selected value.\n * To get all selected options check the `selected` property on each `<option>`\n * or the select property `selectedOptions`.\n * On focus all options will show until the user starts typing. Select is the\n * selection value of the combobox. The input is the filter value.\n */\nexport const MultiSelect = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option selected value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Combobox with an empty initial value using a disabled placeholder option. Use when no default selection exists and the user must make an explicit choice.\n */\nexport const EmptyDefault = () => {\n return html`\n<nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select>\n <option disabled selected></option>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n</nve-combobox>\n `\n};\n\n/**\n * @summary Combobox options with display labels that differ from underlying values. Use when option values are IDs or codes but users need to see human-readable text in the input.\n */\nexport const Label = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select>\n <option value=\"1\">Status</option>\n <option value=\"2\">Priority</option>\n <option value=\"3\">Date</option>\n <option value=\"4\">Session</option>\n <option value=\"5\">Configuration</option>\n <option value=\"6\">Contains</option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select with labeled options where display text differs from option values.\n */\nexport const LabelMultiSelect = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"1\">Status</option>\n <option value=\"2\">Priority</option>\n <option value=\"3\">Date</option>\n <option value=\"4\">Session</option>\n <option value=\"5\">Configuration</option>\n <option value=\"6\">Contains</option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Overflow behavior where many tags collapse into a simple text label when the parent container is too narrow.\n * @tags test-case\n */\nexport const Overflow = () => {\n return html`\n <nve-combobox style=\"width: 250px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option selected value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option selected value=\"date\"></option>\n <option selected value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Combobox handling of long option text in constrained width containers.\n * @tags test-case\n */\nexport const PopoverOverflow = () => {\n return html`\n <nve-combobox style=\"width: 100px\">\n <label>label</label>\n <input type=\"search\">\n <select>\n <option value=\"really-long-text-option-that-keeps-going\"></option> \n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select with reset functionality via icon button and footer button to clear all selections.\n */\nexport const Reset = () => {\n return html`\n <nve-combobox id=\"combobox-reset\" style=\"width: 500px; --scroll-height: 220px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-icon-button aria-label=\"clear selection\" icon-name=\"cancel\" container=\"inline\"></nve-icon-button>\n <nve-button slot=\"footer\" aria-label=\"clear selection\" container=\"flat\">reset</nve-button>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n <script type=\"module\">\n document.querySelector('#combobox-reset nve-icon-button').addEventListener('click', e => e.target.parentElement.reset());\n document.querySelector('#combobox-reset nve-button').addEventListener('click', e => e.target.parentElement.reset());\n </script>\n `\n};\n\n/**\n * @summary Multi-select with footer action button for extra operations on selected items.\n */\nexport const Footer = () => {\n return html`\n <nve-combobox style=\"width: 500px; --scroll-height: 200px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n <option value=\"includes\"></option>\n <option value=\"user\"></option>\n <option value=\"progress\"></option>\n </select>\n <nve-button slot=\"footer\" container=\"flat\">action</nve-button>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select with bulk selection controls (Select All/Deselect All) in footer for efficient mass operations.\n */\nexport const SelectAll = () => {\n return html`\n <nve-combobox id=\"combobox-select-all\" style=\"width: 500px; --scroll-height: 200px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n <option value=\"includes\"></option>\n <option value=\"user\"></option>\n <option value=\"progress\"></option>\n </select>\n <div slot=\"footer\" nve-layout=\"row align:stretch full\">\n <nve-button container=\"flat\">Select All</nve-button>\n <nve-button container=\"flat\">Deselect All</nve-button>\n </div>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n <script type=\"module\">\n const combobox = document.querySelector('#combobox-select-all');\n const [selectAll, deselectAll] = Array.from(combobox.querySelectorAll('nve-button'));\n\n selectAll.addEventListener('click', () => combobox.selectAll());\n deselectAll.addEventListener('click', () => combobox.reset());\n combobox.addEventListener('change', e => console.log(e.target.selectedOptions));\n </script>\n `\n};\n\n/**\n * @summary Multi-select with disabled options to show unavailable choices while maintaining visual context.\n */\nexport const DisabledOptions = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\" disabled></option>\n <option value=\"priority\" disabled></option>\n <option value=\"date\" disabled></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select without built-in tags, using external tag management for custom selection display.\n */\nexport const NoTags = () => {\n return html`\n <form id=\"notags\" nve-layout=\"column gap:lg align:stretch\">\n <nve-combobox notags>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option selected value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n </nve-combobox>\n <div id=\"tags\" nve-layout=\"row gap:xs\">\n </div>\n </form>\n <script type=\"module\">\n const form = document.querySelector('#notags');\n const select = form.querySelector('select');\n const tags = form.querySelector('#tags');\n updateTags();\n select.addEventListener('change', e => updateTags());\n tags.addEventListener('close', e => {\n Array.from(select.options).find(o => o.value === e.target.value).selected = false;\n updateTags();\n });\n function updateTags() {\n tags.innerHTML = '';\n Array.from(select.selectedOptions).forEach(o => tags.innerHTML += '<nve-tag closable value=\"' + o.value + '\">' + o.value + '</nve-tag>');\n }\n </script>\n `\n};\n\n/**\n * @summary Complete form integration showing combobox with form submission, reset, and programmatic value setting.\n */\nexport const Form = () => {\n return html`\n<form nve-layout=\"column gap:lg align:stretch\">\n <nve-combobox style=\"--scroll-height: 220px\">\n <label>label</label>\n <input type=\"search\" name=\"input\">\n <select multiple name=\"select\" value=\"priority\">\n <option value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option selected value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <div nve-layout=\"row gap:xs\">\n <nve-button type=\"button\">set</nve-button>\n <nve-button type=\"reset\">reset</nve-button>\n <nve-button type=\"submit\">submit</nve-button>\n </div>\n</form>\n\n<script type=\"module\">\n const form = document.querySelector('form');\n const select = document.querySelector('select');\n const input = document.querySelector('input');\n const btn = document.querySelector('[type=button]');\n\n form.addEventListener('submit', e => {\n e.preventDefault();\n const formData = new FormData(form);\n console.log('input: ', formData.get('input'));\n console.log('select: ', formData.get('select'));\n console.log('selectedOptions: ', Array.from(select.selectedOptions).map(o => o.value));\n });\n\n btn.addEventListener('click', () => {\n select.value = 'status';\n select.options[0].selected = true;\n input.value = 'test';\n });\n</script>\n `\n};\n\n/**\n * @summary Fetches options asynchronously as the user types, cancelling stale requests with AbortController. Use for server-backed search where the full option set is too large to load up front.\n * @tags pattern\n */\nexport const DynamicTypeaheadSearch = () => {\n return html`\n<nve-combobox id=\"combobox\">\n <label>GPU Search</label>\n <input type=\"search\" placeholder=\"Type to search…\" />\n <datalist>\n <option disabled selected></option>\n </datalist>\n</nve-combobox>\n\n<script type=\"module\">\n const combobox = document.getElementById('combobox');\n const input = combobox.querySelector('input');\n const datalist = combobox.querySelector('datalist');\n\n let controller = null;\n input.addEventListener('input', async () => {\n if (controller) controller.abort();\n controller = new AbortController();\n\n const query = input.value.trim();\n if (!query) {\n datalist.innerHTML = '';\n return;\n }\n try {\n const results = await mockFetch(query, controller.signal);\n datalist.innerHTML = results.map((v) => '<option value=\"' + v + '\">').join('');\n } catch (err) {\n if (err.name !== 'AbortError') datalist.innerHTML = '';\n }\n });\n\n function mockFetch(query, signal) {\n return new Promise((resolve, reject) => {\n const timer = setTimeout(() => {\n resolve(['A100 GPU', 'H100 GPU', 'H200 GPU', 'DGX A100', 'DGX H100', 'CUDA Toolkit'].filter((v) => v.toLowerCase().startsWith(query.toLowerCase())));\n }, 500);\n signal.addEventListener('abort', () => {\n clearTimeout(timer);\n reject(new DOMException('Aborted', 'AbortError'));\n });\n });\n }\n</script>\n `\n}\n\n/**\n * @summary Performance test with 1000 options to show filtering efficiency with large datasets.\n * @tags test-case performance\n */\nexport const Performance = () => {\n return html`\n<nve-combobox id=\"performance-combobox\">\n <input type=\"search\" aria-label=\"performance test\">\n <datalist></datalist>\n</nve-combobox>\n<script type=\"module\">\n const datalist = document.querySelector('#performance-combobox datalist');\n const options = new Array(1000).fill('').map((_, i) => {\n const option = document.createElement('option');\n option.value = i + ' item';\n return option;\n });\n datalist.append(...options);\n</script>`\n}\n\n/* eslint-disable @nvidia-elements/lint/no-missing-slotted-elements */\n/**\n * @summary Performance test with 1000 options to show filtering efficiency with large datasets.\n * @tags test-case performance\n */\nexport const PerformanceSelect = () => {\n return html`\n<div nve-layout=\"pad:lg\">\n <nve-combobox id=\"performance-combobox\">\n <input type=\"search\" aria-label=\"performance test\">\n <select multiple></select>\n </nve-combobox>\n</div>\n<script type=\"module\">\n const select = document.querySelector('#performance-combobox select');\n const options = new Array(1000).fill('').map((_, i) => {\n const option = document.createElement('option');\n option.value = i + ' item';\n return option;\n });\n select.append(...options);\n</script>`\n}\n\n/**\n * @summary Dynamic options with datalist and select variants of combobox.\n * @tags test-case\n */\nexport const DynamicOptions = () => {\n return html`\n<nve-combobox id=\"dynamic-options-combobox\">\n <input type=\"search\" aria-label=\"performance test\" />\n <datalist>\n <option>default</option>\n </datalist>\n</nve-combobox>\n<script type=\"module\">\n let i = 0;\n setInterval(function() {\n if (i > 100) clearInterval(interval);\n const datalist = document.querySelector(\"#dynamic-options-combobox datalist\");\n const option = document.createElement(\"option\");\n option.value = i + \" item\";\n datalist.append(option);\n i++;\n console.log('append');\n }, 1000);\n</script>\n `\n}\n\n/**\n * @summary Interactive demo showing progressive filter chips with dynamic combobox creation for complex filtering interfaces.\n * @tags test-case\n */\nexport const FilterDemo = {\n render: () => html`<combobox-demo></combobox-demo>`\n}\n\nconst schema = {\n status: {\n type: 'select',\n options: ['success', 'failure', 'processing'],\n initial: 'success'\n },\n priority: {\n type: 'select',\n options: ['high', 'medium', 'low'],\n initial: 'high'\n },\n created: {\n type: 'date',\n initial: new Date()\n },\n progress: {\n type: 'number',\n initial: 0\n },\n sessionId: {\n type: 'text',\n initial: ''\n }\n};\n\n\nclass ComboboxDemo extends LitElement {\n @state() private value = [{ name: '', value: '' }];\n\n // todo\n /* eslint-disable @nvidia-elements/lint/no-deprecated-popover-attributes */\n render() {\n return html`\n <nve-button id=\"filter-btn\" ?pressed=${!!this.value.filter(v => v.name.length).length}><nve-icon name=\"filter\"></nve-icon> </nve-icon>filters</nve-button>\n <nve-dropdown id=\"one\" hidden trigger=\"filter-btn\" anchor=\"filter-btn\" @open=${e => e.target.hidden = false} @close=${e => e.target.hidden = true} style=\"--min-width: 400px; --min-height: 500px;\">\n <progressive-filter-demo @change=${e => this.value = e.detail} .value=${this.value} .schema=${schema}></progressive-filter-demo>\n </nve-dropdown>\n <pre style=\"margin-top: 300px\">${JSON.stringify(this.value.filter(v => v.name.length), null, 2)}</pre>\n `;\n }\n}\n\ncustomElements.get('combobox-demo') || customElements.define('combobox-demo', ComboboxDemo);\n\nclass ProgressiveFilterDemo extends LitElement {\n static styles = [unsafeCSS(`\n :host {\n display: flex;\n flex-direction: column;\n gap: var(--nve-ref-space-xs);\n }\n\n nve-progressive-filter-chip {\n width: 100%;\n }\n `)];\n\n @property({ type: Object }) schema = {};\n\n @property({ type: Array }) value: { name: string, value: string }[] = [{ name: '', value: '' }];\n\n get #unusedFilters() {\n return Object.entries(this.schema).filter(([key]) => !this.value.find(f => f.name === key));\n }\n\n render() {\n return html`\n ${this.value.map(filter => html`\n <nve-progressive-filter-chip closable @close=${() => this.#removeFilter(filter)}>\n <nve-combobox>\n <span slot=\"prefix-icon\"></span>\n <input type=\"search\" placeholder=\"filter\" .value=${filter.name} @change=${e => this.#createfilter(e.target.value, filter)} aria-label=\"filter\" />\n <datalist>${this.#unusedFilters.map(([key]) => html`<option .value=${key}>${key}</option>`)}</datalist>\n </nve-combobox>\n ${choose(this.schema[filter.name]?.type, [\n ['text', () => html`<nve-combobox><input type=\"text\" @change=${e => this.#updateFilter(e.target.value, filter)} .value=${filter.value} placeholder=\"value\" aria-label=\"filter value\" /></nve-combobox>`],\n ['number', () => html`<nve-combobox><input type=\"number\" @change=${e => this.#updateFilter(e.target.value, filter)} .value=${filter.value} aria-label=\"filter value\" /></nve-combobox>`],\n ['date', () => html`<nve-date><input type=\"date\" @change=${e => this.#updateFilter(e.target.value, filter)} .value=${filter.value} aria-label=\"filter value\" /></nve-date>`],\n ['select', () => html`<nve-select><select @change=${e => this.#updateFilter(e.target.value, filter)} value=${filter.value} aria-label=\"filter value\">${this.schema[filter.name]?.options?.map(v => html`<option value=${v}>${v}</option>`)}</select></nve-select>`]\n ], () => html`<nve-combobox><input type=\"text\" placeholder=\"value\" disabled aria-label=\"filter value\" /></nve-combobox>`)}\n </nve-progressive-filter-chip>`)}\n <nve-button container=\"flat\" @click=${this.#addFilter} .disabled=${this.#unusedFilters.length === 0 || !!this.value.find(v => v.name === '')} style=\"align: center; margin-top: 12px;\">\n <nve-icon name=\"add\"></nve-icon> Add Filter\n </nve-button>\n `;\n }\n\n #addFilter() {\n this.value = [...this.value, { name: '', value: '' }];\n this.#valueChange();\n }\n\n #removeFilter(filter: { name: string, value: string }) {\n this.value = this.value.filter(o => o.name !== filter.name);\n this.#valueChange();\n }\n\n #updateFilter(value: string, filter: { name: string, value: string }) {\n this.value = this.value.map(v => v.name === filter.name ? { ...filter, value } : v);\n this.#valueChange();\n }\n\n #createfilter(name, filter: { name: string, value: string }) {\n this.value = this.value.map(v => v.name === filter.name ? { name, value: this.schema[name]?.initial ?? '' } : v);\n this.#valueChange();\n }\n\n #valueChange() {\n this.dispatchEvent(new CustomEvent('change', { detail: this.value }));\n }\n}\n\ncustomElements.get('progressive-filter-demo') || customElements.define('progressive-filter-demo', ProgressiveFilterDemo);\n"],"mappings":";AAGA,IAAA,IAAe"}
1
+ {"version":3,"file":"combobox.examples.js","names":["#unusedFilters","#removeFilter","#createfilter","#updateFilter","#addFilter","#valueChange"],"sources":["../../src/combobox/combobox.examples.ts"],"sourcesContent":["// SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { html, LitElement, unsafeCSS } from 'lit';\nimport { property } from 'lit/decorators/property.js';\nimport { state } from 'lit/decorators/state.js';\nimport { choose } from 'lit/directives/choose.js';\nimport '@nvidia-elements/core/progressive-filter-chip/define.js';\nimport '@nvidia-elements/core/combobox/define.js';\nimport '@nvidia-elements/core/icon/define.js';\nimport '@nvidia-elements/core/tag/define.js';\nimport '@nvidia-elements/core/date/define.js';\nimport '@nvidia-elements/core/select/define.js';\nimport '@nvidia-elements/core/input/define.js';\nimport '@nvidia-elements/core/dropdown/define.js';\n\nexport default {\n title: 'Elements/Combobox',\n component: 'nve-combobox',\n};\n\n/**\n * @summary Basic combobox with search input and datalist options for filtering and selection.\n */\nexport const Default = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Vertical layout showing different combobox states (normal, disabled, success, error) stacked for comparison.\n * @tags test-case\n */\nexport const Vertical = () => {\n return html`\n<div nve-layout=\"column gap:lg full\">\n <nve-combobox>\n <label>label</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox>\n <label>disabled</label>\n <input type=\"search\" disabled />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox>\n <label>success</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"success\">message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox>\n <label>error</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"error\">message</nve-control-message>\n </nve-combobox>\n</div>`\n};\n\n/**\n * @summary Horizontal layout showing different combobox states (normal, disabled, success, error) for inline form layouts.\n */\nexport const Horizontal = () => {\n return html`\n<div nve-layout=\"column gap:lg full\">\n <nve-combobox layout=\"horizontal\">\n <label>label</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox layout=\"horizontal\">\n <label>disabled</label>\n <input type=\"search\" disabled />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox layout=\"horizontal\">\n <label>success</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"success\">message</nve-control-message>\n </nve-combobox>\n\n <nve-combobox layout=\"horizontal\">\n <label>error</label>\n <input type=\"search\" />\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n <nve-control-message status=\"error\">message</nve-control-message>\n </nve-combobox>\n</div>`\n};\n\n/**\n * @summary Flat container style with prefix icon for compact inline filtering interfaces.\n */\nexport const Flat = () => {\n return html`\n <nve-combobox container=\"flat\">\n <nve-icon name=\"filter\" slot=\"prefix-icon\"></nve-icon>\n <input type=\"search\" aria-label=\"search\">\n <datalist>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </datalist>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Single select allows users only to select from a predefined list\n * of options. Invalid input is automatically cleared. All options are visible\n * on focus until typing begins for filtering.\n */\nexport const Select = () => {\n return html`\n<nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n</nve-combobox>\n `\n};\n\n/**\n * @summary Multi select allows users to select many options from a\n * predefined list. The select `value` will only reflect the first selected value.\n * To get all selected options check the `selected` property on each `<option>`\n * or the select property `selectedOptions`.\n * On focus all options will show until the user starts typing. Select is the\n * selection value of the combobox. The input is the filter value.\n */\nexport const MultiSelect = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option selected value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Combobox with an empty initial value using a disabled placeholder option. Use when no default selection exists and the user must make an explicit choice.\n */\nexport const EmptyDefault = () => {\n return html`\n<nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select>\n <option disabled selected></option>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n</nve-combobox>\n `\n};\n\n/**\n * @summary Combobox options with display labels that differ from underlying values. Use when option values are IDs or codes but users need to see human-readable text in the input.\n */\nexport const Label = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select>\n <option value=\"1\">Status</option>\n <option value=\"2\">Priority</option>\n <option value=\"3\">Date</option>\n <option value=\"4\">Session</option>\n <option value=\"5\">Configuration</option>\n <option value=\"6\">Contains</option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select with labeled options where display text differs from option values.\n */\nexport const LabelMultiSelect = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"1\">Status</option>\n <option value=\"2\">Priority</option>\n <option value=\"3\">Date</option>\n <option value=\"4\">Session</option>\n <option value=\"5\">Configuration</option>\n <option value=\"6\">Contains</option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Overflow behavior where many tags collapse into a simple text label when the parent container is too narrow.\n * @tags test-case\n */\nexport const Overflow = () => {\n return html`\n <nve-combobox style=\"width: 250px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option selected value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option selected value=\"date\"></option>\n <option selected value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Combobox handling of long option text in constrained width containers.\n * @tags test-case\n */\nexport const PopoverOverflow = () => {\n return html`\n <nve-combobox style=\"width: 100px\">\n <label>label</label>\n <input type=\"search\">\n <select>\n <option value=\"really-long-text-option-that-keeps-going\"></option> \n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select with reset functionality via icon button and footer button to clear all selections.\n */\nexport const Reset = () => {\n return html`\n <nve-combobox id=\"combobox-reset\" style=\"width: 500px; --scroll-height: 220px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-icon-button aria-label=\"clear selection\" icon-name=\"cancel\" container=\"inline\"></nve-icon-button>\n <nve-button slot=\"footer\" aria-label=\"clear selection\" container=\"flat\">reset</nve-button>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n <script type=\"module\">\n document.querySelector('#combobox-reset nve-icon-button').addEventListener('click', e => e.target.parentElement.reset());\n document.querySelector('#combobox-reset nve-button').addEventListener('click', e => e.target.parentElement.reset());\n </script>\n `\n};\n\n/**\n * @summary Multi-select with footer action button for extra operations on selected items.\n */\nexport const Footer = () => {\n return html`\n <nve-combobox style=\"width: 500px; --scroll-height: 200px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n <option value=\"includes\"></option>\n <option value=\"user\"></option>\n <option value=\"progress\"></option>\n </select>\n <nve-button slot=\"footer\" container=\"flat\">action</nve-button>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select with bulk selection controls (Select All/Deselect All) in footer for efficient mass operations.\n */\nexport const SelectAll = () => {\n return html`\n <nve-combobox id=\"combobox-select-all\" style=\"width: 500px; --scroll-height: 200px\">\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\"></option>\n <option value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n <option value=\"includes\"></option>\n <option value=\"user\"></option>\n <option value=\"progress\"></option>\n </select>\n <div slot=\"footer\" nve-layout=\"row align:stretch full\">\n <nve-button container=\"flat\">Select All</nve-button>\n <nve-button container=\"flat\">Deselect All</nve-button>\n </div>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n <script type=\"module\">\n const combobox = document.querySelector('#combobox-select-all');\n const [selectAll, deselectAll] = Array.from(combobox.querySelectorAll('nve-button'));\n\n selectAll.addEventListener('click', () => combobox.selectAll());\n deselectAll.addEventListener('click', () => combobox.reset());\n combobox.addEventListener('change', e => console.log(e.target.selectedOptions));\n </script>\n `\n};\n\n/**\n * @summary Multi-select with disabled options to show unavailable choices while maintaining visual context.\n */\nexport const DisabledOptions = () => {\n return html`\n <nve-combobox>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option value=\"status\" disabled></option>\n <option value=\"priority\" disabled></option>\n <option value=\"date\" disabled></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n `\n};\n\n/**\n * @summary Multi-select without built-in tags, using external tag management for custom selection display.\n */\nexport const NoTags = () => {\n return html`\n <form id=\"notags\" nve-layout=\"column gap:lg align:stretch\">\n <nve-combobox notags>\n <label>label</label>\n <input type=\"search\">\n <select multiple>\n <option selected value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n </nve-combobox>\n <div id=\"tags\" nve-layout=\"row gap:xs\">\n </div>\n </form>\n <script type=\"module\">\n const form = document.querySelector('#notags');\n const select = form.querySelector('select');\n const tags = form.querySelector('#tags');\n updateTags();\n select.addEventListener('change', e => updateTags());\n tags.addEventListener('close', e => {\n Array.from(select.options).find(o => o.value === e.target.value).selected = false;\n updateTags();\n });\n function updateTags() {\n tags.innerHTML = '';\n Array.from(select.selectedOptions).forEach(o => tags.innerHTML += '<nve-tag closable value=\"' + o.value + '\">' + o.value + '</nve-tag>');\n }\n </script>\n `\n};\n\n/**\n * @summary Complete form integration showing combobox with form submission, reset, and programmatic value setting.\n */\nexport const Form = () => {\n return html`\n<form nve-layout=\"column gap:lg align:stretch\">\n <nve-combobox style=\"--scroll-height: 220px\">\n <label>label</label>\n <input type=\"search\" name=\"input\">\n <select multiple name=\"select\" value=\"priority\">\n <option value=\"status\"></option>\n <option selected value=\"priority\"></option>\n <option selected value=\"date\"></option>\n <option value=\"session\"></option>\n <option value=\"configuration\"></option>\n <option value=\"contains\"></option>\n </select>\n <nve-control-message>message</nve-control-message>\n </nve-combobox>\n\n <div nve-layout=\"row gap:xs\">\n <nve-button type=\"button\">set</nve-button>\n <nve-button type=\"reset\">reset</nve-button>\n <nve-button type=\"submit\">submit</nve-button>\n </div>\n</form>\n\n<script type=\"module\">\n const form = document.querySelector('form');\n const select = document.querySelector('select');\n const input = document.querySelector('input');\n const btn = document.querySelector('[type=button]');\n\n form.addEventListener('submit', e => {\n e.preventDefault();\n const formData = new FormData(form);\n console.log('input: ', formData.get('input'));\n console.log('select: ', formData.get('select'));\n console.log('selectedOptions: ', Array.from(select.selectedOptions).map(o => o.value));\n });\n\n btn.addEventListener('click', () => {\n select.value = 'status';\n select.options[0].selected = true;\n input.value = 'test';\n });\n</script>\n `\n};\n\n/**\n * @summary Fetches options asynchronously as the user types, cancelling stale requests with AbortController. Use for server-backed search where the full option set is too large to load up front.\n * @tags pattern\n */\nexport const DynamicTypeaheadSearch = () => {\n return html`\n<nve-combobox id=\"combobox\">\n <label>GPU Search</label>\n <input type=\"search\" placeholder=\"Type to search…\" />\n <datalist>\n <option disabled selected></option>\n </datalist>\n</nve-combobox>\n\n<script type=\"module\">\n const combobox = document.getElementById('combobox');\n const input = combobox.querySelector('input');\n const datalist = combobox.querySelector('datalist');\n\n let controller = null;\n input.addEventListener('input', async () => {\n if (controller) controller.abort();\n controller = new AbortController();\n\n const query = input.value.trim();\n if (!query) {\n datalist.innerHTML = '';\n return;\n }\n try {\n const results = await mockFetch(query, controller.signal);\n datalist.innerHTML = results.map((v) => '<option value=\"' + v + '\">').join('');\n } catch (err) {\n if (err.name !== 'AbortError') datalist.innerHTML = '';\n }\n });\n\n function mockFetch(query, signal) {\n return new Promise((resolve, reject) => {\n const timer = setTimeout(() => {\n resolve(['A100 GPU', 'H100 GPU', 'H200 GPU', 'DGX A100', 'DGX H100', 'CUDA Toolkit'].filter((v) => v.toLowerCase().startsWith(query.toLowerCase())));\n }, 500);\n signal.addEventListener('abort', () => {\n clearTimeout(timer);\n reject(new DOMException('Aborted', 'AbortError'));\n });\n });\n }\n</script>\n `\n}\n\n/**\n * @summary Infinite scroll combobox loading, using the scroll event to append options as the user nears the bottom. Use for server-backed datasets where loading all options up front is impractical.\n * @tags pattern\n */\nexport const InfiniteScroll = () => {\n return html`\n<nve-combobox id=\"infinite-scroll-combo\" style=\"--scroll-height: 200px\">\n <label>GPU Models</label>\n <input type=\"search\" placeholder=\"Scroll to load more…\" />\n <datalist id=\"infinite-scroll-list\"></datalist>\n</nve-combobox>\n\n<script type=\"module\">\n const combo = document.getElementById('infinite-scroll-combo');\n const datalist = document.getElementById('infinite-scroll-list');\n let loading = false;\n\n async function loadBatch() {\n if (loading) return;\n loading = true;\n const items = await new Promise(resolve => setTimeout(() => resolve(Array.from({ length: 100 }, (_, i) => 'GPU Model ' + (datalist.options.length + i + 1))), 300));\n datalist.append(...items.map(v => new Option(v)));\n loading = false;\n }\n\n loadBatch();\n\n combo.addEventListener('scroll', (e) => {\n if (e.detail.scrollHeight - e.detail.scrollTop - e.detail.clientHeight <= 128) {\n loadBatch();\n }\n });\n</script>\n `\n}\n\n/**\n * @summary Performance test with 1000 options to show filtering efficiency with large datasets.\n * @tags test-case performance\n */\nexport const Performance = () => {\n return html`\n<nve-combobox id=\"performance-combobox\">\n <input type=\"search\" aria-label=\"performance test\">\n <datalist></datalist>\n</nve-combobox>\n<script type=\"module\">\n const datalist = document.querySelector('#performance-combobox datalist');\n const options = new Array(1000).fill('').map((_, i) => {\n const option = document.createElement('option');\n option.value = i + ' item';\n return option;\n });\n datalist.append(...options);\n</script>`\n}\n\n/* eslint-disable @nvidia-elements/lint/no-missing-slotted-elements */\n/**\n * @summary Performance test with 1000 options to show filtering efficiency with large datasets.\n * @tags test-case performance\n */\nexport const PerformanceSelect = () => {\n return html`\n<div nve-layout=\"pad:lg\">\n <nve-combobox id=\"performance-combobox\">\n <input type=\"search\" aria-label=\"performance test\">\n <select multiple></select>\n </nve-combobox>\n</div>\n<script type=\"module\">\n const select = document.querySelector('#performance-combobox select');\n const options = new Array(1000).fill('').map((_, i) => {\n const option = document.createElement('option');\n option.value = i + ' item';\n return option;\n });\n select.append(...options);\n</script>`\n}\n\n/**\n * @summary Dynamic options with datalist and select variants of combobox.\n * @tags test-case\n */\nexport const DynamicOptions = () => {\n return html`\n<nve-combobox id=\"dynamic-options-combobox\">\n <input type=\"search\" aria-label=\"performance test\" />\n <datalist>\n <option>default</option>\n </datalist>\n</nve-combobox>\n<script type=\"module\">\n let i = 0;\n setInterval(function() {\n if (i > 100) clearInterval(interval);\n const datalist = document.querySelector(\"#dynamic-options-combobox datalist\");\n const option = document.createElement(\"option\");\n option.value = i + \" item\";\n datalist.append(option);\n i++;\n console.log('append');\n }, 1000);\n</script>\n `\n}\n\n/**\n * @summary Interactive demo showing progressive filter chips with dynamic combobox creation for complex filtering interfaces.\n * @tags test-case\n */\nexport const FilterDemo = {\n render: () => html`<combobox-demo></combobox-demo>`\n}\n\nconst schema = {\n status: {\n type: 'select',\n options: ['success', 'failure', 'processing'],\n initial: 'success'\n },\n priority: {\n type: 'select',\n options: ['high', 'medium', 'low'],\n initial: 'high'\n },\n created: {\n type: 'date',\n initial: new Date()\n },\n progress: {\n type: 'number',\n initial: 0\n },\n sessionId: {\n type: 'text',\n initial: ''\n }\n};\n\n\nclass ComboboxDemo extends LitElement {\n @state() private value = [{ name: '', value: '' }];\n\n // todo\n /* eslint-disable @nvidia-elements/lint/no-deprecated-popover-attributes */\n render() {\n return html`\n <nve-button id=\"filter-btn\" ?pressed=${!!this.value.filter(v => v.name.length).length}><nve-icon name=\"filter\"></nve-icon> </nve-icon>filters</nve-button>\n <nve-dropdown id=\"one\" hidden trigger=\"filter-btn\" anchor=\"filter-btn\" @open=${e => e.target.hidden = false} @close=${e => e.target.hidden = true} style=\"--min-width: 400px; --min-height: 500px;\">\n <progressive-filter-demo @change=${e => this.value = e.detail} .value=${this.value} .schema=${schema}></progressive-filter-demo>\n </nve-dropdown>\n <pre style=\"margin-top: 300px\">${JSON.stringify(this.value.filter(v => v.name.length), null, 2)}</pre>\n `;\n }\n}\n\ncustomElements.get('combobox-demo') || customElements.define('combobox-demo', ComboboxDemo);\n\nclass ProgressiveFilterDemo extends LitElement {\n static styles = [unsafeCSS(`\n :host {\n display: flex;\n flex-direction: column;\n gap: var(--nve-ref-space-xs);\n }\n\n nve-progressive-filter-chip {\n width: 100%;\n }\n `)];\n\n @property({ type: Object }) schema = {};\n\n @property({ type: Array }) value: { name: string, value: string }[] = [{ name: '', value: '' }];\n\n get #unusedFilters() {\n return Object.entries(this.schema).filter(([key]) => !this.value.find(f => f.name === key));\n }\n\n render() {\n return html`\n ${this.value.map(filter => html`\n <nve-progressive-filter-chip closable @close=${() => this.#removeFilter(filter)}>\n <nve-combobox>\n <span slot=\"prefix-icon\"></span>\n <input type=\"search\" placeholder=\"filter\" .value=${filter.name} @change=${e => this.#createfilter(e.target.value, filter)} aria-label=\"filter\" />\n <datalist>${this.#unusedFilters.map(([key]) => html`<option .value=${key}>${key}</option>`)}</datalist>\n </nve-combobox>\n ${choose(this.schema[filter.name]?.type, [\n ['text', () => html`<nve-combobox><input type=\"text\" @change=${e => this.#updateFilter(e.target.value, filter)} .value=${filter.value} placeholder=\"value\" aria-label=\"filter value\" /></nve-combobox>`],\n ['number', () => html`<nve-combobox><input type=\"number\" @change=${e => this.#updateFilter(e.target.value, filter)} .value=${filter.value} aria-label=\"filter value\" /></nve-combobox>`],\n ['date', () => html`<nve-date><input type=\"date\" @change=${e => this.#updateFilter(e.target.value, filter)} .value=${filter.value} aria-label=\"filter value\" /></nve-date>`],\n ['select', () => html`<nve-select><select @change=${e => this.#updateFilter(e.target.value, filter)} value=${filter.value} aria-label=\"filter value\">${this.schema[filter.name]?.options?.map(v => html`<option value=${v}>${v}</option>`)}</select></nve-select>`]\n ], () => html`<nve-combobox><input type=\"text\" placeholder=\"value\" disabled aria-label=\"filter value\" /></nve-combobox>`)}\n </nve-progressive-filter-chip>`)}\n <nve-button container=\"flat\" @click=${this.#addFilter} .disabled=${this.#unusedFilters.length === 0 || !!this.value.find(v => v.name === '')} style=\"align: center; margin-top: 12px;\">\n <nve-icon name=\"add\"></nve-icon> Add Filter\n </nve-button>\n `;\n }\n\n #addFilter() {\n this.value = [...this.value, { name: '', value: '' }];\n this.#valueChange();\n }\n\n #removeFilter(filter: { name: string, value: string }) {\n this.value = this.value.filter(o => o.name !== filter.name);\n this.#valueChange();\n }\n\n #updateFilter(value: string, filter: { name: string, value: string }) {\n this.value = this.value.map(v => v.name === filter.name ? { ...filter, value } : v);\n this.#valueChange();\n }\n\n #createfilter(name, filter: { name: string, value: string }) {\n this.value = this.value.map(v => v.name === filter.name ? { name, value: this.schema[name]?.initial ?? '' } : v);\n this.#valueChange();\n }\n\n #valueChange() {\n this.dispatchEvent(new CustomEvent('change', { detail: this.value }));\n }\n}\n\ncustomElements.get('progressive-filter-demo') || customElements.define('progressive-filter-demo', ProgressiveFilterDemo);\n"],"mappings":";AAGA,IAAA,IAAe"}
@@ -172,6 +172,17 @@
172
172
  "pattern"
173
173
  ]
174
174
  },
175
+ {
176
+ "id": "core-combobox_infinite-scroll",
177
+ "name": "InfiniteScroll",
178
+ "template": "<nve-combobox id=\"infinite-scroll-combo\" style=\"--scroll-height: 200px\">\n <label>GPU Models</label>\n <input type=\"search\" placeholder=\"Scroll to load more…\" />\n <datalist id=\"infinite-scroll-list\"></datalist>\n</nve-combobox>\n<script type=\"module\">\n const combo = document.getElementById(\"infinite-scroll-combo\");\n const datalist = document.getElementById(\"infinite-scroll-list\");\n let loading = false;\n async function loadBatch() {\n if (loading) return;\n loading = true;\n const items = await new Promise((resolve) =>\n setTimeout(\n () => resolve(Array.from({ length: 100 }, (_, i) => \"GPU Model \" + (datalist.options.length + i + 1))),\n 300,\n ),\n );\n datalist.append(...items.map((v) => new Option(v)));\n loading = false;\n }\n loadBatch();\n combo.addEventListener(\"scroll\", (e) => {\n if (e.detail.scrollHeight - e.detail.scrollTop - e.detail.clientHeight <= 128) {\n loadBatch();\n }\n });\n</script>\n",
179
+ "summary": "Infinite scroll combobox loading, using the scroll event to append options as the user nears the bottom. Use for server-backed datasets where loading all options up front is impractical.",
180
+ "description": "",
181
+ "composition": false,
182
+ "tags": [
183
+ "pattern"
184
+ ]
185
+ },
175
186
  {
176
187
  "id": "core-combobox_performance",
177
188
  "name": "Performance",