@transferwise/components 46.130.2 → 46.131.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (610) hide show
  1. package/build/dateInput/DateInput.js +12 -5
  2. package/build/dateInput/DateInput.js.map +1 -1
  3. package/build/dateInput/DateInput.mjs +11 -4
  4. package/build/dateInput/DateInput.mjs.map +1 -1
  5. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.js +16 -8
  6. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.js.map +1 -1
  7. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.mjs +14 -6
  8. package/build/expressiveMoneyInput/currencySelector/CurrencySelector.mjs.map +1 -1
  9. package/build/index.js +12 -7
  10. package/build/index.js.map +1 -1
  11. package/build/index.mjs +9 -3
  12. package/build/index.mjs.map +1 -1
  13. package/build/inputs/{_BottomSheet.js → SelectInput/BottomSheet/SelectInputBottomSheet.js} +7 -7
  14. package/build/inputs/SelectInput/BottomSheet/SelectInputBottomSheet.js.map +1 -0
  15. package/build/inputs/{_BottomSheet.mjs → SelectInput/BottomSheet/SelectInputBottomSheet.mjs} +7 -7
  16. package/build/inputs/SelectInput/BottomSheet/SelectInputBottomSheet.mjs.map +1 -0
  17. package/build/inputs/{_ButtonInput.js → SelectInput/ButtonInput/SelectInputButtonInput.js} +5 -5
  18. package/build/inputs/SelectInput/ButtonInput/SelectInputButtonInput.js.map +1 -0
  19. package/build/inputs/{_ButtonInput.mjs → SelectInput/ButtonInput/SelectInputButtonInput.mjs} +5 -5
  20. package/build/inputs/SelectInput/ButtonInput/SelectInputButtonInput.mjs.map +1 -0
  21. package/build/inputs/SelectInput/ClearButton/SelectInputClearButton.js +26 -0
  22. package/build/inputs/SelectInput/ClearButton/SelectInputClearButton.js.map +1 -0
  23. package/build/inputs/SelectInput/ClearButton/SelectInputClearButton.mjs +24 -0
  24. package/build/inputs/SelectInput/ClearButton/SelectInputClearButton.mjs.map +1 -0
  25. package/build/inputs/SelectInput/DefaultRenderTrigger/SelectInputDefaultRenderTrigger.js +59 -0
  26. package/build/inputs/SelectInput/DefaultRenderTrigger/SelectInputDefaultRenderTrigger.js.map +1 -0
  27. package/build/inputs/SelectInput/DefaultRenderTrigger/SelectInputDefaultRenderTrigger.mjs +56 -0
  28. package/build/inputs/SelectInput/DefaultRenderTrigger/SelectInputDefaultRenderTrigger.mjs.map +1 -0
  29. package/build/inputs/SelectInput/ItemView/GroupItemView/SelectInputGroupItemView.js +50 -0
  30. package/build/inputs/SelectInput/ItemView/GroupItemView/SelectInputGroupItemView.js.map +1 -0
  31. package/build/inputs/SelectInput/ItemView/GroupItemView/SelectInputGroupItemView.mjs +48 -0
  32. package/build/inputs/SelectInput/ItemView/GroupItemView/SelectInputGroupItemView.mjs.map +1 -0
  33. package/build/inputs/SelectInput/ItemView/SelectInputItemView.js +47 -0
  34. package/build/inputs/SelectInput/ItemView/SelectInputItemView.js.map +1 -0
  35. package/build/inputs/SelectInput/ItemView/SelectInputItemView.mjs +45 -0
  36. package/build/inputs/SelectInput/ItemView/SelectInputItemView.mjs.map +1 -0
  37. package/build/inputs/SelectInput/Option/SelectInputOption.js +42 -0
  38. package/build/inputs/SelectInput/Option/SelectInputOption.js.map +1 -0
  39. package/build/inputs/SelectInput/Option/SelectInputOption.mjs +40 -0
  40. package/build/inputs/SelectInput/Option/SelectInputOption.mjs.map +1 -0
  41. package/build/inputs/SelectInput/OptionContent/SelectInputOptionContent.js +40 -0
  42. package/build/inputs/SelectInput/OptionContent/SelectInputOptionContent.js.map +1 -0
  43. package/build/inputs/SelectInput/OptionContent/SelectInputOptionContent.mjs +38 -0
  44. package/build/inputs/SelectInput/OptionContent/SelectInputOptionContent.mjs.map +1 -0
  45. package/build/inputs/SelectInput/Options/OptionsContainer/SelectInputOptionsContainer.js +48 -0
  46. package/build/inputs/SelectInput/Options/OptionsContainer/SelectInputOptionsContainer.js.map +1 -0
  47. package/build/inputs/SelectInput/Options/OptionsContainer/SelectInputOptionsContainer.mjs +46 -0
  48. package/build/inputs/SelectInput/Options/OptionsContainer/SelectInputOptionsContainer.mjs.map +1 -0
  49. package/build/inputs/SelectInput/Options/SelectInputOptions.js +300 -0
  50. package/build/inputs/SelectInput/Options/SelectInputOptions.js.map +1 -0
  51. package/build/inputs/SelectInput/Options/SelectInputOptions.mjs +298 -0
  52. package/build/inputs/SelectInput/Options/SelectInputOptions.mjs.map +1 -0
  53. package/build/inputs/{_Popover.js → SelectInput/Popover/SelectInputPopover.js} +7 -7
  54. package/build/inputs/SelectInput/Popover/SelectInputPopover.js.map +1 -0
  55. package/build/inputs/{_Popover.mjs → SelectInput/Popover/SelectInputPopover.mjs} +7 -7
  56. package/build/inputs/SelectInput/Popover/SelectInputPopover.mjs.map +1 -0
  57. package/build/inputs/SelectInput/SelectInput.contexts.js +29 -0
  58. package/build/inputs/SelectInput/SelectInput.contexts.js.map +1 -0
  59. package/build/inputs/SelectInput/SelectInput.contexts.mjs +24 -0
  60. package/build/inputs/SelectInput/SelectInput.contexts.mjs.map +1 -0
  61. package/build/inputs/SelectInput/SelectInput.js +222 -0
  62. package/build/inputs/SelectInput/SelectInput.js.map +1 -0
  63. package/build/inputs/SelectInput/SelectInput.messages.js.map +1 -0
  64. package/build/inputs/SelectInput/SelectInput.messages.mjs.map +1 -0
  65. package/build/inputs/SelectInput/SelectInput.mjs +216 -0
  66. package/build/inputs/SelectInput/SelectInput.mjs.map +1 -0
  67. package/build/inputs/SelectInput/SelectInput.utils.js +164 -0
  68. package/build/inputs/SelectInput/SelectInput.utils.js.map +1 -0
  69. package/build/inputs/SelectInput/SelectInput.utils.mjs +154 -0
  70. package/build/inputs/SelectInput/SelectInput.utils.mjs.map +1 -0
  71. package/build/inputs/SelectInput/TriggerButton/SelectInputTriggerButton.js +42 -0
  72. package/build/inputs/SelectInput/TriggerButton/SelectInputTriggerButton.js.map +1 -0
  73. package/build/inputs/SelectInput/TriggerButton/SelectInputTriggerButton.mjs +36 -0
  74. package/build/inputs/SelectInput/TriggerButton/SelectInputTriggerButton.mjs.map +1 -0
  75. package/build/main.css +27094 -97
  76. package/build/moneyInput/MoneyInput.js +9 -2
  77. package/build/moneyInput/MoneyInput.js.map +1 -1
  78. package/build/moneyInput/MoneyInput.mjs +8 -1
  79. package/build/moneyInput/MoneyInput.mjs.map +1 -1
  80. package/build/phoneNumberInput/PhoneNumberInput.js +10 -3
  81. package/build/phoneNumberInput/PhoneNumberInput.js.map +1 -1
  82. package/build/phoneNumberInput/PhoneNumberInput.mjs +9 -2
  83. package/build/phoneNumberInput/PhoneNumberInput.mjs.map +1 -1
  84. package/build/styles/css/accordion.css +255 -0
  85. package/build/styles/css/alerts.css +751 -0
  86. package/build/styles/css/background.css +20 -0
  87. package/build/styles/css/badge.css +90 -0
  88. package/build/styles/css/border-radius.css +3 -0
  89. package/build/styles/css/breadcrumbs.css +20 -0
  90. package/build/styles/css/button-groups.css +495 -0
  91. package/build/styles/css/buttons.css +1390 -0
  92. package/build/styles/css/chevron.css +35 -0
  93. package/build/styles/css/circles.css +248 -0
  94. package/build/styles/css/close.css +47 -0
  95. package/build/styles/css/column-layout.css +164 -0
  96. package/build/styles/css/currency-flags.css +46 -0
  97. package/build/styles/css/decision.css +136 -0
  98. package/build/styles/css/dropdowns.css +472 -0
  99. package/build/styles/css/droppable.css +221 -0
  100. package/build/styles/css/flex.css +598 -0
  101. package/build/styles/css/footer.css +129 -0
  102. package/build/styles/css/grid.css +2369 -0
  103. package/build/styles/css/input-groups.css +2740 -0
  104. package/build/styles/css/link-callout.css +20 -0
  105. package/build/styles/css/list-group.css +350 -0
  106. package/build/styles/css/loaders.css +76 -0
  107. package/build/styles/css/media.css +74 -0
  108. package/build/styles/css/modals.css +185 -0
  109. package/build/styles/css/navbar.css +2406 -0
  110. package/build/styles/css/navs.css +296 -0
  111. package/build/styles/css/neptune-social-media.css +144 -0
  112. package/build/styles/css/neptune.css +25724 -0
  113. package/build/styles/css/panels.css +528 -0
  114. package/build/styles/css/popovers.css +661 -0
  115. package/build/styles/css/process.css +483 -0
  116. package/build/styles/css/progress-bars.css +64 -0
  117. package/build/styles/css/ring.css +55 -0
  118. package/build/styles/css/select.css +99 -0
  119. package/build/styles/css/sequences.css +855 -0
  120. package/build/styles/css/table.css +446 -0
  121. package/build/styles/css/tick.css +32 -0
  122. package/build/styles/css/tooltip.css +160 -0
  123. package/build/styles/css/utilities.css +145 -0
  124. package/build/styles/css/wells.css +74 -0
  125. package/build/styles/fonts/TW-Averta-Bold.woff +0 -0
  126. package/build/styles/fonts/TW-Averta-Bold.woff2 +0 -0
  127. package/build/styles/fonts/TW-Averta-Regular.woff +0 -0
  128. package/build/styles/fonts/TW-Averta-Regular.woff2 +0 -0
  129. package/build/styles/fonts/TW-Averta-Semibold.woff +0 -0
  130. package/build/styles/fonts/TW-Averta-Semibold.woff2 +0 -0
  131. package/build/styles/fonts/TransferWise-IconFont.svg +81 -0
  132. package/build/styles/fonts/TransferWise-IconFont.woff +0 -0
  133. package/build/styles/fonts/WiseSans-Heavy.woff2 +0 -0
  134. package/build/styles/fonts/inter-cyrillic-ext-variable-wghtOnly-normal.woff2 +0 -0
  135. package/build/styles/fonts/inter-cyrillic-variable-wghtOnly-normal.woff2 +0 -0
  136. package/build/styles/fonts/inter-greek-ext-variable-wghtOnly-normal.woff2 +0 -0
  137. package/build/styles/fonts/inter-greek-variable-wghtOnly-normal.woff2 +0 -0
  138. package/build/styles/fonts/inter-latin-ext-variable-wghtOnly-normal.woff2 +0 -0
  139. package/build/styles/fonts/inter-latin-variable-wghtOnly-normal.woff2 +0 -0
  140. package/build/styles/fonts/inter-vietnamese-variable-wghtOnly-normal.woff2 +0 -0
  141. package/build/styles/img/bg-dark.svg +31 -0
  142. package/build/styles/img/bg-light.svg +26 -0
  143. package/build/styles/inputs/SelectInput/BottomSheet/SelectInputBottomSheet.css +96 -0
  144. package/build/styles/inputs/SelectInput/ButtonInput/SelectInputButtonInput.css +16 -0
  145. package/build/styles/inputs/SelectInput/ClearButton/SelectInputClearButton.css +46 -0
  146. package/build/styles/inputs/SelectInput/ItemView/SelectInputItemView.css +16 -0
  147. package/build/styles/inputs/SelectInput/Option/SelectInputOption.css +33 -0
  148. package/build/styles/inputs/SelectInput/OptionContent/SelectInputOptionContent.css +37 -0
  149. package/build/styles/inputs/SelectInput/Options/SelectInputOptions.css +81 -0
  150. package/build/styles/inputs/SelectInput/Popover/SelectInputPopover.css +46 -0
  151. package/build/styles/less/legacy-variables.css +0 -0
  152. package/build/styles/less/legacy-variables.less +1299 -0
  153. package/build/styles/less/neptune-tokens.css +0 -0
  154. package/build/styles/less/neptune-tokens.less +275 -0
  155. package/build/styles/less/zindex.css +0 -0
  156. package/build/styles/less/zindex.less +17 -0
  157. package/build/styles/main.css +27094 -97
  158. package/build/styles/props/custom-media.css +11 -0
  159. package/build/styles/props/legacy-custom-props.css +69 -0
  160. package/build/styles/props/neptune-tokens.css +152 -0
  161. package/build/styles/styles/less/accordion.css +255 -0
  162. package/build/styles/styles/less/alerts.css +751 -0
  163. package/build/styles/styles/less/background.css +20 -0
  164. package/build/styles/styles/less/badge.css +90 -0
  165. package/build/styles/styles/less/border-radius.css +3 -0
  166. package/build/styles/styles/less/breadcrumbs.css +20 -0
  167. package/build/styles/styles/less/button-groups.css +495 -0
  168. package/build/styles/styles/less/buttons.css +1390 -0
  169. package/build/styles/styles/less/chevron.css +35 -0
  170. package/build/styles/styles/less/circles.css +248 -0
  171. package/build/styles/styles/less/close.css +47 -0
  172. package/build/styles/styles/less/column-layout.css +164 -0
  173. package/build/styles/styles/less/core/viewport-themes.css +59 -0
  174. package/build/styles/styles/less/currency-flags.css +46 -0
  175. package/build/styles/styles/less/decision.css +136 -0
  176. package/build/styles/styles/less/dropdowns.css +472 -0
  177. package/build/styles/styles/less/droppable.css +221 -0
  178. package/build/styles/styles/less/flex.css +598 -0
  179. package/build/styles/styles/less/footer.css +129 -0
  180. package/build/styles/styles/less/forms/bootstrap-forms.css +2277 -0
  181. package/build/styles/styles/less/forms/checkbox-radio.css +367 -0
  182. package/build/styles/styles/less/grid.css +2369 -0
  183. package/build/styles/styles/less/input-groups.css +2740 -0
  184. package/build/styles/styles/less/link-callout.css +20 -0
  185. package/build/styles/styles/less/list-group.css +350 -0
  186. package/build/styles/styles/less/loaders.css +76 -0
  187. package/build/styles/styles/less/media.css +74 -0
  188. package/build/styles/styles/less/modals.css +185 -0
  189. package/build/styles/styles/less/navbar.css +2406 -0
  190. package/build/styles/styles/less/navs.css +296 -0
  191. package/build/styles/styles/less/neptune-social-media.css +144 -0
  192. package/build/styles/styles/less/neptune.css +25724 -0
  193. package/build/styles/styles/less/panels.css +528 -0
  194. package/build/styles/styles/less/popovers.css +661 -0
  195. package/build/styles/styles/less/process.css +483 -0
  196. package/build/styles/styles/less/progress-bars.css +64 -0
  197. package/build/styles/styles/less/ring.css +55 -0
  198. package/build/styles/styles/less/select.css +99 -0
  199. package/build/styles/styles/less/sequences.css +855 -0
  200. package/build/styles/styles/less/table.css +446 -0
  201. package/build/styles/styles/less/tick.css +32 -0
  202. package/build/styles/styles/less/tooltip.css +160 -0
  203. package/build/styles/styles/less/utilities.css +145 -0
  204. package/build/styles/styles/less/wells.css +74 -0
  205. package/build/styles/styles/props/custom-media.css +11 -0
  206. package/build/styles/styles/props/legacy-custom-props.css +69 -0
  207. package/build/styles/styles/props/neptune-tokens.css +11 -0
  208. package/build/styles/styles/variables/legacy-variables.css +0 -0
  209. package/build/styles/styles/variables/neptune-tokens.css +0 -0
  210. package/build/styles/styles/variables/zindex.css +0 -0
  211. package/build/types/index.d.ts +1 -1
  212. package/build/types/index.d.ts.map +1 -1
  213. package/build/types/inputs/{_BottomSheet.d.ts → SelectInput/BottomSheet/SelectInputBottomSheet.d.ts} +3 -3
  214. package/build/types/inputs/SelectInput/BottomSheet/SelectInputBottomSheet.d.ts.map +1 -0
  215. package/build/types/inputs/SelectInput/BottomSheet/index.d.ts +3 -0
  216. package/build/types/inputs/SelectInput/BottomSheet/index.d.ts.map +1 -0
  217. package/build/types/inputs/SelectInput/ButtonInput/SelectInputButtonInput.d.ts +5 -0
  218. package/build/types/inputs/SelectInput/ButtonInput/SelectInputButtonInput.d.ts.map +1 -0
  219. package/build/types/inputs/SelectInput/ButtonInput/index.d.ts +3 -0
  220. package/build/types/inputs/SelectInput/ButtonInput/index.d.ts.map +1 -0
  221. package/build/types/inputs/SelectInput/ClearButton/SelectInputClearButton.d.ts +7 -0
  222. package/build/types/inputs/SelectInput/ClearButton/SelectInputClearButton.d.ts.map +1 -0
  223. package/build/types/inputs/SelectInput/ClearButton/index.d.ts +3 -0
  224. package/build/types/inputs/SelectInput/ClearButton/index.d.ts.map +1 -0
  225. package/build/types/inputs/SelectInput/DefaultRenderTrigger/SelectInputDefaultRenderTrigger.d.ts +16 -0
  226. package/build/types/inputs/SelectInput/DefaultRenderTrigger/SelectInputDefaultRenderTrigger.d.ts.map +1 -0
  227. package/build/types/inputs/SelectInput/DefaultRenderTrigger/index.d.ts +2 -0
  228. package/build/types/inputs/SelectInput/DefaultRenderTrigger/index.d.ts.map +1 -0
  229. package/build/types/inputs/SelectInput/ItemView/GroupItemView/SelectInputGroupItemView.d.ts +9 -0
  230. package/build/types/inputs/SelectInput/ItemView/GroupItemView/SelectInputGroupItemView.d.ts.map +1 -0
  231. package/build/types/inputs/SelectInput/ItemView/GroupItemView/index.d.ts +3 -0
  232. package/build/types/inputs/SelectInput/ItemView/GroupItemView/index.d.ts.map +1 -0
  233. package/build/types/inputs/SelectInput/ItemView/SelectInputItemView.d.ts +11 -0
  234. package/build/types/inputs/SelectInput/ItemView/SelectInputItemView.d.ts.map +1 -0
  235. package/build/types/inputs/SelectInput/ItemView/index.d.ts +4 -0
  236. package/build/types/inputs/SelectInput/ItemView/index.d.ts.map +1 -0
  237. package/build/types/inputs/SelectInput/Option/SelectInputOption.d.ts +11 -0
  238. package/build/types/inputs/SelectInput/Option/SelectInputOption.d.ts.map +1 -0
  239. package/build/types/inputs/SelectInput/Option/index.d.ts +3 -0
  240. package/build/types/inputs/SelectInput/Option/index.d.ts.map +1 -0
  241. package/build/types/inputs/SelectInput/OptionContent/SelectInputOptionContent.d.ts +13 -0
  242. package/build/types/inputs/SelectInput/OptionContent/SelectInputOptionContent.d.ts.map +1 -0
  243. package/build/types/inputs/SelectInput/OptionContent/index.d.ts +3 -0
  244. package/build/types/inputs/SelectInput/OptionContent/index.d.ts.map +1 -0
  245. package/build/types/inputs/SelectInput/Options/OptionsContainer/SelectInputOptionsContainer.d.ts +9 -0
  246. package/build/types/inputs/SelectInput/Options/OptionsContainer/SelectInputOptionsContainer.d.ts.map +1 -0
  247. package/build/types/inputs/SelectInput/Options/OptionsContainer/index.d.ts +3 -0
  248. package/build/types/inputs/SelectInput/Options/OptionsContainer/index.d.ts.map +1 -0
  249. package/build/types/inputs/SelectInput/Options/SelectInputOptions.d.ts +21 -0
  250. package/build/types/inputs/SelectInput/Options/SelectInputOptions.d.ts.map +1 -0
  251. package/build/types/inputs/SelectInput/Options/index.d.ts +4 -0
  252. package/build/types/inputs/SelectInput/Options/index.d.ts.map +1 -0
  253. package/build/types/inputs/{_Popover.d.ts → SelectInput/Popover/SelectInputPopover.d.ts} +3 -3
  254. package/build/types/inputs/SelectInput/Popover/SelectInputPopover.d.ts.map +1 -0
  255. package/build/types/inputs/SelectInput/Popover/index.d.ts +3 -0
  256. package/build/types/inputs/SelectInput/Popover/index.d.ts.map +1 -0
  257. package/build/types/inputs/SelectInput/SelectInput.contexts.d.ts +33 -0
  258. package/build/types/inputs/SelectInput/SelectInput.contexts.d.ts.map +1 -0
  259. package/build/types/inputs/SelectInput/SelectInput.d.ts +10 -0
  260. package/build/types/inputs/SelectInput/SelectInput.d.ts.map +1 -0
  261. package/build/types/inputs/SelectInput/SelectInput.messages.d.ts.map +1 -0
  262. package/build/types/inputs/{SelectInput.d.ts → SelectInput/SelectInput.types.d.ts} +12 -38
  263. package/build/types/inputs/SelectInput/SelectInput.types.d.ts.map +1 -0
  264. package/build/types/inputs/SelectInput/SelectInput.utils.d.ts +60 -0
  265. package/build/types/inputs/SelectInput/SelectInput.utils.d.ts.map +1 -0
  266. package/build/types/inputs/SelectInput/TriggerButton/SelectInputTriggerButton.d.ts +12 -0
  267. package/build/types/inputs/SelectInput/TriggerButton/SelectInputTriggerButton.d.ts.map +1 -0
  268. package/build/types/inputs/SelectInput/TriggerButton/index.d.ts +3 -0
  269. package/build/types/inputs/SelectInput/TriggerButton/index.d.ts.map +1 -0
  270. package/build/types/inputs/SelectInput/components.d.ts +10 -0
  271. package/build/types/inputs/SelectInput/components.d.ts.map +1 -0
  272. package/build/types/inputs/SelectInput/index.d.ts +10 -0
  273. package/build/types/inputs/SelectInput/index.d.ts.map +1 -0
  274. package/package.json +9 -5
  275. package/src/{DisabledComponents.story.tsx → DisabledComponents.test.story.tsx} +1 -1
  276. package/src/accordion/Accordion.less +3 -3
  277. package/src/actionButton/ActionButton.less +1 -1
  278. package/src/actionButton/ActionButton.story.tsx +11 -7
  279. package/src/actionButton/ActionButton.test.story.tsx +49 -0
  280. package/src/avatar/Avatar.less +3 -3
  281. package/src/avatarLayout/AvatarLayout.story.tsx +1 -105
  282. package/src/avatarLayout/AvatarLayout.test.story.tsx +117 -0
  283. package/src/avatarView/AvatarView.story.tsx +1 -104
  284. package/src/avatarView/AvatarView.test.story.tsx +114 -0
  285. package/src/badge/Badge.less +1 -1
  286. package/src/button/LegacyButton.less +1 -1
  287. package/src/button/LegacyButton.story.tsx +1 -26
  288. package/src/button/LegacyButton.test.story.tsx +37 -0
  289. package/src/button/_stories/Button.story.tsx +17 -41
  290. package/src/button/_stories/Button.test.story.tsx +30 -0
  291. package/src/card/Card.less +3 -3
  292. package/src/checkbox/Checkbox.story.tsx +1 -6
  293. package/src/checkbox/Checkbox.test.story.tsx +33 -0
  294. package/src/checkboxButton/CheckboxButton.less +2 -2
  295. package/src/chips/Chip.less +3 -3
  296. package/src/circularButton/CircularButton.story.tsx +0 -6
  297. package/src/circularButton/CircularButton.test.story.tsx +62 -1
  298. package/src/common/Option/Option.less +2 -2
  299. package/src/common/bottomSheet/BottomSheet.less +1 -1
  300. package/src/common/bottomSheet/BottomSheet.story.tsx +0 -55
  301. package/src/common/bottomSheet/BottomSheet.test.story.tsx +118 -2
  302. package/src/common/closeButton/CloseButton.less +1 -1
  303. package/src/common/panel/Panel.less +1 -1
  304. package/src/criticalBanner/CriticalCommsBanner.test.story.tsx +1 -1
  305. package/src/dateLookup/DateLookup.less +1 -1
  306. package/src/dateLookup/DateLookup.story.tsx +0 -18
  307. package/src/dateLookup/DateLookup.test.story.tsx +51 -1
  308. package/src/dateLookup/dateTrigger/DateTrigger.less +2 -2
  309. package/src/decision/Decision.less +1 -1
  310. package/src/definitionList/DefinitionList.less +1 -1
  311. package/src/dimmer/Dimmer.less +2 -2
  312. package/src/divider/Divider.story.tsx +0 -1
  313. package/src/expressiveMoneyInput/ExpressiveMoneyInput.autofocus.docs.mdx +1 -1
  314. package/src/expressiveMoneyInput/ExpressiveMoneyInput.story.tsx +0 -9
  315. package/src/expressiveMoneyInput/ExpressiveMoneyInput.test.story.tsx +23 -0
  316. package/src/field/Field.story.tsx +1 -34
  317. package/src/field/Field.test.story.tsx +56 -0
  318. package/src/flowNavigation/FlowNavigation.less +1 -1
  319. package/src/flowNavigation/animatedLabel/AnimatedLabel.less +2 -2
  320. package/src/header/Header.test.story.tsx +1 -1
  321. package/src/index.ts +0 -1
  322. package/src/info/Info.less +1 -1
  323. package/src/info/Info.story.tsx +1 -34
  324. package/src/info/Info.test.story.tsx +53 -0
  325. package/src/inputWithDisplayFormat/InputWithDisplayFormat.story.tsx +0 -12
  326. package/src/inputWithDisplayFormat/InputWithDisplayFormat.test.story.tsx +33 -0
  327. package/src/inputs/SelectInput/BottomSheet/SelectInputBottomSheet.css +96 -0
  328. package/src/inputs/{_BottomSheet.tsx → SelectInput/BottomSheet/SelectInputBottomSheet.tsx} +7 -7
  329. package/src/inputs/SelectInput/BottomSheet/index.ts +2 -0
  330. package/src/inputs/SelectInput/ButtonInput/SelectInputButtonInput.css +16 -0
  331. package/src/inputs/{_ButtonInput.tsx → SelectInput/ButtonInput/SelectInputButtonInput.tsx} +5 -5
  332. package/src/inputs/SelectInput/ButtonInput/index.ts +2 -0
  333. package/src/inputs/SelectInput/ClearButton/SelectInputClearButton.css +46 -0
  334. package/src/inputs/SelectInput/ClearButton/SelectInputClearButton.less +39 -0
  335. package/src/inputs/SelectInput/ClearButton/SelectInputClearButton.tsx +27 -0
  336. package/src/inputs/SelectInput/ClearButton/index.ts +2 -0
  337. package/src/inputs/SelectInput/DefaultRenderTrigger/SelectInputDefaultRenderTrigger.tsx +74 -0
  338. package/src/inputs/SelectInput/DefaultRenderTrigger/index.ts +5 -0
  339. package/src/inputs/SelectInput/ItemView/GroupItemView/SelectInputGroupItemView.tsx +61 -0
  340. package/src/inputs/SelectInput/ItemView/GroupItemView/index.ts +2 -0
  341. package/src/inputs/SelectInput/ItemView/SelectInputItemView.css +16 -0
  342. package/src/inputs/SelectInput/ItemView/SelectInputItemView.less +17 -0
  343. package/src/inputs/SelectInput/ItemView/SelectInputItemView.tsx +48 -0
  344. package/src/inputs/SelectInput/ItemView/index.ts +3 -0
  345. package/src/inputs/SelectInput/Option/SelectInputOption.css +33 -0
  346. package/src/inputs/SelectInput/Option/SelectInputOption.less +32 -0
  347. package/src/inputs/SelectInput/Option/SelectInputOption.tsx +57 -0
  348. package/src/inputs/SelectInput/Option/index.ts +2 -0
  349. package/src/inputs/SelectInput/OptionContent/SelectInputOptionContent.css +37 -0
  350. package/src/inputs/SelectInput/OptionContent/SelectInputOptionContent.less +38 -0
  351. package/src/inputs/SelectInput/OptionContent/SelectInputOptionContent.tsx +72 -0
  352. package/src/inputs/SelectInput/OptionContent/index.ts +2 -0
  353. package/src/inputs/SelectInput/Options/OptionsContainer/SelectInputOptionsContainer.tsx +59 -0
  354. package/src/inputs/SelectInput/Options/OptionsContainer/index.ts +2 -0
  355. package/src/inputs/SelectInput/Options/SelectInputOptions.css +81 -0
  356. package/src/inputs/SelectInput/Options/SelectInputOptions.less +77 -0
  357. package/src/inputs/SelectInput/Options/SelectInputOptions.tsx +411 -0
  358. package/src/inputs/SelectInput/Options/index.ts +3 -0
  359. package/src/inputs/SelectInput/Popover/SelectInputPopover.css +46 -0
  360. package/src/inputs/{_Popover.tsx → SelectInput/Popover/SelectInputPopover.tsx} +7 -7
  361. package/src/inputs/SelectInput/Popover/index.ts +2 -0
  362. package/src/inputs/SelectInput/SelectInput.contexts.tsx +40 -0
  363. package/src/inputs/SelectInput/SelectInput.less +22 -0
  364. package/src/inputs/{SelectInput.test.tsx → SelectInput/SelectInput.test.tsx} +9 -11
  365. package/src/inputs/SelectInput/SelectInput.tsx +257 -0
  366. package/src/inputs/SelectInput/SelectInput.types.ts +113 -0
  367. package/src/inputs/SelectInput/SelectInput.utils.ts +205 -0
  368. package/src/inputs/SelectInput/TriggerButton/SelectInputTriggerButton.tsx +36 -0
  369. package/src/inputs/SelectInput/TriggerButton/index.ts +5 -0
  370. package/src/inputs/{SelectInput.docs.mdx → SelectInput/_stories/SelectInput.docs.mdx} +0 -1
  371. package/src/inputs/{SelectInput.story.tsx → SelectInput/_stories/SelectInput.story.tsx} +11 -8
  372. package/src/inputs/{SelectInput.test.story.tsx → SelectInput/_stories/SelectInput.test.story.tsx} +6 -10
  373. package/src/inputs/SelectInput/components.ts +10 -0
  374. package/src/inputs/SelectInput/index.ts +12 -0
  375. package/src/inputs/_common.less +1 -1
  376. package/src/legacylistItem/LegacyListItem.test.story.tsx +1 -1
  377. package/src/link/Link.less +2 -2
  378. package/src/listItem/ListItem.less +1 -1
  379. package/src/listItem/_stories/Breakpoints/ListItem.noMedia.test.story.tsx +1 -1
  380. package/src/listItem/_stories/Breakpoints/ListItem.sideMedia.test.story.tsx +1 -1
  381. package/src/listItem/_stories/Breakpoints/ListItem.stackedMedia.test.story.tsx +1 -1
  382. package/src/listItem/_stories/ListItem.focus.test.story.tsx +1 -1
  383. package/src/listItem/_stories/ListItem.layout.test.story.tsx +1 -1
  384. package/src/listItem/_stories/variants/ListItem.brightGreen.test.story.tsx +1 -1
  385. package/src/listItem/_stories/variants/ListItem.dark.test.story.tsx +1 -1
  386. package/src/listItem/_stories/variants/ListItem.forestGreen.test.story.tsx +1 -1
  387. package/src/listItem/_stories/variants/ListItem.medium.test.story.tsx +1 -1
  388. package/src/listItem/_stories/variants/ListItem.neutral.test.story.tsx +1 -1
  389. package/src/listItem/_stories/variants/ListItem.personal.test.story.tsx +1 -1
  390. package/src/listItem/_stories/variants/ListItem.rtl.test.story.tsx +1 -1
  391. package/src/listItem/_stories/variants/ListItem.small.test.story.tsx +1 -1
  392. package/src/logo/Logo.story.tsx +0 -6
  393. package/src/logo/Logo.test.story.tsx +20 -0
  394. package/src/main.css +27094 -97
  395. package/src/main.less +3 -1
  396. package/src/modal/Modal.story.tsx +0 -26
  397. package/src/modal/Modal.test.story.tsx +125 -0
  398. package/src/moneyInput/MoneyInput.less +2 -2
  399. package/src/moneyInput/MoneyInput.test.story.tsx +1 -1
  400. package/src/navigationOption/NavigationOption.less +3 -3
  401. package/src/navigationOptionsList/NavigationOptionsList.less +1 -1
  402. package/src/overlayHeader/OverlayHeader.less +1 -1
  403. package/src/phoneNumberInput/PhoneNumberInput.less +1 -1
  404. package/src/popover/Popover.less +1 -1
  405. package/src/prompt/ActionPrompt/ActionPrompt.story.tsx +0 -43
  406. package/src/prompt/ActionPrompt/ActionPrompt.test.story.tsx +45 -1
  407. package/src/radioGroup/RadioGroup.test.story.tsx +1 -1
  408. package/src/select/Select.less +5 -5
  409. package/src/slidingPanel/SlidingPanel.less +2 -2
  410. package/src/snackbar/Snackbar.less +3 -3
  411. package/src/stepper/Stepper.less +3 -3
  412. package/src/stepper/Stepper.test.story.tsx +1 -1
  413. package/src/styles/fonts/TW-Averta-Bold.woff +0 -0
  414. package/src/styles/fonts/TW-Averta-Bold.woff2 +0 -0
  415. package/src/styles/fonts/TW-Averta-Regular.woff +0 -0
  416. package/src/styles/fonts/TW-Averta-Regular.woff2 +0 -0
  417. package/src/styles/fonts/TW-Averta-Semibold.woff +0 -0
  418. package/src/styles/fonts/TW-Averta-Semibold.woff2 +0 -0
  419. package/src/styles/fonts/TransferWise-IconFont.svg +81 -0
  420. package/src/styles/fonts/TransferWise-IconFont.woff +0 -0
  421. package/src/styles/fonts/WiseSans-Heavy.woff2 +0 -0
  422. package/src/styles/fonts/inter-cyrillic-ext-variable-wghtOnly-normal.woff2 +0 -0
  423. package/src/styles/fonts/inter-cyrillic-variable-wghtOnly-normal.woff2 +0 -0
  424. package/src/styles/fonts/inter-greek-ext-variable-wghtOnly-normal.woff2 +0 -0
  425. package/src/styles/fonts/inter-greek-variable-wghtOnly-normal.woff2 +0 -0
  426. package/src/styles/fonts/inter-latin-ext-variable-wghtOnly-normal.woff2 +0 -0
  427. package/src/styles/fonts/inter-latin-variable-wghtOnly-normal.woff2 +0 -0
  428. package/src/styles/fonts/inter-vietnamese-variable-wghtOnly-normal.woff2 +0 -0
  429. package/src/styles/img/bg-dark.svg +31 -0
  430. package/src/styles/img/bg-light.svg +26 -0
  431. package/src/styles/less/accordion.css +255 -0
  432. package/src/styles/less/accordion.less +15 -0
  433. package/src/styles/less/addons/_background-utilities.less +82 -0
  434. package/src/styles/less/addons/_border.less +3 -0
  435. package/src/styles/less/addons/_display-utilities.less +159 -0
  436. package/src/styles/less/addons/_spacing-utilities.less +73 -0
  437. package/src/styles/less/addons/_utilities.less +147 -0
  438. package/src/styles/less/alerts.css +751 -0
  439. package/src/styles/less/alerts.less +252 -0
  440. package/src/styles/less/background.css +20 -0
  441. package/src/styles/less/background.less +24 -0
  442. package/src/styles/less/badge.css +90 -0
  443. package/src/styles/less/badge.less +85 -0
  444. package/src/styles/less/border-radius.css +3 -0
  445. package/src/styles/less/border-radius.less +3 -0
  446. package/src/styles/less/breadcrumbs.css +20 -0
  447. package/src/styles/less/breadcrumbs.less +27 -0
  448. package/src/styles/less/button-groups.css +495 -0
  449. package/src/styles/less/button-groups.less +424 -0
  450. package/src/styles/less/buttons.css +1390 -0
  451. package/src/styles/less/buttons.less +738 -0
  452. package/src/styles/less/chevron.css +35 -0
  453. package/src/styles/less/chevron.less +39 -0
  454. package/src/styles/less/circles.css +248 -0
  455. package/src/styles/less/circles.less +233 -0
  456. package/src/styles/less/close.css +47 -0
  457. package/src/styles/less/close.less +48 -0
  458. package/src/styles/less/column-layout.css +164 -0
  459. package/src/styles/less/column-layout.less +167 -0
  460. package/src/styles/less/components/_component-animations.less +43 -0
  461. package/src/styles/less/core/_fonts.less +97 -0
  462. package/src/styles/less/core/_scaffolding.less +380 -0
  463. package/src/styles/less/core/_typography-utilities.less +255 -0
  464. package/src/styles/less/core/_typography.less +575 -0
  465. package/src/styles/less/core/viewport-themes.css +59 -0
  466. package/src/styles/less/core/viewport-themes.less +65 -0
  467. package/src/styles/less/currency-flags.css +46 -0
  468. package/src/styles/less/currency-flags.less +42 -0
  469. package/src/styles/less/decision.css +136 -0
  470. package/src/styles/less/decision.less +98 -0
  471. package/src/styles/less/dropdowns.css +472 -0
  472. package/src/styles/less/dropdowns.less +365 -0
  473. package/src/styles/less/droppable.css +221 -0
  474. package/src/styles/less/droppable.less +213 -0
  475. package/src/styles/less/flex.css +598 -0
  476. package/src/styles/less/flex.less +113 -0
  477. package/src/styles/less/footer.css +129 -0
  478. package/src/styles/less/footer.less +113 -0
  479. package/src/styles/less/forms/bootstrap-forms.css +2277 -0
  480. package/src/styles/less/forms/bootstrap-forms.less +1218 -0
  481. package/src/styles/less/forms/checkbox-radio.css +367 -0
  482. package/src/styles/less/forms/checkbox-radio.less +330 -0
  483. package/src/styles/less/grid.css +2369 -0
  484. package/src/styles/less/grid.less +187 -0
  485. package/src/styles/less/input-groups.css +2740 -0
  486. package/src/styles/less/input-groups.less +298 -0
  487. package/src/styles/less/link-callout.css +20 -0
  488. package/src/styles/less/link-callout.less +17 -0
  489. package/src/styles/less/list-group.css +350 -0
  490. package/src/styles/less/list-group.less +260 -0
  491. package/src/styles/less/loaders.css +76 -0
  492. package/src/styles/less/loaders.less +91 -0
  493. package/src/styles/less/media.css +74 -0
  494. package/src/styles/less/media.less +68 -0
  495. package/src/styles/less/mixins/_alerts.less +39 -0
  496. package/src/styles/less/mixins/_arrows.less +52 -0
  497. package/src/styles/less/mixins/_border-radius.less +11 -0
  498. package/src/styles/less/mixins/_buttons.less +82 -0
  499. package/src/styles/less/mixins/_center-block.less +7 -0
  500. package/src/styles/less/mixins/_clearfix.less +23 -0
  501. package/src/styles/less/mixins/_flex.less +105 -0
  502. package/src/styles/less/mixins/_forms.less +130 -0
  503. package/src/styles/less/mixins/_grid-framework.less +104 -0
  504. package/src/styles/less/mixins/_grid.less +158 -0
  505. package/src/styles/less/mixins/_hide-text.less +20 -0
  506. package/src/styles/less/mixins/_links.less +7 -0
  507. package/src/styles/less/mixins/_list-group.less +23 -0
  508. package/src/styles/less/mixins/_logical-properties-IE-friendly.less +381 -0
  509. package/src/styles/less/mixins/_logical-properties-modern-browsers.less +226 -0
  510. package/src/styles/less/mixins/_logical-properties.less +5 -0
  511. package/src/styles/less/mixins/_nav-divider.less +10 -0
  512. package/src/styles/less/mixins/_nav-vertical-align.less +10 -0
  513. package/src/styles/less/mixins/_panels.less +23 -0
  514. package/src/styles/less/mixins/_physical-properties.less +110 -0
  515. package/src/styles/less/mixins/_reset-text.less +22 -0
  516. package/src/styles/less/mixins/_sequence.less +203 -0
  517. package/src/styles/less/mixins/_spacing.less +122 -0
  518. package/src/styles/less/mixins/_tab-focus.less +5 -0
  519. package/src/styles/less/mixins/_table-row.less +47 -0
  520. package/src/styles/less/mixins/_text-emphasis.less +39 -0
  521. package/src/styles/less/mixins/_text-overflow.less +8 -0
  522. package/src/styles/less/mixins/_theming.less +7 -0
  523. package/src/styles/less/modals.css +185 -0
  524. package/src/styles/less/modals.less +239 -0
  525. package/src/styles/less/navbar.css +2406 -0
  526. package/src/styles/less/navbar.less +1442 -0
  527. package/src/styles/less/navs.css +296 -0
  528. package/src/styles/less/navs.less +339 -0
  529. package/src/styles/less/neptune-social-media.css +144 -0
  530. package/src/styles/less/neptune-social-media.less +84 -0
  531. package/src/styles/less/neptune.bundle.less +63 -0
  532. package/src/styles/less/neptune.css +25724 -0
  533. package/src/styles/less/panels.css +528 -0
  534. package/src/styles/less/panels.less +410 -0
  535. package/src/styles/less/popovers.css +661 -0
  536. package/src/styles/less/popovers.less +379 -0
  537. package/src/styles/less/process.css +483 -0
  538. package/src/styles/less/process.less +426 -0
  539. package/src/styles/less/progress-bars.css +64 -0
  540. package/src/styles/less/progress-bars.less +55 -0
  541. package/src/styles/less/ring.css +55 -0
  542. package/src/styles/less/ring.less +51 -0
  543. package/src/styles/less/select.css +99 -0
  544. package/src/styles/less/select.less +88 -0
  545. package/src/styles/less/sequences.css +855 -0
  546. package/src/styles/less/sequences.less +511 -0
  547. package/src/styles/less/table.css +446 -0
  548. package/src/styles/less/table.less +385 -0
  549. package/src/styles/less/tick.css +32 -0
  550. package/src/styles/less/tick.less +37 -0
  551. package/src/styles/less/tooltip.css +160 -0
  552. package/src/styles/less/tooltip.less +130 -0
  553. package/src/styles/less/utilities.css +145 -0
  554. package/src/styles/less/utilities.less +153 -0
  555. package/src/styles/less/wells.css +74 -0
  556. package/src/styles/less/wells.less +37 -0
  557. package/src/styles/props/custom-media.css +11 -0
  558. package/src/styles/props/legacy-custom-props.css +69 -0
  559. package/src/styles/props/neptune-tokens.css +11 -0
  560. package/src/styles/variables/legacy-variables.css +0 -0
  561. package/src/styles/variables/legacy-variables.less +1299 -0
  562. package/src/styles/variables/neptune-tokens.css +0 -0
  563. package/src/styles/variables/neptune-tokens.less +12 -0
  564. package/src/styles/variables/zindex.css +0 -0
  565. package/src/styles/variables/zindex.less +17 -0
  566. package/src/summary/Summary.less +1 -1
  567. package/src/summary/Summary.test.story.tsx +1 -1
  568. package/src/switch/Switch.less +4 -4
  569. package/src/table/Table.less +1 -1
  570. package/src/tabs/Tabs.less +4 -4
  571. package/src/tile/Tile.less +1 -1
  572. package/src/tooltip/Tooltip.less +1 -1
  573. package/src/tooltip/Tooltip.story.tsx +0 -21
  574. package/src/tooltip/Tooltip.test.story.tsx +42 -0
  575. package/src/typeahead/Typeahead.less +2 -2
  576. package/src/typeahead/Typeahead.story.tsx +0 -52
  577. package/src/typeahead/Typeahead.test.story.tsx +73 -0
  578. package/src/typeahead/typeaheadOption/TypeaheadOption.less +1 -1
  579. package/src/upload/Upload.less +1 -1
  580. package/src/uploadInput/UploadInput.less +2 -2
  581. package/src/uploadInput/uploadItem/UploadItem.less +1 -1
  582. package/build/inputs/SelectInput.js +0 -890
  583. package/build/inputs/SelectInput.js.map +0 -1
  584. package/build/inputs/SelectInput.messages.js.map +0 -1
  585. package/build/inputs/SelectInput.messages.mjs.map +0 -1
  586. package/build/inputs/SelectInput.mjs +0 -881
  587. package/build/inputs/SelectInput.mjs.map +0 -1
  588. package/build/inputs/_BottomSheet.js.map +0 -1
  589. package/build/inputs/_BottomSheet.mjs.map +0 -1
  590. package/build/inputs/_ButtonInput.js.map +0 -1
  591. package/build/inputs/_ButtonInput.mjs.map +0 -1
  592. package/build/inputs/_Popover.js.map +0 -1
  593. package/build/inputs/_Popover.mjs.map +0 -1
  594. package/build/types/inputs/SelectInput.d.ts.map +0 -1
  595. package/build/types/inputs/SelectInput.messages.d.ts.map +0 -1
  596. package/build/types/inputs/_BottomSheet.d.ts.map +0 -1
  597. package/build/types/inputs/_ButtonInput.d.ts +0 -5
  598. package/build/types/inputs/_ButtonInput.d.ts.map +0 -1
  599. package/build/types/inputs/_Popover.d.ts.map +0 -1
  600. package/src/inputs/SelectInput.less +0 -219
  601. package/src/inputs/SelectInput.tsx +0 -1269
  602. package/build/inputs/{SelectInput.messages.js → SelectInput/SelectInput.messages.js} +0 -0
  603. package/build/inputs/{SelectInput.messages.mjs → SelectInput/SelectInput.messages.mjs} +0 -0
  604. package/build/styles/inputs/{SelectInput.css → SelectInput/SelectInput.css} +90 -90
  605. package/build/types/inputs/{SelectInput.messages.d.ts → SelectInput/SelectInput.messages.d.ts} +0 -0
  606. package/src/inputs/{_BottomSheet.less → SelectInput/BottomSheet/SelectInputBottomSheet.less} +0 -0
  607. package/src/inputs/{_ButtonInput.less → SelectInput/ButtonInput/SelectInputButtonInput.less} +0 -0
  608. package/src/inputs/{_Popover.less → SelectInput/Popover/SelectInputPopover.less} +0 -0
  609. package/src/inputs/{SelectInput.css → SelectInput/SelectInput.css} +90 -90
  610. /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
- }