@transferwise/components 0.0.0-experimental-e9426b6 → 0.0.0-experimental-ce46fbc

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 (172) hide show
  1. package/build/dateInput/DateInput.js +3 -6
  2. package/build/dateInput/DateInput.js.map +1 -1
  3. package/build/dateInput/DateInput.mjs +2 -5
  4. package/build/dateInput/DateInput.mjs.map +1 -1
  5. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.js +3 -5
  6. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.js.map +1 -1
  7. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.mjs +1 -3
  8. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.mjs.map +1 -1
  9. package/build/index.js +3 -5
  10. package/build/index.js.map +1 -1
  11. package/build/index.mjs +1 -3
  12. package/build/index.mjs.map +1 -1
  13. package/build/inputs/SelectInput.js +821 -0
  14. package/build/inputs/SelectInput.js.map +1 -0
  15. package/build/inputs/SelectInput.messages.js.map +1 -0
  16. package/build/inputs/SelectInput.messages.mjs.map +1 -0
  17. package/build/inputs/SelectInput.mjs +813 -0
  18. package/build/inputs/SelectInput.mjs.map +1 -0
  19. package/build/main.css +47 -47
  20. package/build/moneyInput/MoneyInput.js +2 -5
  21. package/build/moneyInput/MoneyInput.js.map +1 -1
  22. package/build/moneyInput/MoneyInput.mjs +1 -4
  23. package/build/moneyInput/MoneyInput.mjs.map +1 -1
  24. package/build/phoneNumberInput/PhoneNumberInput.js +2 -5
  25. package/build/phoneNumberInput/PhoneNumberInput.js.map +1 -1
  26. package/build/phoneNumberInput/PhoneNumberInput.mjs +1 -4
  27. package/build/phoneNumberInput/PhoneNumberInput.mjs.map +1 -1
  28. package/build/styles/main.css +47 -47
  29. package/build/types/inputs/{SelectInput/SelectInput.types.d.ts → SelectInput.d.ts} +7 -4
  30. package/build/types/inputs/SelectInput.d.ts.map +1 -0
  31. package/build/types/inputs/SelectInput.messages.d.ts.map +1 -0
  32. package/package.json +1 -1
  33. package/src/inputs/{SelectInput/SelectInput.docs.mdx → SelectInput.docs.mdx} +1 -0
  34. package/src/inputs/SelectInput.less +219 -0
  35. package/src/inputs/{SelectInput/SelectInput.story.tsx → SelectInput.story.tsx} +7 -7
  36. package/src/inputs/SelectInput.tsx +1190 -0
  37. package/src/listItem/_stories/ListItem.story.tsx +76 -1
  38. package/src/main.css +47 -47
  39. package/src/main.less +1 -1
  40. package/build/inputs/SelectInput/SelectInput.helpers.js +0 -115
  41. package/build/inputs/SelectInput/SelectInput.helpers.js.map +0 -1
  42. package/build/inputs/SelectInput/SelectInput.helpers.mjs +0 -109
  43. package/build/inputs/SelectInput/SelectInput.helpers.mjs.map +0 -1
  44. package/build/inputs/SelectInput/SelectInput.js +0 -216
  45. package/build/inputs/SelectInput/SelectInput.js.map +0 -1
  46. package/build/inputs/SelectInput/SelectInput.messages.js.map +0 -1
  47. package/build/inputs/SelectInput/SelectInput.messages.mjs.map +0 -1
  48. package/build/inputs/SelectInput/SelectInput.mjs +0 -210
  49. package/build/inputs/SelectInput/SelectInput.mjs.map +0 -1
  50. package/build/inputs/SelectInput/components/SelectInputClearButton/SelectInputClearButton.js +0 -26
  51. package/build/inputs/SelectInput/components/SelectInputClearButton/SelectInputClearButton.js.map +0 -1
  52. package/build/inputs/SelectInput/components/SelectInputClearButton/SelectInputClearButton.mjs +0 -24
  53. package/build/inputs/SelectInput/components/SelectInputClearButton/SelectInputClearButton.mjs.map +0 -1
  54. package/build/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.js +0 -54
  55. package/build/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.js.map +0 -1
  56. package/build/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.mjs +0 -52
  57. package/build/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.mjs.map +0 -1
  58. package/build/inputs/SelectInput/components/SelectInputGroupItemView/SelectInputGroupItemView.js +0 -50
  59. package/build/inputs/SelectInput/components/SelectInputGroupItemView/SelectInputGroupItemView.js.map +0 -1
  60. package/build/inputs/SelectInput/components/SelectInputGroupItemView/SelectInputGroupItemView.mjs +0 -48
  61. package/build/inputs/SelectInput/components/SelectInputGroupItemView/SelectInputGroupItemView.mjs.map +0 -1
  62. package/build/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.js +0 -47
  63. package/build/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.js.map +0 -1
  64. package/build/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.mjs +0 -45
  65. package/build/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.mjs.map +0 -1
  66. package/build/inputs/SelectInput/components/SelectInputOption/SelectInputOption.js +0 -45
  67. package/build/inputs/SelectInput/components/SelectInputOption/SelectInputOption.js.map +0 -1
  68. package/build/inputs/SelectInput/components/SelectInputOption/SelectInputOption.mjs +0 -41
  69. package/build/inputs/SelectInput/components/SelectInputOption/SelectInputOption.mjs.map +0 -1
  70. package/build/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.js +0 -41
  71. package/build/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.js.map +0 -1
  72. package/build/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.mjs +0 -38
  73. package/build/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.mjs.map +0 -1
  74. package/build/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.js +0 -270
  75. package/build/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.js.map +0 -1
  76. package/build/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.mjs +0 -268
  77. package/build/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.mjs.map +0 -1
  78. package/build/inputs/SelectInput/components/SelectInputOptionsContainer/SelectInputOptionsContainer.js +0 -48
  79. package/build/inputs/SelectInput/components/SelectInputOptionsContainer/SelectInputOptionsContainer.js.map +0 -1
  80. package/build/inputs/SelectInput/components/SelectInputOptionsContainer/SelectInputOptionsContainer.mjs +0 -46
  81. package/build/inputs/SelectInput/components/SelectInputOptionsContainer/SelectInputOptionsContainer.mjs.map +0 -1
  82. package/build/inputs/SelectInput/components/SelectInputTriggerButton/SelectInputTriggerButton.js +0 -41
  83. package/build/inputs/SelectInput/components/SelectInputTriggerButton/SelectInputTriggerButton.js.map +0 -1
  84. package/build/inputs/SelectInput/components/SelectInputTriggerButton/SelectInputTriggerButton.mjs +0 -34
  85. package/build/inputs/SelectInput/components/SelectInputTriggerButton/SelectInputTriggerButton.mjs.map +0 -1
  86. package/build/styles/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.css +0 -17
  87. package/build/styles/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.css +0 -16
  88. package/build/styles/inputs/SelectInput/components/SelectInputOption/SelectInputOption.css +0 -33
  89. package/build/styles/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.css +0 -37
  90. package/build/types/inputs/SelectInput/SelectInput.d.ts +0 -3
  91. package/build/types/inputs/SelectInput/SelectInput.d.ts.map +0 -1
  92. package/build/types/inputs/SelectInput/SelectInput.helpers.d.ts +0 -28
  93. package/build/types/inputs/SelectInput/SelectInput.helpers.d.ts.map +0 -1
  94. package/build/types/inputs/SelectInput/SelectInput.messages.d.ts.map +0 -1
  95. package/build/types/inputs/SelectInput/SelectInput.types.d.ts.map +0 -1
  96. package/build/types/inputs/SelectInput/components/SelectInputClearButton/SelectInputClearButton.d.ts +0 -5
  97. package/build/types/inputs/SelectInput/components/SelectInputClearButton/SelectInputClearButton.d.ts.map +0 -1
  98. package/build/types/inputs/SelectInput/components/SelectInputClearButton/index.d.ts +0 -2
  99. package/build/types/inputs/SelectInput/components/SelectInputClearButton/index.d.ts.map +0 -1
  100. package/build/types/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.d.ts +0 -9
  101. package/build/types/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.d.ts.map +0 -1
  102. package/build/types/inputs/SelectInput/components/SelectInputDefaultTrigger/index.d.ts +0 -2
  103. package/build/types/inputs/SelectInput/components/SelectInputDefaultTrigger/index.d.ts.map +0 -1
  104. package/build/types/inputs/SelectInput/components/SelectInputGroupItemView/SelectInputGroupItemView.d.ts +0 -9
  105. package/build/types/inputs/SelectInput/components/SelectInputGroupItemView/SelectInputGroupItemView.d.ts.map +0 -1
  106. package/build/types/inputs/SelectInput/components/SelectInputGroupItemView/index.d.ts +0 -2
  107. package/build/types/inputs/SelectInput/components/SelectInputGroupItemView/index.d.ts.map +0 -1
  108. package/build/types/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.d.ts +0 -8
  109. package/build/types/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.d.ts.map +0 -1
  110. package/build/types/inputs/SelectInput/components/SelectInputItemView/index.d.ts +0 -2
  111. package/build/types/inputs/SelectInput/components/SelectInputItemView/index.d.ts.map +0 -1
  112. package/build/types/inputs/SelectInput/components/SelectInputOption/SelectInputOption.d.ts +0 -10
  113. package/build/types/inputs/SelectInput/components/SelectInputOption/SelectInputOption.d.ts.map +0 -1
  114. package/build/types/inputs/SelectInput/components/SelectInputOption/index.d.ts +0 -2
  115. package/build/types/inputs/SelectInput/components/SelectInputOption/index.d.ts.map +0 -1
  116. package/build/types/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.d.ts +0 -9
  117. package/build/types/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.d.ts.map +0 -1
  118. package/build/types/inputs/SelectInput/components/SelectInputOptionContent/index.d.ts +0 -3
  119. package/build/types/inputs/SelectInput/components/SelectInputOptionContent/index.d.ts.map +0 -1
  120. package/build/types/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.d.ts +0 -15
  121. package/build/types/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.d.ts.map +0 -1
  122. package/build/types/inputs/SelectInput/components/SelectInputOptions/index.d.ts +0 -2
  123. package/build/types/inputs/SelectInput/components/SelectInputOptions/index.d.ts.map +0 -1
  124. package/build/types/inputs/SelectInput/components/SelectInputOptionsContainer/SelectInputOptionsContainer.d.ts +0 -6
  125. package/build/types/inputs/SelectInput/components/SelectInputOptionsContainer/SelectInputOptionsContainer.d.ts.map +0 -1
  126. package/build/types/inputs/SelectInput/components/SelectInputOptionsContainer/index.d.ts +0 -2
  127. package/build/types/inputs/SelectInput/components/SelectInputOptionsContainer/index.d.ts.map +0 -1
  128. package/build/types/inputs/SelectInput/components/SelectInputTriggerButton/SelectInputTriggerButton.d.ts +0 -15
  129. package/build/types/inputs/SelectInput/components/SelectInputTriggerButton/SelectInputTriggerButton.d.ts.map +0 -1
  130. package/build/types/inputs/SelectInput/components/SelectInputTriggerButton/index.d.ts +0 -3
  131. package/build/types/inputs/SelectInput/components/SelectInputTriggerButton/index.d.ts.map +0 -1
  132. package/build/types/inputs/SelectInput/index.d.ts +0 -5
  133. package/build/types/inputs/SelectInput/index.d.ts.map +0 -1
  134. package/src/inputs/SelectInput/SelectInput.helpers.ts +0 -152
  135. package/src/inputs/SelectInput/SelectInput.less +0 -42
  136. package/src/inputs/SelectInput/SelectInput.test.tsx +0 -606
  137. package/src/inputs/SelectInput/SelectInput.tsx +0 -247
  138. package/src/inputs/SelectInput/SelectInput.types.ts +0 -114
  139. package/src/inputs/SelectInput/components/SelectInputClearButton/SelectInputClearButton.tsx +0 -25
  140. package/src/inputs/SelectInput/components/SelectInputClearButton/index.ts +0 -1
  141. package/src/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.css +0 -17
  142. package/src/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.less +0 -15
  143. package/src/inputs/SelectInput/components/SelectInputDefaultTrigger/SelectInputDefaultTrigger.tsx +0 -56
  144. package/src/inputs/SelectInput/components/SelectInputDefaultTrigger/index.ts +0 -1
  145. package/src/inputs/SelectInput/components/SelectInputGroupItemView/SelectInputGroupItemView.tsx +0 -64
  146. package/src/inputs/SelectInput/components/SelectInputGroupItemView/index.ts +0 -1
  147. package/src/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.css +0 -16
  148. package/src/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.less +0 -17
  149. package/src/inputs/SelectInput/components/SelectInputItemView/SelectInputItemView.tsx +0 -55
  150. package/src/inputs/SelectInput/components/SelectInputItemView/index.ts +0 -1
  151. package/src/inputs/SelectInput/components/SelectInputOption/SelectInputOption.css +0 -33
  152. package/src/inputs/SelectInput/components/SelectInputOption/SelectInputOption.less +0 -32
  153. package/src/inputs/SelectInput/components/SelectInputOption/SelectInputOption.tsx +0 -51
  154. package/src/inputs/SelectInput/components/SelectInputOption/index.ts +0 -5
  155. package/src/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.css +0 -37
  156. package/src/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.less +0 -38
  157. package/src/inputs/SelectInput/components/SelectInputOptionContent/SelectInputOptionContent.tsx +0 -67
  158. package/src/inputs/SelectInput/components/SelectInputOptionContent/index.ts +0 -5
  159. package/src/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.less +0 -75
  160. package/src/inputs/SelectInput/components/SelectInputOptions/SelectInputOptions.tsx +0 -369
  161. package/src/inputs/SelectInput/components/SelectInputOptions/index.ts +0 -1
  162. package/src/inputs/SelectInput/components/SelectInputOptionsContainer/SelectInputOptionsContainer.tsx +0 -56
  163. package/src/inputs/SelectInput/components/SelectInputOptionsContainer/index.ts +0 -1
  164. package/src/inputs/SelectInput/components/SelectInputTriggerButton/SelectInputTriggerButton.tsx +0 -39
  165. package/src/inputs/SelectInput/components/SelectInputTriggerButton/index.ts +0 -5
  166. package/src/inputs/SelectInput/index.ts +0 -13
  167. package/build/inputs/{SelectInput/SelectInput.messages.js → SelectInput.messages.js} +0 -0
  168. package/build/inputs/{SelectInput/SelectInput.messages.mjs → SelectInput.messages.mjs} +0 -0
  169. package/{src/inputs/SelectInput → build/styles/inputs}/SelectInput.css +47 -47
  170. package/build/types/inputs/{SelectInput/SelectInput.messages.d.ts → SelectInput.messages.d.ts} +0 -0
  171. package/{build/styles/inputs/SelectInput → src/inputs}/SelectInput.css +47 -47
  172. /package/src/inputs/{SelectInput/SelectInput.messages.ts → SelectInput.messages.ts} +0 -0
@@ -0,0 +1,1190 @@
1
+ import {
2
+ Listbox as ListboxBase,
3
+ ListboxButton,
4
+ ListboxOption,
5
+ ListboxOptions,
6
+ } from '@headlessui/react';
7
+ import { Check, ChevronDown, Cross, CrossCircle } from '@transferwise/icons';
8
+ import { clsx } from 'clsx';
9
+ import mergeProps from 'merge-props';
10
+ import {
11
+ createContext,
12
+ forwardRef,
13
+ ReactNode,
14
+ useContext,
15
+ useDeferredValue,
16
+ useEffect,
17
+ useId,
18
+ useMemo,
19
+ useRef,
20
+ useState,
21
+ } from 'react';
22
+ import { useIntl } from 'react-intl';
23
+ import { Virtualizer, type VirtualizerHandle } from 'virtua';
24
+
25
+ import { useEffectEvent } from '../common/hooks/useEffectEvent';
26
+ import { useScreenSize } from '../common/hooks/useScreenSize';
27
+ import { PolymorphicWithOverrides } from '../common/polymorphicWithOverrides/PolymorphicWithOverrides';
28
+ import { Breakpoint } from '../common/propsValues/breakpoint';
29
+ import dateTriggerMessages from '../dateLookup/dateTrigger/DateTrigger.messages';
30
+ import { Merge } from '../utils';
31
+
32
+ import { BottomSheet } from './_BottomSheet';
33
+ import { ButtonInput } from './_ButtonInput';
34
+ import { Popover } from './_Popover';
35
+ import { useInputAttributes, WithInputAttributesProps } from './contexts';
36
+ import { InputGroup } from './InputGroup';
37
+ import { SearchInput } from './SearchInput';
38
+ import messages from './SelectInput.messages';
39
+ import Header from '../header';
40
+ import Section from '../section';
41
+ import { ButtonProps } from '../button/Button.types';
42
+
43
+ const MAX_ITEMS_WITHOUT_VIRTUALIZATION = 50;
44
+
45
+ function searchableString(value: string) {
46
+ return (
47
+ value
48
+ .trim()
49
+ .replace(/\s+/gu, ' ')
50
+ // NFD converts an Å to A + ̊ (and other special characters)
51
+ .normalize('NFD')
52
+ // and then this replaces the ̊ with nothing (and other special characters)
53
+ .replace(/[\u0300-\u036f]/g, '')
54
+ .toLowerCase()
55
+ );
56
+ }
57
+
58
+ function inferSearchableStrings(value: unknown) {
59
+ if (typeof value === 'string') {
60
+ return [searchableString(value)];
61
+ }
62
+
63
+ if (typeof value === 'object' && value != null) {
64
+ return Object.values(value)
65
+ .filter((innerValue) => typeof innerValue === 'string')
66
+ .map((innerValue) => searchableString(innerValue));
67
+ }
68
+
69
+ return [];
70
+ }
71
+
72
+ export interface SelectInputOptionItem<T = string> {
73
+ type: 'option';
74
+ value: T;
75
+ filterMatchers?: readonly string[];
76
+ disabled?: boolean;
77
+ }
78
+
79
+ export interface SelectInputGroupItem<T = string> {
80
+ type: 'group';
81
+ label: ReactNode;
82
+ options: readonly SelectInputOptionItem<T>[];
83
+ action?: {
84
+ label: string;
85
+ onClick: ButtonProps['onClick'];
86
+ };
87
+ }
88
+
89
+ export interface SelectInputSeparatorItem {
90
+ type: 'separator';
91
+ }
92
+
93
+ export type SelectInputItem<T = string> =
94
+ | SelectInputOptionItem<T>
95
+ | SelectInputGroupItem<T>
96
+ | SelectInputSeparatorItem;
97
+
98
+ function dedupeSelectInputOptionItem<T>(
99
+ item: SelectInputOptionItem<T>,
100
+ existingValues: Set<T>,
101
+ compareValues?: (a: T, b: T) => boolean,
102
+ ): SelectInputOptionItem<T | undefined> {
103
+ const isDuplicate = compareValues
104
+ ? Array.from(existingValues).some((existingValue) => compareValues(item.value, existingValue))
105
+ : existingValues.has(item.value);
106
+
107
+ if (!isDuplicate) {
108
+ existingValues.add(item.value);
109
+ return item;
110
+ }
111
+ return { ...item, value: undefined };
112
+ }
113
+
114
+ /**
115
+ * Sets the `value` of duplicate option items to `undefined`, hiding them when
116
+ * rendered. Indexes are kept intact within groups to preserve the active item
117
+ * between filter changes when possible.
118
+ */
119
+ function dedupeSelectInputItems<T>(
120
+ items: readonly SelectInputItem<T>[],
121
+ compareValues?: (a: T, b: T) => boolean,
122
+ ): SelectInputItem<T | undefined>[] {
123
+ const existingValues = new Set<T>();
124
+
125
+ return items.map((item) => {
126
+ switch (item.type) {
127
+ case 'option': {
128
+ return dedupeSelectInputOptionItem(item, existingValues, compareValues);
129
+ }
130
+ case 'group': {
131
+ return {
132
+ ...item,
133
+ options: item.options.map((option) =>
134
+ dedupeSelectInputOptionItem(option, existingValues, compareValues),
135
+ ),
136
+ };
137
+ }
138
+ default:
139
+ }
140
+ return item;
141
+ });
142
+ }
143
+
144
+ function selectInputOptionItemIncludesNeedle<T>(item: SelectInputOptionItem<T>, needle: string) {
145
+ return inferSearchableStrings(item.filterMatchers ?? item.value).some((haystack) =>
146
+ haystack.includes(needle),
147
+ );
148
+ }
149
+
150
+ function filterSelectInputItems<T>(
151
+ items: readonly SelectInputItem<T>[],
152
+ predicate: (item: SelectInputOptionItem<T>) => boolean,
153
+ ) {
154
+ return items.filter((item) => {
155
+ switch (item.type) {
156
+ case 'option': {
157
+ return predicate(item);
158
+ }
159
+ case 'group': {
160
+ return item.options.some((option) => predicate(option));
161
+ }
162
+ default:
163
+ }
164
+ return false;
165
+ });
166
+ }
167
+
168
+ /**
169
+ * Flattens and sorts filtered options using the provided comparator.
170
+ * Extracts all options from groups, filters out undefined values (deduplicated items),
171
+ * sorts them, and returns as a flat list of option items.
172
+ */
173
+ function sortSelectInputItems<T>(
174
+ items: readonly SelectInputItem<T | undefined>[],
175
+ compareFn: (
176
+ a: SelectInputOptionItem<NonNullable<T>>,
177
+ b: SelectInputOptionItem<NonNullable<T>>,
178
+ searchQuery: string,
179
+ ) => number,
180
+ searchQuery: string,
181
+ ): SelectInputItem<NonNullable<T>>[] {
182
+ const flattenedOption = items.flatMap((item) => {
183
+ if (item.type === 'option') {
184
+ return item.value !== undefined ? [item as SelectInputOptionItem<NonNullable<T>>] : [];
185
+ }
186
+
187
+ if (item.type === 'group') {
188
+ return item.options.filter(
189
+ (option): option is SelectInputOptionItem<NonNullable<T>> => option.value !== undefined,
190
+ );
191
+ }
192
+
193
+ return [];
194
+ });
195
+
196
+ // eslint-disable-next-line functional/immutable-data
197
+ return flattenedOption.sort((a, b) => compareFn(a, b, searchQuery));
198
+ }
199
+
200
+ export interface SelectInputProps<T = string, M extends boolean = false> {
201
+ id?: string;
202
+ /**
203
+ * Sets the `data-wds-parent` attribute on the listbox container, which is needed for complex components like DateInput to correctly manage event handling.
204
+ * @internal
205
+ */
206
+ parentId?: string;
207
+ name?: string;
208
+ multiple?: M;
209
+ placeholder?: string;
210
+ items: readonly SelectInputItem<NonNullable<T>>[];
211
+ /**
212
+ * Enables browser autocomplete integration through the search input.
213
+ * Accepts standard HTML autocomplete values (e.g., "country-name", "address-level1").
214
+ *
215
+ * Requires `filterable={true}` to enable the search input.
216
+ *
217
+ * @example
218
+ * <SelectInput
219
+ * name="country"
220
+ * autocomplete="country-name"
221
+ * filterable={true}
222
+ * items={[{
223
+ * type: 'option',
224
+ * value: 'GB',
225
+ * filterMatchers: ['United Kingdom', 'UK']
226
+ * }]}
227
+ * />
228
+ */
229
+ autocomplete?: string;
230
+ defaultValue?: M extends true ? readonly T[] : T;
231
+ value?: M extends true ? readonly T[] : T;
232
+ compareValues?:
233
+ | (keyof NonNullable<T> & string)
234
+ | ((a: T | undefined, b: T | undefined) => boolean);
235
+ renderValue?: (value: NonNullable<T>, withinTrigger: boolean) => React.ReactNode;
236
+ renderFooter?: (args: {
237
+ resultsEmpty: boolean;
238
+ queryNormalized: string | null | undefined;
239
+ }) => React.ReactNode;
240
+ renderTrigger?: (args: {
241
+ content: React.ReactNode;
242
+ placeholderShown: boolean;
243
+ clear: (() => void) | undefined;
244
+ disabled: boolean;
245
+ size: 'sm' | 'md' | 'lg';
246
+ className: string | undefined;
247
+ }) => React.ReactNode;
248
+ filterable?: boolean;
249
+ filterPlaceholder?: string;
250
+ sortFilteredOptions?: (
251
+ a: SelectInputOptionItem<NonNullable<T>>,
252
+ b: SelectInputOptionItem<NonNullable<T>>,
253
+ searchQuery: string,
254
+ ) => number;
255
+ disabled?: boolean;
256
+ size?: 'sm' | 'md' | 'lg';
257
+ className?: string;
258
+ UNSAFE_triggerButtonProps?: WithInputAttributesProps['inputAttributes'] & {
259
+ 'aria-label'?: string;
260
+ };
261
+ /** Ref to the select trigger button element. */
262
+ triggerRef?: React.MutableRefObject<HTMLButtonElement | null>;
263
+ onFilterChange?: (args: { query: string; queryNormalized: string | null }) => void;
264
+ onChange?: (value: M extends true ? T[] : T) => void;
265
+ onOpen?: () => void;
266
+ onClose?: () => void;
267
+ onClear?: () => void;
268
+ }
269
+
270
+ const defaultRenderTrigger = (({ content, placeholderShown, clear, disabled, size, className }) => (
271
+ <InputGroup
272
+ addonEnd={{
273
+ content: (
274
+ <span className={clsx('np-select-input-addon-container', disabled && 'disabled')}>
275
+ {clear != null && !placeholderShown ? (
276
+ <>
277
+ <SelectInputClearButton
278
+ onClick={(event) => {
279
+ event.preventDefault();
280
+ clear();
281
+ }}
282
+ />
283
+ <span className="np-select-input-addon-separator" />
284
+ </>
285
+ ) : null}
286
+
287
+ <span className="np-select-input-addon">
288
+ <ChevronDown size={16} />
289
+ </span>
290
+ </span>
291
+ ),
292
+ initialContentWidth: 24 + 4,
293
+ padding: 'sm',
294
+ }}
295
+ disabled={disabled}
296
+ className={className}
297
+ >
298
+ <SelectInputTriggerButton as={ButtonInput} size={size}>
299
+ <span
300
+ className={clsx(
301
+ 'np-select-input-content',
302
+ placeholderShown && 'np-select-input-placeholder',
303
+ )}
304
+ >
305
+ {content}
306
+ </span>
307
+ </SelectInputTriggerButton>
308
+ </InputGroup>
309
+ )) satisfies SelectInputProps['renderTrigger'];
310
+
311
+ interface SelectInputClearButtonProps extends Pick<
312
+ React.ComponentPropsWithoutRef<'button'>,
313
+ 'className' | 'onClick'
314
+ > {}
315
+
316
+ function SelectInputClearButton({ className, onClick }: SelectInputClearButtonProps) {
317
+ const intl = useIntl();
318
+
319
+ return (
320
+ <button
321
+ type="button"
322
+ aria-label={intl.formatMessage(dateTriggerMessages.ariaLabel)}
323
+ className={clsx(className, 'np-select-input-addon np-select-input-addon--interactive')}
324
+ onClick={onClick}
325
+ >
326
+ <Cross size={16} />
327
+ </button>
328
+ );
329
+ }
330
+
331
+ const noop = () => {};
332
+
333
+ export function SelectInput<T = string, M extends boolean = false>({
334
+ id: idProp,
335
+ parentId,
336
+ name,
337
+ multiple,
338
+ placeholder,
339
+ autocomplete,
340
+ items,
341
+ defaultValue,
342
+ value: controlledValue,
343
+ compareValues,
344
+ renderValue = String,
345
+ renderFooter,
346
+ renderTrigger = defaultRenderTrigger,
347
+ filterable,
348
+ filterPlaceholder,
349
+ sortFilteredOptions,
350
+ disabled,
351
+ size = 'md',
352
+ className,
353
+ UNSAFE_triggerButtonProps,
354
+ triggerRef: externalTriggerRef,
355
+ onFilterChange = noop,
356
+ onChange,
357
+ onOpen,
358
+ onClose,
359
+ onClear,
360
+ }: SelectInputProps<T, M>) {
361
+ const inputAttributes = useInputAttributes({ nonLabelable: true });
362
+ const id = idProp ?? inputAttributes.id;
363
+
364
+ const [open, setOpen] = useState(false);
365
+
366
+ const initialized = useRef(false);
367
+ const handleClose = useEffectEvent(onClose ?? (() => {}));
368
+ const handleOpen = useEffectEvent(onOpen ?? (() => {}));
369
+ useEffect(() => {
370
+ if (initialized.current) {
371
+ if (open) {
372
+ handleOpen?.();
373
+ } else {
374
+ handleClose?.();
375
+ }
376
+ } else {
377
+ initialized.current = true;
378
+ }
379
+ }, [handleClose, handleOpen, open]);
380
+
381
+ const [filterQuery, _setFilterQuery] = useState('');
382
+ const deferredFilterQuery = useDeferredValue(filterQuery);
383
+ const setFilterQuery = useEffectEvent((query: string) => {
384
+ _setFilterQuery(query);
385
+ if (query !== filterQuery) {
386
+ onFilterChange({
387
+ query,
388
+ queryNormalized: query ? searchableString(query) : null,
389
+ });
390
+ }
391
+ });
392
+
393
+ const internalTriggerRef = useRef<HTMLButtonElement | null>(null);
394
+
395
+ const screenSm = useScreenSize(Breakpoint.SMALL);
396
+ const OptionsOverlay = screenSm ? Popover : BottomSheet;
397
+
398
+ const searchInputRef = useRef<HTMLInputElement>(null);
399
+ const listboxRef = useRef<HTMLDivElement>(null);
400
+ const controllerRef = filterable ? searchInputRef : listboxRef;
401
+
402
+ /**
403
+ * Attempts to resolve the `listbox` label
404
+ * @see https://storybook.wise.design/?path=/docs/forms-selectinput-accessibility--docs#labelling
405
+ */
406
+ const getListBoxLabelProps = (): {
407
+ listBoxLabel?: string;
408
+ listBoxLabelledBy?: string;
409
+ } => {
410
+ if (UNSAFE_triggerButtonProps?.['aria-label']) {
411
+ return {
412
+ listBoxLabel: UNSAFE_triggerButtonProps['aria-label'],
413
+ };
414
+ }
415
+
416
+ if (UNSAFE_triggerButtonProps?.['aria-labelledby']) {
417
+ return {
418
+ listBoxLabelledBy: UNSAFE_triggerButtonProps['aria-labelledby'],
419
+ };
420
+ }
421
+
422
+ if (inputAttributes['aria-labelledby']) {
423
+ return {
424
+ listBoxLabelledBy: inputAttributes['aria-labelledby'],
425
+ };
426
+ }
427
+
428
+ return {};
429
+ };
430
+
431
+ return (
432
+ <ListboxBase
433
+ name={name}
434
+ multiple={multiple}
435
+ defaultValue={defaultValue as M extends true ? T[] : T}
436
+ value={controlledValue as M extends true ? T[] : T}
437
+ by={compareValues}
438
+ disabled={disabled}
439
+ onChange={
440
+ ((value) => {
441
+ if (!multiple) {
442
+ setOpen(false);
443
+ }
444
+ onChange?.(value);
445
+ }) satisfies SelectInputProps<T, M>['onChange']
446
+ }
447
+ >
448
+ {({ disabled: uiDisabled, value }) => {
449
+ const placeholderShown =
450
+ multiple && Array.isArray(value) ? value.length === 0 : value == null;
451
+ return (
452
+ <OptionsOverlay
453
+ placement="bottom-start"
454
+ open={open}
455
+ renderTrigger={({ ref, getInteractionProps }) => (
456
+ <SelectInputTriggerButtonPropsContext.Provider
457
+ // eslint-disable-next-line react/jsx-no-constructed-context-values
458
+ value={{
459
+ ref: (node) => {
460
+ ref(node);
461
+ if (externalTriggerRef) {
462
+ // eslint-disable-next-line no-param-reassign
463
+ externalTriggerRef.current = node;
464
+ } else {
465
+ internalTriggerRef.current = node;
466
+ }
467
+ },
468
+ ...inputAttributes,
469
+ ...UNSAFE_triggerButtonProps,
470
+ id,
471
+ ...mergeProps(
472
+ {
473
+ onClick: () => {
474
+ setOpen((prev) => !prev);
475
+ },
476
+ onKeyDown: (event: React.KeyboardEvent) => {
477
+ if (
478
+ event.key === ' ' ||
479
+ event.key === 'Enter' ||
480
+ event.key === 'ArrowDown' ||
481
+ event.key === 'ArrowUp'
482
+ ) {
483
+ setOpen((prev) => !prev);
484
+ }
485
+ },
486
+ },
487
+ getInteractionProps(),
488
+ ),
489
+ }}
490
+ >
491
+ {renderTrigger({
492
+ content: !placeholderShown ? (
493
+ <SelectInputOptionContentWithinTriggerContext.Provider value>
494
+ {multiple && Array.isArray(value)
495
+ ? (value as readonly NonNullable<T>[])
496
+ .map((option) => renderValue(option, true))
497
+ .filter((node) => node != null)
498
+ .join(', ')
499
+ : renderValue(value as NonNullable<T>, true)}
500
+ </SelectInputOptionContentWithinTriggerContext.Provider>
501
+ ) : (
502
+ placeholder
503
+ ),
504
+ placeholderShown,
505
+ clear:
506
+ onClear != null
507
+ ? () => {
508
+ onClear();
509
+ (externalTriggerRef?.current ?? internalTriggerRef.current)?.focus({
510
+ preventScroll: true,
511
+ });
512
+ }
513
+ : undefined,
514
+ disabled: uiDisabled,
515
+ size,
516
+ className,
517
+ })}
518
+ </SelectInputTriggerButtonPropsContext.Provider>
519
+ )}
520
+ initialFocusRef={controllerRef}
521
+ size={filterable ? 'lg' : 'md'}
522
+ padding="none"
523
+ onClose={() => {
524
+ setOpen(false);
525
+ }}
526
+ onCloseEnd={() => {
527
+ setFilterQuery('');
528
+ }}
529
+ >
530
+ <SelectInputOptions
531
+ id={id ? `${id}Search` : undefined}
532
+ parentId={parentId}
533
+ items={items}
534
+ compareValues={compareValues}
535
+ renderValue={renderValue}
536
+ renderFooter={renderFooter}
537
+ filterable={filterable}
538
+ filterPlaceholder={filterPlaceholder}
539
+ sortFilteredOptions={sortFilteredOptions}
540
+ searchInputRef={searchInputRef}
541
+ listboxRef={listboxRef}
542
+ filterQuery={deferredFilterQuery}
543
+ autocomplete={autocomplete}
544
+ name={name}
545
+ onFilterChange={setFilterQuery}
546
+ onAutocompleteSelect={(matchedValue) => {
547
+ onChange?.(matchedValue as M extends true ? T[] : T);
548
+ if (!multiple) {
549
+ setOpen(false);
550
+ }
551
+ }}
552
+ {...getListBoxLabelProps()}
553
+ />
554
+ </OptionsOverlay>
555
+ );
556
+ }}
557
+ </ListboxBase>
558
+ );
559
+ }
560
+
561
+ const SelectInputTriggerButtonPropsContext = createContext<{
562
+ ref?: React.ForwardedRef<HTMLButtonElement | null>;
563
+ id?: string;
564
+ onClick?: (event: React.MouseEvent) => void;
565
+ onKeyDown?: (event: React.KeyboardEvent) => void;
566
+ [key: string]: unknown;
567
+ }>({});
568
+
569
+ type SelectInputTriggerButtonElementType = 'button' | React.ComponentType;
570
+
571
+ export type SelectInputTriggerButtonProps<
572
+ T extends SelectInputTriggerButtonElementType = 'button',
573
+ > = Merge<React.ComponentPropsWithoutRef<T>, { as?: T }>;
574
+
575
+ export function SelectInputTriggerButton<T extends SelectInputTriggerButtonElementType = 'button'>({
576
+ as = 'button' as T,
577
+ ...restProps
578
+ }: SelectInputTriggerButtonProps<T>) {
579
+ const { ref, onClick, onKeyDown, ...interactionProps } = useContext(
580
+ SelectInputTriggerButtonPropsContext,
581
+ );
582
+
583
+ return (
584
+ <ListboxButton
585
+ ref={ref}
586
+ as={PolymorphicWithOverrides}
587
+ role="combobox"
588
+ __overrides={{ as, ...interactionProps }}
589
+ {...mergeProps({ onClick, onKeyDown }, restProps)}
590
+ />
591
+ );
592
+ }
593
+
594
+ interface SelectInputOptionsContainerProps extends React.ComponentPropsWithRef<'div'> {
595
+ onAriaActiveDescendantChange: (value: React.AriaAttributes['aria-activedescendant']) => void;
596
+ }
597
+
598
+ const SelectInputOptionsContainer = forwardRef(function SelectInputOptionsContainer(
599
+ {
600
+ 'aria-orientation': ariaOrientation,
601
+ 'aria-activedescendant': ariaActiveDescendant,
602
+ role,
603
+ tabIndex,
604
+ onAriaActiveDescendantChange,
605
+ onKeyDown,
606
+ ...restProps
607
+ }: SelectInputOptionsContainerProps,
608
+ ref: React.ForwardedRef<HTMLDivElement | null>,
609
+ ) {
610
+ const handleAriaActiveDescendantChange = useEffectEvent(onAriaActiveDescendantChange);
611
+ useEffect(() => {
612
+ handleAriaActiveDescendantChange(ariaActiveDescendant);
613
+ }, [ariaActiveDescendant, handleAriaActiveDescendantChange]);
614
+
615
+ return (
616
+ <div
617
+ ref={ref}
618
+ role="none"
619
+ onKeyDown={(event) => {
620
+ // Prevent confirmation close without an active item
621
+ if (event.key === 'Enter' && ariaActiveDescendant == null) {
622
+ return;
623
+ }
624
+
625
+ // Required to make ListBox focusable
626
+ if (event.key === 'Tab') {
627
+ return;
628
+ }
629
+
630
+ // Prevent absorbing Escape early
631
+ if (event.key === 'Escape') {
632
+ onKeyDown?.({
633
+ ...event,
634
+ preventDefault: () => {},
635
+ stopPropagation: () => {},
636
+ });
637
+ return;
638
+ }
639
+
640
+ onKeyDown?.(event);
641
+ }}
642
+ {...restProps}
643
+ />
644
+ );
645
+ });
646
+
647
+ interface SelectInputOptionsProps<T = string> extends Pick<
648
+ SelectInputProps<T>,
649
+ | 'items'
650
+ | 'renderValue'
651
+ | 'renderFooter'
652
+ | 'filterable'
653
+ | 'filterPlaceholder'
654
+ | 'id'
655
+ | 'parentId'
656
+ | 'compareValues'
657
+ | 'sortFilteredOptions'
658
+ > {
659
+ searchInputRef: React.MutableRefObject<HTMLInputElement | null>;
660
+ listboxRef: React.MutableRefObject<HTMLDivElement | null>;
661
+ filterQuery: string;
662
+ onFilterChange: (query: string) => void;
663
+ listBoxLabel?: string;
664
+ listBoxLabelledBy?: string;
665
+ autocomplete?: string;
666
+ name?: string;
667
+ onAutocompleteSelect?: (value: T) => void;
668
+ }
669
+
670
+ function SelectInputOptions<T = string>({
671
+ id,
672
+ parentId,
673
+ items,
674
+ compareValues: compareValuesProp,
675
+ renderValue = String,
676
+ renderFooter,
677
+ filterable = false,
678
+ filterPlaceholder,
679
+ sortFilteredOptions,
680
+ searchInputRef,
681
+ listboxRef,
682
+ filterQuery,
683
+ onFilterChange,
684
+ listBoxLabel,
685
+ listBoxLabelledBy,
686
+ autocomplete,
687
+ name,
688
+ onAutocompleteSelect,
689
+ }: SelectInputOptionsProps<T>) {
690
+ const intl = useIntl();
691
+ const virtualiserHandlerRef = useRef<VirtualizerHandle>(null);
692
+ const controllerRef = filterable ? searchInputRef : listboxRef;
693
+ const [initialRender, setInitialRender] = useState(true);
694
+
695
+ const needle = useMemo(() => {
696
+ if (filterable) {
697
+ return filterQuery ? searchableString(filterQuery) : null;
698
+ }
699
+ return undefined;
700
+ }, [filterQuery, filterable]);
701
+ useEffect(() => {
702
+ if (needle) {
703
+ // Ensure having an active option while filtering.
704
+ // Without `requestAnimationFrame` upon which React depends for scheduling
705
+ // updates, the active status would only show for a split second and then
706
+ // disappear inadvertently.
707
+ requestAnimationFrame(() => {
708
+ if (
709
+ controllerRef.current != null &&
710
+ !controllerRef.current.hasAttribute('aria-activedescendant')
711
+ ) {
712
+ // Activate first option via synthetic key press
713
+ controllerRef.current.dispatchEvent(
714
+ new KeyboardEvent('keydown', { key: 'Home', bubbles: true }),
715
+ );
716
+ }
717
+ });
718
+ }
719
+ }, [controllerRef, needle]);
720
+
721
+ const compareValues = useMemo(() => {
722
+ if (!compareValuesProp) {
723
+ return undefined;
724
+ }
725
+
726
+ if (typeof compareValuesProp === 'function') {
727
+ return (a: NonNullable<T>, b: NonNullable<T>) => compareValuesProp(a, b);
728
+ }
729
+
730
+ const key = compareValuesProp;
731
+ return (a: NonNullable<T>, b: NonNullable<T>) => {
732
+ if (typeof a === 'object' && a != null && typeof b === 'object' && b != null) {
733
+ return (a as Record<string, unknown>)[key] === (b as Record<string, unknown>)[key];
734
+ }
735
+ return a === b;
736
+ };
737
+ }, [compareValuesProp]);
738
+
739
+ const filteredItems: readonly SelectInputItem<NonNullable<T> | undefined>[] = useMemo(() => {
740
+ if (needle == null) {
741
+ return items;
742
+ }
743
+
744
+ const filtered = filterSelectInputItems(dedupeSelectInputItems(items, compareValues), (item) =>
745
+ selectInputOptionItemIncludesNeedle(item, needle),
746
+ );
747
+
748
+ if (sortFilteredOptions) {
749
+ return sortSelectInputItems(filtered, sortFilteredOptions, filterQuery);
750
+ }
751
+
752
+ return filtered;
753
+ // eslint-disable-next-line react-hooks/exhaustive-deps
754
+ }, [needle, items, compareValues]);
755
+ const resultsEmpty = needle != null && filteredItems.length === 0;
756
+
757
+ const virtualized = filteredItems.length > MAX_ITEMS_WITHOUT_VIRTUALIZATION;
758
+
759
+ // Items shown once shall be kept mounted until the needle changes, otherwise
760
+ // the scroll position may jump around inadvertently. Pattern adopted from:
761
+ // https://inokawa.github.io/virtua/?path=/story/advanced-keep-offscreen-items--append-only
762
+ const [mountedIndexes, setMountedIndexes] = useState<number[]>([]);
763
+ useEffect(() => {
764
+ // Ensure the 'End' key works as intended by keeping the last item mounted
765
+ setMountedIndexes((prevMountedIndexes) => {
766
+ const indexes = new Set(prevMountedIndexes);
767
+ indexes.add(filteredItems.length - 1);
768
+ return [...indexes]; // Sorting is redundant by nature here
769
+ });
770
+ }, [
771
+ needle, // Needed as `filteredItems.length` may be equal between two updates
772
+ filteredItems.length,
773
+ ]);
774
+
775
+ const listboxContainerRef = useRef<HTMLDivElement>(null);
776
+ useEffect(() => {
777
+ if (listboxContainerRef.current != null) {
778
+ listboxContainerRef.current.style.setProperty(
779
+ '--initial-height',
780
+ `${listboxContainerRef.current.offsetHeight}px`,
781
+ );
782
+ }
783
+ }, []);
784
+
785
+ useEffect(() => {
786
+ setInitialRender(false);
787
+ }, []);
788
+
789
+ const showStatus = resultsEmpty;
790
+ const statusId = useId();
791
+ const listboxId = useId();
792
+
793
+ const getItemNode = (index: number) => {
794
+ const item = filteredItems[index];
795
+ return (
796
+ <SelectInputItemView key={index} item={item} renderValue={renderValue} needle={needle} />
797
+ );
798
+ };
799
+
800
+ const findMatchingItem = (autocompleteValue: string): T | null => {
801
+ const flatOptions = items
802
+ .flatMap((item) =>
803
+ item.type === 'group' ? item.options : item.type === 'option' ? [item] : [],
804
+ )
805
+ .filter(
806
+ (item): item is SelectInputOptionItem<NonNullable<T>> =>
807
+ item.type === 'option' && item.value != null,
808
+ );
809
+
810
+ const exactMatch = flatOptions.find(
811
+ (option) =>
812
+ String(option.value) === autocompleteValue ||
813
+ option.filterMatchers?.some((matcher) => matcher === autocompleteValue),
814
+ );
815
+
816
+ if (exactMatch) {
817
+ return exactMatch.value;
818
+ }
819
+
820
+ const fuzzyMatch = flatOptions.find((option) =>
821
+ option.filterMatchers?.some((matcher) =>
822
+ matcher.toLowerCase().includes(autocompleteValue.toLowerCase()),
823
+ ),
824
+ );
825
+
826
+ return fuzzyMatch ? fuzzyMatch.value : null;
827
+ };
828
+
829
+ return (
830
+ <ListboxOptions
831
+ modal
832
+ as={SelectInputOptionsContainer}
833
+ static
834
+ className="np-select-input-options-container"
835
+ onAriaActiveDescendantChange={(value: React.AriaAttributes['aria-activedescendant']) => {
836
+ if (controllerRef.current != null) {
837
+ if (!initialRender && value != null) {
838
+ controllerRef.current.setAttribute('aria-activedescendant', value);
839
+ } else {
840
+ controllerRef.current.removeAttribute('aria-activedescendant');
841
+ }
842
+ }
843
+ }}
844
+ >
845
+ {filterable ? (
846
+ <div className="np-select-input-query-container">
847
+ <SearchInput
848
+ ref={searchInputRef}
849
+ id={id}
850
+ name={name}
851
+ autoComplete={autocomplete}
852
+ role="combobox"
853
+ shape="rectangle"
854
+ placeholder={filterPlaceholder}
855
+ aria-label={filterPlaceholder}
856
+ defaultValue={filterQuery}
857
+ aria-autocomplete="list"
858
+ aria-expanded
859
+ aria-controls={listboxId}
860
+ aria-describedby={showStatus ? statusId : undefined}
861
+ onKeyDown={(event) => {
862
+ // Prevent interfering with the matcher of Headless UI
863
+ // https://mathiasbynens.be/notes/javascript-unicode#regex
864
+ if (/^.$/u.test(event.key)) {
865
+ event.stopPropagation();
866
+ }
867
+ }}
868
+ onChange={(event) => {
869
+ // Free up resources and ensure not to go out of bounds when the
870
+ // resulting item count is less than before
871
+ const inputValue = event.currentTarget.value;
872
+
873
+ // Free up resources and ensure not to go out of bounds
874
+ setMountedIndexes([]);
875
+ onFilterChange(inputValue);
876
+ }}
877
+ onInput={(event) => {
878
+ const inputValue = event.currentTarget.value;
879
+ const inputElement = event.currentTarget;
880
+
881
+ if (autocomplete && onAutocompleteSelect && inputValue) {
882
+ setTimeout(() => {
883
+ if (inputElement.value === inputValue && inputValue.length > 2) {
884
+ const matchedValue = findMatchingItem(inputValue);
885
+ if (matchedValue !== null) {
886
+ onAutocompleteSelect(matchedValue);
887
+ }
888
+ }
889
+ }, 50);
890
+ }
891
+ }}
892
+ />
893
+ </div>
894
+ ) : null}
895
+
896
+ <section
897
+ ref={listboxContainerRef}
898
+ tabIndex={-1}
899
+ className={clsx(
900
+ 'np-select-input-listbox-container',
901
+ virtualized && 'np-select-input-listbox-container--virtualized',
902
+ needle == null && // Groups aren't shown when filtering
903
+ items.some((item) => item.type === 'group') &&
904
+ 'np-select-input-listbox-container--has-group',
905
+ )}
906
+ data-wds-parent={parentId ?? undefined}
907
+ >
908
+ {resultsEmpty ? (
909
+ <div id={statusId} className="np-select-input-options-status">
910
+ <CrossCircle size={16} className="np-select-input-options-status-icon" />
911
+ {intl.formatMessage(messages.noResultsFound)}
912
+ </div>
913
+ ) : null}
914
+
915
+ <div
916
+ ref={listboxRef}
917
+ id={listboxId}
918
+ role="listbox"
919
+ aria-orientation="vertical"
920
+ aria-label={listBoxLabel}
921
+ aria-labelledby={listBoxLabelledBy}
922
+ tabIndex={0}
923
+ className="np-select-input-listbox"
924
+ >
925
+ {!virtualized ? (
926
+ filteredItems.map((_, index) => getItemNode(index))
927
+ ) : (
928
+ <Virtualizer
929
+ ref={virtualiserHandlerRef}
930
+ key={needle}
931
+ data={filteredItems}
932
+ keepMounted={mountedIndexes}
933
+ scrollRef={listboxRef} // `VList` doesn't expose this
934
+ onScroll={async () => {
935
+ if (!virtualiserHandlerRef.current) return;
936
+
937
+ const startIndex = virtualiserHandlerRef.current.findItemIndex(
938
+ virtualiserHandlerRef.current.scrollOffset,
939
+ );
940
+ const endIndex = virtualiserHandlerRef.current.findItemIndex(
941
+ virtualiserHandlerRef.current.scrollOffset +
942
+ virtualiserHandlerRef.current.viewportSize,
943
+ );
944
+
945
+ setMountedIndexes((prevMountedIndexes) => {
946
+ const indexes = new Set(prevMountedIndexes);
947
+
948
+ for (let index = startIndex; index <= endIndex; index += 1) {
949
+ indexes.add(index);
950
+ }
951
+
952
+ return [...indexes].sort((a, b) => a - b);
953
+ });
954
+ }}
955
+ >
956
+ {(item, index) => (
957
+ // The position of each item can't be inferred by browsers when
958
+ // virtualizing, as some of the items may not be in the DOM
959
+ <SelectInputItemsCountContext.Provider value={filteredItems.length}>
960
+ <SelectInputItemPositionContext.Provider value={index + 1}>
961
+ {getItemNode(index)}
962
+ </SelectInputItemPositionContext.Provider>
963
+ </SelectInputItemsCountContext.Provider>
964
+ )}
965
+ </Virtualizer>
966
+ )}
967
+ </div>
968
+
969
+ {renderFooter != null ? (
970
+ <footer className="np-select-input-footer">
971
+ <div
972
+ role="none"
973
+ onKeyDown={(event) => {
974
+ // Prevent interfering with Headless UI
975
+ if (event.key !== 'Escape') {
976
+ event.stopPropagation();
977
+ }
978
+ }}
979
+ >
980
+ {renderFooter({
981
+ resultsEmpty,
982
+ queryNormalized: needle,
983
+ })}
984
+ </div>
985
+ </footer>
986
+ ) : null}
987
+ </section>
988
+ </ListboxOptions>
989
+ );
990
+ }
991
+
992
+ interface SelectInputItemViewProps<
993
+ T = string,
994
+ I extends SelectInputItem<T | undefined> = SelectInputItem<T | undefined>,
995
+ > extends Required<Pick<SelectInputProps<T>, 'renderValue'>> {
996
+ item: I;
997
+ needle: string | null | undefined;
998
+ }
999
+
1000
+ function SelectInputItemView<T = string>({
1001
+ item,
1002
+ renderValue,
1003
+ needle,
1004
+ }: SelectInputItemViewProps<T>) {
1005
+ switch (item.type) {
1006
+ case 'option': {
1007
+ if (
1008
+ item.value != null &&
1009
+ (needle == null || selectInputOptionItemIncludesNeedle(item, needle))
1010
+ ) {
1011
+ return (
1012
+ <SelectInputOption value={item.value} disabled={item.disabled}>
1013
+ {renderValue(item.value, false)}
1014
+ </SelectInputOption>
1015
+ );
1016
+ }
1017
+ break;
1018
+ }
1019
+ case 'group': {
1020
+ return <SelectInputGroupItemView item={item} renderValue={renderValue} needle={needle} />;
1021
+ }
1022
+ case 'separator': {
1023
+ if (needle == null) {
1024
+ return <hr className="np-select-input-separator-item" />;
1025
+ }
1026
+ break;
1027
+ }
1028
+ }
1029
+ return null;
1030
+ }
1031
+
1032
+ interface SelectInputGroupItemViewProps<T = string> extends SelectInputItemViewProps<
1033
+ T,
1034
+ SelectInputGroupItem<T | undefined>
1035
+ > {}
1036
+
1037
+ function SelectInputGroupItemView<T = string>({
1038
+ item,
1039
+ renderValue,
1040
+ needle,
1041
+ }: SelectInputGroupItemViewProps<T>) {
1042
+ const headerId = useId();
1043
+
1044
+ const header = (
1045
+ <Header
1046
+ as="header"
1047
+ role="none"
1048
+ id={headerId}
1049
+ title={item.label}
1050
+ // @ts-expect-error when we migrate ActionButton to new Button this should be sorted
1051
+ action={
1052
+ item.action && {
1053
+ text: item.action.label,
1054
+ onClick: item.action.onClick,
1055
+ }
1056
+ }
1057
+ className="np-select-input-group-item-header p-x-1"
1058
+ />
1059
+ );
1060
+
1061
+ return (
1062
+ // An empty container may be rendered when no options match `needle`
1063
+ // However, pre-filtering would result in worse performance overall
1064
+ <Section
1065
+ as="section"
1066
+ role="group"
1067
+ aria-labelledby={headerId}
1068
+ className={clsx('m-y-0', needle === null && 'np-select-input-group-item--without-needle')}
1069
+ >
1070
+ {needle == null ? header : null}
1071
+ {item.options.map((option, index) => (
1072
+ <SelectInputItemView
1073
+ // eslint-disable-next-line react/no-array-index-key
1074
+ key={index}
1075
+ item={option}
1076
+ renderValue={renderValue}
1077
+ needle={needle}
1078
+ />
1079
+ ))}
1080
+ </Section>
1081
+ );
1082
+ }
1083
+
1084
+ const SelectInputItemsCountContext = createContext<number | undefined>(undefined);
1085
+ const SelectInputItemPositionContext = createContext<number | undefined>(undefined);
1086
+
1087
+ interface SelectInputOptionProps<T = string> {
1088
+ value: T;
1089
+ disabled?: boolean;
1090
+ children?: React.ReactNode;
1091
+ }
1092
+
1093
+ function SelectInputOption<T = string>({ value, disabled, children }: SelectInputOptionProps<T>) {
1094
+ const itemsCount = useContext(SelectInputItemsCountContext);
1095
+ const itemPosition = useContext(SelectInputItemPositionContext);
1096
+ return (
1097
+ <ListboxOption
1098
+ as="div"
1099
+ value={value}
1100
+ aria-setsize={itemsCount}
1101
+ aria-posinset={itemPosition}
1102
+ disabled={disabled}
1103
+ className={({ active, disabled: uiDisabled }) =>
1104
+ clsx(
1105
+ 'np-select-input-option-container np-text-body-large',
1106
+ active && 'np-select-input-option-container--active',
1107
+ uiDisabled && 'np-select-input-option-container--disabled',
1108
+ )
1109
+ }
1110
+ >
1111
+ {({ selected }) => (
1112
+ <>
1113
+ <div className="np-select-input-option">{children}</div>
1114
+ <Check
1115
+ size={16}
1116
+ className={clsx(
1117
+ 'np-select-input-option-check',
1118
+ !selected && 'np-select-input-option-check--not-selected',
1119
+ )}
1120
+ />
1121
+ </>
1122
+ )}
1123
+ </ListboxOption>
1124
+ );
1125
+ }
1126
+
1127
+ const SelectInputOptionContentWithinTriggerContext = createContext(false);
1128
+
1129
+ export interface SelectInputOptionContentProps {
1130
+ title: string;
1131
+ note?: string;
1132
+ description?: string;
1133
+ icon?: React.ReactNode;
1134
+ }
1135
+
1136
+ export function SelectInputOptionContent({
1137
+ title,
1138
+ note,
1139
+ description,
1140
+ icon,
1141
+ }: SelectInputOptionContentProps) {
1142
+ const withinTrigger = useContext(SelectInputOptionContentWithinTriggerContext);
1143
+
1144
+ return (
1145
+ <div
1146
+ className={clsx(
1147
+ 'np-select-input-option-content-container',
1148
+ (note || description) && 'np-text-body-large',
1149
+ )}
1150
+ >
1151
+ {icon ? (
1152
+ <div
1153
+ className={clsx(
1154
+ 'np-select-input-option-content-icon',
1155
+ !withinTrigger && 'np-select-input-option-content-icon--not-within-trigger',
1156
+ )}
1157
+ >
1158
+ {icon}
1159
+ </div>
1160
+ ) : null}
1161
+
1162
+ <div className="np-select-input-option-content-text">
1163
+ <div
1164
+ className={clsx(
1165
+ 'np-select-input-option-content-text-line-1',
1166
+ withinTrigger && 'np-select-input-option-content-text-within-trigger',
1167
+ )}
1168
+ >
1169
+ <div className="d-inline">{title}</div>
1170
+ {note ? (
1171
+ <span className="np-select-input-option-content-text-secondary np-text-body-default">
1172
+ {note}
1173
+ </span>
1174
+ ) : null}
1175
+ </div>
1176
+
1177
+ {description ? (
1178
+ <div
1179
+ className={clsx(
1180
+ 'np-select-input-option-content-text-secondary np-text-body-default',
1181
+ withinTrigger && 'np-select-input-option-content-text-within-trigger',
1182
+ )}
1183
+ >
1184
+ {description}
1185
+ </div>
1186
+ ) : null}
1187
+ </div>
1188
+ </div>
1189
+ );
1190
+ }