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