@wise/dynamic-flow-client 5.8.1 → 5.9.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 (420) hide show
  1. package/build/DynamicFlowCore.js +5 -0
  2. package/build/common/errorBoundary/ErrorBoundary.js +26 -0
  3. package/build/common/errorBoundary/ErrorBoundary.test.js +29 -0
  4. package/build/common/errorBoundary/ErrorBoundaryAlert.js +11 -0
  5. package/build/common/makeHttpClient/api-utils.js +3 -0
  6. package/build/common/makeHttpClient/index.js +1 -0
  7. package/build/common/makeHttpClient/makeHttpClient.js +10 -0
  8. package/build/common/makeHttpClient/makeHttpClient.test.js +186 -0
  9. package/build/common/messages/external-confirmation.messages.js +23 -0
  10. package/build/common/messages/file-upload.messages.js +13 -0
  11. package/build/common/messages/generic-error.messages.js +18 -0
  12. package/build/common/messages/help.messages.js +8 -0
  13. package/build/common/messages/multi-file-upload.messages.js +18 -0
  14. package/build/common/messages/multi-select.messages.js +8 -0
  15. package/build/common/messages/paragraph.messages.js +13 -0
  16. package/build/common/messages/repeatable.messages.js +23 -0
  17. package/build/common/messages/search.messages.js +8 -0
  18. package/build/common/messages/validation.array.messages.js +13 -0
  19. package/build/common/messages/validation.messages.js +53 -0
  20. package/build/controller/FlowController.js +368 -0
  21. package/build/controller/executePoll.js +49 -0
  22. package/build/controller/executeRefresh.js +39 -0
  23. package/build/controller/executeRequest.js +77 -0
  24. package/build/controller/executeSubmission.js +69 -0
  25. package/build/controller/getErrorMessage.js +7 -0
  26. package/build/controller/getRequestAbortController.js +13 -0
  27. package/build/controller/getResponseType.js +35 -0
  28. package/build/controller/getSafeHttpClient.js +8 -0
  29. package/build/controller/getStepCounter.js +16 -0
  30. package/build/controller/handleErrorResponse.js +26 -0
  31. package/build/controller/makeSafeHttpClient.js +8 -0
  32. package/build/controller/response-utils.js +72 -0
  33. package/build/domain/components/AlertComponent.js +1 -0
  34. package/build/domain/components/AllOfComponent.js +40 -0
  35. package/build/domain/components/BooleanInputComponent.js +50 -0
  36. package/build/domain/components/BoxComponent.js +3 -0
  37. package/build/domain/components/ButtonComponent.js +1 -0
  38. package/build/domain/components/ColumnsComponent.js +3 -0
  39. package/build/domain/components/ConstComponent.js +18 -0
  40. package/build/domain/components/ContainerComponent.js +3 -0
  41. package/build/domain/components/DateInputComponent.js +75 -0
  42. package/build/domain/components/DecisionComponent.js +1 -0
  43. package/build/domain/components/DividerComponent.js +1 -0
  44. package/build/domain/components/FormComponent.js +3 -0
  45. package/build/domain/components/FormattedValueComponent.js +44 -0
  46. package/build/domain/components/HeadingComponent.js +1 -0
  47. package/build/domain/components/ImageComponent.js +1 -0
  48. package/build/domain/components/InstructionsComponent.js +1 -0
  49. package/build/domain/components/IntegerInputComponent.js +74 -0
  50. package/build/domain/components/ListComponent.js +1 -0
  51. package/build/domain/components/LoadingIndicatorComponent.js +1 -0
  52. package/build/domain/components/MarkdownComponent.js +1 -0
  53. package/build/domain/components/MediaComponent.js +1 -0
  54. package/build/domain/components/ModalComponent.js +16 -0
  55. package/build/domain/components/ModalLayoutComponent.js +3 -0
  56. package/build/domain/components/MoneyInputComponent.js +57 -0
  57. package/build/domain/components/MultiSelectInputComponent.js +81 -0
  58. package/build/domain/components/MultiUploadInputComponent.js +88 -0
  59. package/build/domain/components/NumberInputComponent.js +73 -0
  60. package/build/domain/components/ObjectComponent.js +45 -0
  61. package/build/domain/components/ParagraphComponent.js +1 -0
  62. package/build/domain/components/PersistAsyncComponent.js +92 -0
  63. package/build/domain/components/ProgressComponent.js +1 -0
  64. package/build/domain/components/RepeatableComponent.js +103 -0
  65. package/build/domain/components/ReviewComponent.js +1 -0
  66. package/build/domain/components/RootDomainComponent.js +173 -0
  67. package/build/domain/components/SectionComponent.js +5 -0
  68. package/build/domain/components/SelectInputComponent.js +88 -0
  69. package/build/domain/components/StatusListComponent.js +1 -0
  70. package/build/domain/components/SubflowDomainComponent.js +9 -0
  71. package/build/domain/components/TabsComponent.js +1 -0
  72. package/build/domain/components/TextInputComponent.js +76 -0
  73. package/build/domain/components/TupleComponent.js +41 -0
  74. package/build/domain/components/UploadInputComponent.js +83 -0
  75. package/build/domain/components/UpsellComponent.js +25 -0
  76. package/build/domain/components/searchComponent/SearchComponent.js +92 -0
  77. package/build/domain/components/searchComponent/SearchComponent.test.js +190 -0
  78. package/build/domain/components/step/ExternalConfirmationComponent.js +28 -0
  79. package/build/domain/components/step/StepDomainComponent.js +73 -0
  80. package/build/domain/components/step/ToolbarComponent.js +1 -0
  81. package/build/domain/components/utils/WithUpdate.js +1 -0
  82. package/build/domain/components/utils/component-utils.js +12 -0
  83. package/build/domain/components/utils/debounce.js +34 -0
  84. package/build/domain/components/utils/debounce.test.js +67 -0
  85. package/build/domain/components/utils/file-utils.js +21 -0
  86. package/build/domain/components/utils/file-utils.test.js +27 -0
  87. package/build/domain/components/utils/getRandomId.js +1 -0
  88. package/build/domain/components/utils/isExactLocalValueMatch.js +14 -0
  89. package/build/domain/components/utils/isOrWasValid.js +5 -0
  90. package/build/domain/components/utils/isPartialModelMatch.js +18 -0
  91. package/build/domain/components/utils/isPartialModelMatch.test.js +74 -0
  92. package/build/domain/features/eventNames.js +24 -0
  93. package/build/domain/features/events.js +1 -0
  94. package/build/domain/features/persistAsync/getComponentMultiPersistAsync.js +50 -0
  95. package/build/domain/features/persistAsync/getInitialPersistedState.js +7 -0
  96. package/build/domain/features/persistAsync/getPerformPersistAsync.js +43 -0
  97. package/build/domain/features/persistAsync/getPerformPersistAsync.test.js +139 -0
  98. package/build/domain/features/polling/getStepPolling.js +43 -0
  99. package/build/domain/features/polling/getStepPolling.test.js +90 -0
  100. package/build/domain/features/prefetch/getStepPrefetch.js +43 -0
  101. package/build/domain/features/prefetch/request-cache.js +49 -0
  102. package/build/domain/features/prefetch/request-cache.test.js +70 -0
  103. package/build/domain/features/refreshAfter/getStepRefreshAfter.js +24 -0
  104. package/build/domain/features/refreshAfter/getStepRefreshAfter.test.js +40 -0
  105. package/build/domain/features/schema-on-change/getDebouncedSchemaOnChange.js +50 -0
  106. package/build/domain/features/schema-on-change/getSchemaOnChange.js +34 -0
  107. package/build/domain/features/search/getPerformSearchFunction.js +75 -0
  108. package/build/domain/features/search/getPerformSearchFunction.test.js +301 -0
  109. package/build/domain/features/summary/summary-utils.js +40 -0
  110. package/build/domain/features/summary/summary-utils.test.js +125 -0
  111. package/build/domain/features/utils/http-utils.js +21 -0
  112. package/build/domain/features/utils/response-utils.js +9 -0
  113. package/build/domain/features/validation/spec-utils.js +19 -0
  114. package/build/domain/features/validation/validateStringPattern.js +24 -0
  115. package/build/domain/features/validation/validation-functions.js +6 -0
  116. package/build/domain/features/validation/validation-functions.test.js +108 -0
  117. package/build/domain/features/validation/value-checks.js +125 -0
  118. package/build/domain/features/validation/value-checks.test.js +262 -0
  119. package/build/domain/features/validationAsync/getComponentValidationAsync.js +53 -0
  120. package/build/domain/features/validationAsync/getComponentValidationAsync.test.js +67 -0
  121. package/build/domain/features/validationAsync/getInitialValidationAsyncState.js +5 -0
  122. package/build/domain/features/validationAsync/getPerformValidationAsync.js +45 -0
  123. package/build/domain/features/validationAsync/getPerformValidationAsync.test.js +143 -0
  124. package/build/domain/mappers/layout/alertLayoutToComponent.js +16 -0
  125. package/build/domain/mappers/layout/boxLayoutToComponent.js +13 -0
  126. package/build/domain/mappers/layout/buttonLayoutToComponent.js +77 -0
  127. package/build/domain/mappers/layout/columnsLayoutToComponent.js +13 -0
  128. package/build/domain/mappers/layout/decisionLayoutToComponent.js +22 -0
  129. package/build/domain/mappers/layout/deprecatedListLayoutToComponent.js +30 -0
  130. package/build/domain/mappers/layout/dividerLayoutToComponent.js +2 -0
  131. package/build/domain/mappers/layout/formLayoutToComponent.js +19 -0
  132. package/build/domain/mappers/layout/headingLayoutToComponent.js +12 -0
  133. package/build/domain/mappers/layout/imageLayoutToComponent.js +20 -0
  134. package/build/domain/mappers/layout/infoLayoutToComponent.js +12 -0
  135. package/build/domain/mappers/layout/instructionsLayoutToComponent.js +12 -0
  136. package/build/domain/mappers/layout/listLayoutToComponent.js +39 -0
  137. package/build/domain/mappers/layout/loadingIndicatorLayoutToComponent.js +9 -0
  138. package/build/domain/mappers/layout/markdownLayoutToComponent.js +12 -0
  139. package/build/domain/mappers/layout/mediaLayoutToComponent.js +12 -0
  140. package/build/domain/mappers/layout/modalLayoutToComponent.js +17 -0
  141. package/build/domain/mappers/layout/modalToComponent.js +8 -0
  142. package/build/domain/mappers/layout/paragraphLayoutToComponent.js +12 -0
  143. package/build/domain/mappers/layout/progressLayoutToComponent.js +15 -0
  144. package/build/domain/mappers/layout/reviewLayoutToComponent.js +48 -0
  145. package/build/domain/mappers/layout/searchLayoutToComponent.js +44 -0
  146. package/build/domain/mappers/layout/sectionLayoutToComponent.js +15 -0
  147. package/build/domain/mappers/layout/statusListLayoutToComponent.js +15 -0
  148. package/build/domain/mappers/layout/tabsLayoutToComponent.js +16 -0
  149. package/build/domain/mappers/layout/upsellLayoutToComponent.js +25 -0
  150. package/build/domain/mappers/mapLayoutToComponent.js +81 -0
  151. package/build/domain/mappers/mapSchemaToComponent.js +61 -0
  152. package/build/domain/mappers/mapSchemaToComponent.test.js +112 -0
  153. package/build/domain/mappers/mapStepSchemas.js +17 -0
  154. package/build/domain/mappers/mapStepToComponent.js +132 -0
  155. package/build/domain/mappers/mapStepToComponent.test.js +221 -0
  156. package/build/domain/mappers/mapToolbarToComponent.js +15 -0
  157. package/build/domain/mappers/schema/allOfSchemaToComponent.js +16 -0
  158. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToComponent.js +26 -0
  159. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiSelectComponent.js +55 -0
  160. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiUploadComponent.js +67 -0
  161. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToRepeatableComponent.js +57 -0
  162. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToTupleComponent.js +20 -0
  163. package/build/domain/mappers/schema/blobSchemaToComponent.js +15 -0
  164. package/build/domain/mappers/schema/booleanSchemaToComponent.js +29 -0
  165. package/build/domain/mappers/schema/constSchemaToComponent.js +23 -0
  166. package/build/domain/mappers/schema/integerSchemaToComponent.js +28 -0
  167. package/build/domain/mappers/schema/numberSchemaToComponent.js +26 -0
  168. package/build/domain/mappers/schema/objectSchemaToComponent/assertDisplayOrder.js +23 -0
  169. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToFormattedValueComponent.js +9 -0
  170. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.js +119 -0
  171. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.test.js +96 -0
  172. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.js +31 -0
  173. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.test.js +99 -0
  174. package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.js +76 -0
  175. package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.test.js +265 -0
  176. package/build/domain/mappers/schema/persistAsyncSchemaToComponent.js +29 -0
  177. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.js +18 -0
  178. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.test.js +133 -0
  179. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToDateInputComponent.js +48 -0
  180. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToTextInputComponent.js +37 -0
  181. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToUploadInputComponent.js +28 -0
  182. package/build/domain/mappers/schema/tests/test-utils.js +16 -0
  183. package/build/domain/mappers/schema/types.js +1 -0
  184. package/build/domain/mappers/schema/utils/getPerformPersistAsyncFn.js +19 -0
  185. package/build/domain/mappers/schema/utils/getValidationAsyncInitialState.js +23 -0
  186. package/build/domain/mappers/schema/utils/mapCommonSchemaProps.js +16 -0
  187. package/build/domain/mappers/types.js +1 -0
  188. package/build/domain/mappers/utils/FeatureFlags.js +22 -0
  189. package/build/domain/mappers/utils/behavior-utils.js +44 -0
  190. package/build/domain/mappers/utils/call-to-action-utils.js +21 -0
  191. package/build/domain/mappers/utils/getAutocompleteString.js +76 -0
  192. package/build/domain/mappers/utils/getAutocompleteString.test.js +21 -0
  193. package/build/domain/mappers/utils/groupLayoutByPinned.js +38 -0
  194. package/build/domain/mappers/utils/groupLayoutByPinned.test.js +166 -0
  195. package/build/domain/mappers/utils/image.js +9 -0
  196. package/build/domain/mappers/utils/layout-utils.js +11 -0
  197. package/build/domain/mappers/utils/legacy-utils.js +49 -0
  198. package/build/domain/mappers/utils/media-utils.js +14 -0
  199. package/build/domain/mappers/utils/suggestions-utils.js +26 -0
  200. package/build/domain/mappers/utils/suggestions-utils.test.js +36 -0
  201. package/build/domain/mappers/utils/tags-utils.js +1 -0
  202. package/build/domain/mappers/utils/utils.js +35 -0
  203. package/build/domain/types.js +1 -0
  204. package/build/getSubflowCallbacks.js +38 -0
  205. package/build/i18n/index.js +40 -0
  206. package/build/index.js +8 -0
  207. package/build/main.js +112 -54
  208. package/build/main.mjs +111 -53
  209. package/build/renderers/CoreContainerRenderer.js +5 -0
  210. package/build/renderers/CoreRootRenderer.js +12 -0
  211. package/build/renderers/EmptyLoadingStateRenderer.js +5 -0
  212. package/build/renderers/getRenderFunction.js +24 -0
  213. package/build/renderers/getSchemaErrorMessageFunction.js +97 -0
  214. package/build/renderers/mappers/alertComponentToProps.js +2 -0
  215. package/build/renderers/mappers/allOfComponentToProps.js +6 -0
  216. package/build/renderers/mappers/booleanInputComponentToProps.js +5 -0
  217. package/build/renderers/mappers/boxComponentToProps.js +13 -0
  218. package/build/renderers/mappers/buttonComponentToProps.js +4 -0
  219. package/build/renderers/mappers/columnsComponentToProps.js +11 -0
  220. package/build/renderers/mappers/componentToRendererProps.js +164 -0
  221. package/build/renderers/mappers/constComponentToProps.js +5 -0
  222. package/build/renderers/mappers/containerComponentToProps.js +7 -0
  223. package/build/renderers/mappers/dateInputComponentToProps.js +2 -0
  224. package/build/renderers/mappers/decisionComponentToProps.js +16 -0
  225. package/build/renderers/mappers/dividerComponentToProps.js +2 -0
  226. package/build/renderers/mappers/externalComponentToProps.js +3 -0
  227. package/build/renderers/mappers/formComponentToProps.js +12 -0
  228. package/build/renderers/mappers/formattedValueComponentToProps.js +5 -0
  229. package/build/renderers/mappers/headingComponentToProps.js +2 -0
  230. package/build/renderers/mappers/hiddenComponentToProps.js +4 -0
  231. package/build/renderers/mappers/imageComponentToProps.js +2 -0
  232. package/build/renderers/mappers/instructionsComponentToProps.js +2 -0
  233. package/build/renderers/mappers/integerInputComponentToProps.js +2 -0
  234. package/build/renderers/mappers/listComponentToProps.js +2 -0
  235. package/build/renderers/mappers/loadingIndicatorComponentToProps.js +2 -0
  236. package/build/renderers/mappers/markdownComponentToProps.js +2 -0
  237. package/build/renderers/mappers/mediaComponentToProps.js +2 -0
  238. package/build/renderers/mappers/modalComponentToProps.js +11 -0
  239. package/build/renderers/mappers/modalLayoutComponentToProps.js +16 -0
  240. package/build/renderers/mappers/moneyInputComponentToProps.js +36 -0
  241. package/build/renderers/mappers/multiSelectComponentToProps.js +23 -0
  242. package/build/renderers/mappers/multiUploadInputComponentToProps.js +12 -0
  243. package/build/renderers/mappers/numberInputComponentToProps.js +2 -0
  244. package/build/renderers/mappers/objectComponentToProps.js +8 -0
  245. package/build/renderers/mappers/paragraphComponentToProps.js +2 -0
  246. package/build/renderers/mappers/persistAsyncComponentToProps.js +8 -0
  247. package/build/renderers/mappers/progressComponentToProps.js +2 -0
  248. package/build/renderers/mappers/repeatableComponentToProps.js +30 -0
  249. package/build/renderers/mappers/reviewComponentToProps.js +2 -0
  250. package/build/renderers/mappers/rootComponentToProps.js +21 -0
  251. package/build/renderers/mappers/searchComponentToProps.js +57 -0
  252. package/build/renderers/mappers/sectionComponentToProps.js +6 -0
  253. package/build/renderers/mappers/selectInputComponentToProps.js +26 -0
  254. package/build/renderers/mappers/statusListComponentToProps.js +2 -0
  255. package/build/renderers/mappers/subflowComponentToRendererProps.js +4 -0
  256. package/build/renderers/mappers/tabsComponentToProps.js +14 -0
  257. package/build/renderers/mappers/textInputComponentToProps.js +2 -0
  258. package/build/renderers/mappers/tupleComponentToProps.js +8 -0
  259. package/build/renderers/mappers/uploadInputComponentToProps.js +8 -0
  260. package/build/renderers/mappers/upsellComponentToProps.js +2 -0
  261. package/build/renderers/mappers/utils/getValidationState.js +12 -0
  262. package/build/renderers/mappers/utils/inputComponentToProps.js +26 -0
  263. package/build/renderers/mappers/utils/mapErrorsToValidationState.js +9 -0
  264. package/build/renderers/mappers/utils/pick.js +8 -0
  265. package/build/renderers/mappers/utils/selectInputOptionsToProps.js +11 -0
  266. package/build/renderers/stepComponentToProps.js +32 -0
  267. package/build/renderers/utils.js +69 -0
  268. package/build/renderers/utils.test.js +70 -0
  269. package/build/stories/dev-tools/ContainerQueries.story.js +66 -0
  270. package/build/stories/dev-tools/Debugger.story.js +38 -0
  271. package/build/stories/dev-tools/FixtureSelect.story.js +23 -0
  272. package/build/stories/dev-tools/TestServer.story.js +32 -0
  273. package/build/stories/examples/NativeFlow.story.js +80 -0
  274. package/build/stories/examples/Recipients.story.js +568 -0
  275. package/build/stories/spec/behavior/Copy.story.js +59 -0
  276. package/build/stories/spec/behavior/Modal.story.js +76 -0
  277. package/build/stories/spec/behavior/Subflow.story.js +267 -0
  278. package/build/stories/spec/layouts/Decision.story.js +241 -0
  279. package/build/stories/spec/layouts/Image.story.js +42 -0
  280. package/build/stories/spec/layouts/Modal.story.js +81 -0
  281. package/build/stories/spec/layouts/Search.story.js +325 -0
  282. package/build/stories/spec/layouts/Upsell.story.js +55 -0
  283. package/build/stories/spec/layouts/button/Button.story.js +100 -0
  284. package/build/stories/spec/layouts/button/PinnedButton.story.js +81 -0
  285. package/build/stories/spec/response/ActionResponse.story.js +66 -0
  286. package/build/stories/spec/schemas/MultiSelect.story.js +148 -0
  287. package/build/stories/spec/schemas/Upload.story.js +168 -0
  288. package/build/stories/spec/schemas/const/ConstLayout.story.js +159 -0
  289. package/build/stories/spec/schemas/const/ObjectConst.story.js +94 -0
  290. package/build/stories/spec/schemas/features/PersistAsync.story.js +176 -0
  291. package/build/stories/spec/schemas/features/ValidationAsync.story.js +103 -0
  292. package/build/stories/spec/schemas/object/FormattedValue.story.js +92 -0
  293. package/build/stories/spec/schemas/object/MoneyInput.story.js +240 -0
  294. package/build/stories/spec/schemas/oneOf/OneOfInitialisation.story.js +55 -0
  295. package/build/stories/spec/step/Controls.story.js +109 -0
  296. package/build/stories/spec/step/DFModal.story.js +58 -0
  297. package/build/stories/spec/step/Footer.story.js +70 -0
  298. package/build/stories/spec/step/Navigation.story.js +20 -0
  299. package/build/stories/spec/step/Tags.story.js +39 -0
  300. package/build/stories/spec/step/ToolBar.story.js +60 -0
  301. package/build/stories/spec/step/features/ErrorHandling.story.js +92 -0
  302. package/build/stories/spec/step/features/External.story.js +44 -0
  303. package/build/stories/spec/step/features/Polling.story.js +108 -0
  304. package/build/stories/spec/step/features/RefreshAfter.story.js +92 -0
  305. package/build/stories/spec/step/features/refresh/Refresh.story.js +258 -0
  306. package/build/stories/spec/step/features/refresh/RefreshWithPersistAsync.story.js +958 -0
  307. package/build/stories/types.js +1 -0
  308. package/build/stories/utils/fixtureHttpClient.js +70 -0
  309. package/build/stories/utils/getBasicStep.js +223 -0
  310. package/build/stories/utils/mockSearchHandler.js +71 -0
  311. package/build/stories/utils/render-utils.js +41 -0
  312. package/build/stories/visual-tests/layouts/NotUsingListItem.story.js +17 -0
  313. package/build/test-utils/DynamicFlowWise.js +32 -0
  314. package/build/test-utils/DynamicFlowWiseModal.js +34 -0
  315. package/build/test-utils/NeptuneProviders.js +11 -0
  316. package/build/test-utils/component-utils.js +5 -0
  317. package/build/test-utils/fetch-utils.js +45 -0
  318. package/build/test-utils/getMergedTestRenderers.js +34 -0
  319. package/build/test-utils/getRandomId.js +1 -0
  320. package/build/test-utils/index.js +3 -0
  321. package/build/test-utils/rtl-utils.js +7 -0
  322. package/build/test-utils/step-utils.js +6 -0
  323. package/build/test-utils/wait.js +3 -0
  324. package/build/tests/AlertLayout.test.js +78 -0
  325. package/build/tests/ArrayTuple.test.js +118 -0
  326. package/build/tests/ButtonLayout.test.js +308 -0
  327. package/build/tests/ConstLayout.test.js +95 -0
  328. package/build/tests/DateInput.test.js +163 -0
  329. package/build/tests/DecisionLayout.test.js +146 -0
  330. package/build/tests/DynamicFlow.test.js +147 -0
  331. package/build/tests/External.test.js +169 -0
  332. package/build/tests/Flow.test.js +328 -0
  333. package/build/tests/FormLayout.test.js +28 -0
  334. package/build/tests/FormattedValue.test.js +107 -0
  335. package/build/tests/ImageRenderer.test.js +78 -0
  336. package/build/tests/InitialAction.test.js +179 -0
  337. package/build/tests/InitialStep.test.js +168 -0
  338. package/build/tests/InstructionsLayout.test.js +45 -0
  339. package/build/tests/ListLayout.test.js +145 -0
  340. package/build/tests/Logging.test.js +53 -0
  341. package/build/tests/ModalBehavior.test.js +149 -0
  342. package/build/tests/MoneyInput.test.js +316 -0
  343. package/build/tests/MultiUpload.test.js +293 -0
  344. package/build/tests/NativeBack.test.js +267 -0
  345. package/build/tests/OneOfInitialisation.test.js +571 -0
  346. package/build/tests/PersistAsync.test.js +653 -0
  347. package/build/tests/Polling.test.js +617 -0
  348. package/build/tests/Prefetching.test.js +230 -0
  349. package/build/tests/RefreshAfter.test.js +63 -0
  350. package/build/tests/RefreshOnChange.ResponseHandling.test.js +205 -0
  351. package/build/tests/RefreshOnChange.test.js +233 -0
  352. package/build/tests/RefreshOnChange.with.Segmented.test.js +348 -0
  353. package/build/tests/RefreshOnChange.with.Tabs.test.js +358 -0
  354. package/build/tests/RefreshOnChangePreserve.test.js +224 -0
  355. package/build/tests/RendererProps.test.js +342 -0
  356. package/build/tests/Repeatable.test.js +107 -0
  357. package/build/tests/Rerendering.test.js +67 -0
  358. package/build/tests/ReviewLayout.test.js +274 -0
  359. package/build/tests/SchemaOnChange.test.js +133 -0
  360. package/build/tests/ScrollToError.test.js +217 -0
  361. package/build/tests/SegmentedControl.test.js +49 -0
  362. package/build/tests/SingleFileUpload.test.js +88 -0
  363. package/build/tests/StatusList.test.js +85 -0
  364. package/build/tests/Subflow.test.js +710 -0
  365. package/build/tests/Submission.ResponseHandling.test.js +557 -0
  366. package/build/tests/Submission.merging.test.js +202 -0
  367. package/build/tests/Submission.test.js +523 -0
  368. package/build/tests/Tags.test.js +475 -0
  369. package/build/tests/Upsell.test.js +126 -0
  370. package/build/tests/ValidationAsync.test.js +295 -0
  371. package/build/tests/legacy/Actions.test.js +158 -0
  372. package/build/tests/legacy/BackButton.test.js +114 -0
  373. package/build/tests/legacy/HiddenSchemas.test.js +246 -0
  374. package/build/tests/legacy/MultiSelect.test.js +497 -0
  375. package/build/tests/legacy/MultipleFileUploadSchema.test.js +341 -0
  376. package/build/tests/legacy/PersistAsync.blob-schema.test.js +224 -0
  377. package/build/tests/legacy/PersistAsync.string-schema.test.js +211 -0
  378. package/build/tests/legacy/RefreshStepOnChange.debouncing.test.js +209 -0
  379. package/build/tests/legacy/RefreshStepOnChange.test.js +424 -0
  380. package/build/tests/legacy/Search.test.js +437 -0
  381. package/build/tests/renderers/MultiSelectInputRendererProps.test.js +58 -0
  382. package/build/tests/renderers/SelectInputRendererProps.test.js +42 -0
  383. package/build/tests/renderers/TextInputRenderer.test.js +51 -0
  384. package/build/types/domain/components/UpsellComponent.d.ts +2 -3
  385. package/build/types/domain/components/UpsellComponent.d.ts.map +1 -1
  386. package/build/types/domain/components/searchComponent/SearchComponent.d.ts +5 -4
  387. package/build/types/domain/components/searchComponent/SearchComponent.d.ts.map +1 -1
  388. package/build/types/domain/features/search/getPerformSearchFunction.d.ts +1 -1
  389. package/build/types/domain/features/search/getPerformSearchFunction.d.ts.map +1 -1
  390. package/build/types/domain/mappers/layout/searchLayoutToComponent.d.ts +2 -1
  391. package/build/types/domain/mappers/layout/searchLayoutToComponent.d.ts.map +1 -1
  392. package/build/types/domain/mappers/layout/upsellLayoutToComponent.d.ts +1 -1
  393. package/build/types/domain/mappers/layout/upsellLayoutToComponent.d.ts.map +1 -1
  394. package/build/types/renderers/EmptyLoadingStateRenderer.d.ts +3 -0
  395. package/build/types/renderers/EmptyLoadingStateRenderer.d.ts.map +1 -0
  396. package/build/types/renderers/mappers/rootComponentToProps.d.ts.map +1 -1
  397. package/build/types/renderers/mappers/searchComponentToProps.d.ts +1 -1
  398. package/build/types/renderers/mappers/searchComponentToProps.d.ts.map +1 -1
  399. package/build/types/renderers/mappers/upsellComponentToProps.d.ts +2 -2
  400. package/build/types/renderers/mappers/upsellComponentToProps.d.ts.map +1 -1
  401. package/build/types/test-utils/getMergedTestRenderers.d.ts +1 -1
  402. package/build/types/test-utils/getMergedTestRenderers.d.ts.map +1 -1
  403. package/build/types/useDynamicFlow.d.ts.map +1 -1
  404. package/build/types.js +1 -0
  405. package/build/useDynamicFlow.js +104 -0
  406. package/build/useDynamicFlowModal.js +58 -0
  407. package/build/utils/analyse-step.js +14 -0
  408. package/build/utils/component-utils.js +8 -0
  409. package/build/utils/component-utils.test.js +113 -0
  410. package/build/utils/getScrollToTop.js +12 -0
  411. package/build/utils/normalise-flow-id.js +1 -0
  412. package/build/utils/normalise-flow-id.test.js +24 -0
  413. package/build/utils/openLinkInNewTab.js +10 -0
  414. package/build/utils/recursiveMerge.js +40 -0
  415. package/build/utils/recursiveMerge.test.js +93 -0
  416. package/build/utils/type-utils.js +21 -0
  417. package/build/utils/type-validators.js +11 -0
  418. package/build/utils/type-validators.test.js +180 -0
  419. package/build/utils/useStableCallback.js +15 -0
  420. package/package.json +4 -4
@@ -0,0 +1,149 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { renderWithProviders, respondWith } from '../test-utils';
3
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
4
+ import { screen, waitFor } from '@testing-library/react';
5
+ import userEvent from '@testing-library/user-event';
6
+ import { vi } from 'vitest';
7
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
8
+ describe('Modal Behavior', () => {
9
+ const step = {
10
+ id: 'step-id',
11
+ title: 'Modal Behavior',
12
+ schemas: [],
13
+ layout: [
14
+ {
15
+ type: 'review',
16
+ fields: [],
17
+ title: 'Review Layout',
18
+ callToAction: {
19
+ title: 'Open Modal',
20
+ behavior: {
21
+ type: 'modal',
22
+ content: [{ type: 'paragraph', text: 'This is a modal from a review' }],
23
+ },
24
+ },
25
+ },
26
+ {
27
+ type: 'button',
28
+ title: 'Submit',
29
+ behavior: {
30
+ type: 'modal',
31
+ content: [
32
+ {
33
+ type: 'paragraph',
34
+ text: 'This is paragraph body',
35
+ },
36
+ {
37
+ type: 'button',
38
+ title: 'Submit 2',
39
+ behavior: {
40
+ type: 'modal',
41
+ title: 'Second Modal Title',
42
+ content: [
43
+ {
44
+ type: 'button',
45
+ title: 'Dismiss second modal',
46
+ behavior: {
47
+ type: 'dismiss',
48
+ },
49
+ },
50
+ {
51
+ type: 'button',
52
+ title: 'Action',
53
+ behavior: {
54
+ type: 'action',
55
+ action: {
56
+ url: '/action',
57
+ },
58
+ },
59
+ },
60
+ {
61
+ type: 'button',
62
+ title: 'Refresh',
63
+ behavior: {
64
+ type: 'refresh',
65
+ },
66
+ },
67
+ ],
68
+ },
69
+ },
70
+ {
71
+ type: 'button',
72
+ title: 'Dismiss',
73
+ behavior: {
74
+ type: 'dismiss',
75
+ },
76
+ },
77
+ ],
78
+ title: 'This is the modal title',
79
+ },
80
+ },
81
+ ],
82
+ };
83
+ describe('given a button with a modal behavior', () => {
84
+ it('clicking the button opens the modal', async () => {
85
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "id", initialStep: step, httpClient: vi.fn(), onCompletion: vi.fn(), onError: vi.fn() }));
86
+ expect(screen.getByText('Modal Behavior')).toBeInTheDocument();
87
+ await user.click(screen.getByText('Submit'));
88
+ expect(screen.getByText('This is the modal title')).toBeInTheDocument();
89
+ });
90
+ describe('given a button with modal behavior inside that modal', () => {
91
+ it('clicking the button opens the modal', async () => {
92
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "id", initialStep: step, httpClient: vi.fn(), onCompletion: vi.fn(), onError: vi.fn() }));
93
+ expect(screen.getByText('Modal Behavior')).toBeInTheDocument();
94
+ await user.click(screen.getByText('Submit'));
95
+ expect(screen.getByText('This is the modal title')).toBeInTheDocument();
96
+ await user.click(screen.getByText('Submit 2'));
97
+ expect(screen.getByText('Second Modal Title')).toBeInTheDocument();
98
+ });
99
+ });
100
+ });
101
+ describe('when a button with dismiss behavior is clicked from inside a modal', () => {
102
+ it('closees the topmost modal', async () => {
103
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "id", initialStep: step, httpClient: vi.fn(), onCompletion: vi.fn(), onError: vi.fn() }));
104
+ expect(screen.getByText('Modal Behavior')).toBeInTheDocument();
105
+ await user.click(screen.getByText('Submit'));
106
+ expect(screen.getByText('This is the modal title')).toBeInTheDocument();
107
+ await user.click(screen.getByRole('button', { name: 'Dismiss' }));
108
+ await waitFor(() => {
109
+ expect(screen.queryByText('This is the modal title')).not.toBeInTheDocument();
110
+ });
111
+ });
112
+ });
113
+ describe('when a button with action behavior is clicked from inside a modal', () => {
114
+ it('closes all modals', async () => {
115
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "id", initialStep: step, httpClient: vi.fn(), onCompletion: vi.fn(), onError: vi.fn() }));
116
+ expect(screen.getByText('Modal Behavior')).toBeInTheDocument();
117
+ await user.click(screen.getByText('Submit'));
118
+ await user.click(screen.getByText('Submit 2'));
119
+ await user.click(screen.getByRole('button', { name: 'Action' }));
120
+ await waitFor(() => {
121
+ expect(screen.queryByText('This is the modal title')).not.toBeInTheDocument();
122
+ });
123
+ expect(screen.queryByText('Second modal title')).not.toBeInTheDocument();
124
+ });
125
+ });
126
+ describe('when a button with refresh behavior is clicked from inside a modal', () => {
127
+ it('closes all modals (even when it returns a 304 Not Modified)', async () => {
128
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "id", initialStep: step, httpClient: vi.fn().mockImplementation(async () => respondWith(null, { status: 304 })), onCompletion: vi.fn(), onError: vi.fn() }));
129
+ expect(screen.getByText('Modal Behavior')).toBeInTheDocument();
130
+ await user.click(screen.getByText('Submit'));
131
+ await user.click(screen.getByText('Submit 2'));
132
+ await user.click(screen.getByRole('button', { name: 'Refresh' }));
133
+ await waitFor(() => {
134
+ expect(screen.queryByText('This is the modal title')).not.toBeInTheDocument();
135
+ });
136
+ expect(screen.queryByText('Second modal title')).not.toBeInTheDocument();
137
+ });
138
+ });
139
+ describe('when a review CTA with modal behavior is clicked', () => {
140
+ it('opens the modal', async () => {
141
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "id", initialStep: step, httpClient: vi.fn(), onCompletion: vi.fn(), onError: vi.fn() }));
142
+ expect(screen.getByText('Modal Behavior')).toBeInTheDocument();
143
+ await user.click(screen.getByText('Open Modal'));
144
+ await waitFor(() => {
145
+ expect(screen.getByText('This is a modal from a review')).toBeInTheDocument();
146
+ });
147
+ });
148
+ });
149
+ });
@@ -0,0 +1,316 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { screen, waitFor } from '@testing-library/react';
3
+ import { userEvent } from '@testing-library/user-event';
4
+ import { act } from 'react';
5
+ import { renderWithProviders } from '../test-utils';
6
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
7
+ import { vi } from 'vitest';
8
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
9
+ describe('MoneyInput component', () => {
10
+ const getDefaultProps = () => ({
11
+ flowId: 'flow-id',
12
+ httpClient: vi.fn(),
13
+ onCompletion: vi.fn(),
14
+ onError: vi.fn(),
15
+ onEvent: vi.fn(),
16
+ onLog: vi.fn(),
17
+ });
18
+ const getInitialStep = (overrides) => {
19
+ var _a;
20
+ return ({
21
+ id: 'step',
22
+ title: 'Money Input',
23
+ description: 'A money input with amount and currency',
24
+ layout: [
25
+ {
26
+ type: 'form',
27
+ schemaId: '#schema',
28
+ },
29
+ {
30
+ type: 'button',
31
+ title: 'Submit',
32
+ control: 'primary',
33
+ action: {
34
+ url: '/submit',
35
+ },
36
+ },
37
+ ],
38
+ schemas: [
39
+ {
40
+ $id: '#schema',
41
+ type: 'object',
42
+ title: 'Money Input (object schema title)',
43
+ description: 'A money input with amount and currency (object schema description)',
44
+ format: 'money',
45
+ properties: Object.assign({ currency: Object.assign({ oneOf: [
46
+ {
47
+ const: 'USD',
48
+ title: 'USD',
49
+ description: 'US Dollar',
50
+ },
51
+ {
52
+ const: 'EUR',
53
+ title: 'EUR',
54
+ description: 'Euro',
55
+ },
56
+ {
57
+ const: 'GBP',
58
+ title: 'GBP',
59
+ description: 'British Pound',
60
+ },
61
+ ] }, overrides === null || overrides === void 0 ? void 0 : overrides.currencySchemaOverrides), amount: Object.assign({ type: 'string', format: 'numeric', validationMessages: { required: 'Please enter an amount.' } }, overrides === null || overrides === void 0 ? void 0 : overrides.amountSchemaOverrides) }, overrides === null || overrides === void 0 ? void 0 : overrides.additionalProperties),
62
+ displayOrder: ['currency', 'amount', ...Object.keys((_a = overrides === null || overrides === void 0 ? void 0 : overrides.additionalProperties) !== null && _a !== void 0 ? _a : [])],
63
+ required: ['currency', 'amount'],
64
+ },
65
+ ],
66
+ refreshUrl: '/refresh',
67
+ });
68
+ };
69
+ describe('when the model contains an amount', () => {
70
+ describe('and the amount is valid', () => {
71
+ const cases = ['123', '10.1', '1', '0', '0.1', '000.000001', '.123', '123.'];
72
+ test.each(cases)('should initialise the component with amount: %s', async (amount) => {
73
+ const props = getDefaultProps();
74
+ const initialStep = getInitialStep();
75
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: Object.assign(Object.assign({}, initialStep), { model: { amount } }) })));
76
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
77
+ expect(props.httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({
78
+ body: JSON.stringify({ amount, currency: 'USD' }),
79
+ }));
80
+ });
81
+ });
82
+ describe('and the amount is invalid (contains letters, thousand separator, multiple decimals, is negative)', () => {
83
+ const cases = ['abc', '123abc', '123,456', '1.2.3', '-1'];
84
+ test.each(cases)('should show validation error for amount: %s', async (amount) => {
85
+ const props = getDefaultProps();
86
+ const initialStep = getInitialStep();
87
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: Object.assign(Object.assign({}, initialStep), { model: { amount } }) })));
88
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
89
+ expect(screen.getByText('Please enter an amount.')).toBeInTheDocument();
90
+ expect(props.httpClient).not.toHaveBeenCalled();
91
+ });
92
+ });
93
+ });
94
+ describe('When the "amount" schema is required', () => {
95
+ it('should show validation error when submitting empty form', async () => {
96
+ const props = getDefaultProps();
97
+ const initialStep = getInitialStep();
98
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
99
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
100
+ await waitFor(() => {
101
+ expect(screen.getByText('Please enter an amount.')).toBeInTheDocument();
102
+ });
103
+ });
104
+ it('should show validation error when a value is cleared and then submitted', async () => {
105
+ const props = getDefaultProps();
106
+ const initialStep = Object.assign(Object.assign({}, getInitialStep()), { model: { amount: '50', currency: 'USD' } });
107
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
108
+ const amountInput = screen.getByRole('textbox', {
109
+ name: 'Money Input (object schema title)',
110
+ });
111
+ await user.clear(amountInput);
112
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
113
+ await waitFor(() => {
114
+ expect(screen.getByText('Please enter an amount.')).toBeInTheDocument();
115
+ });
116
+ });
117
+ it('should accept 0 as a valid amount', async () => {
118
+ const props = getDefaultProps();
119
+ const initialStep = getInitialStep();
120
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
121
+ const amountInput = screen.getByRole('textbox', {
122
+ name: 'Money Input (object schema title)',
123
+ });
124
+ await user.type(amountInput, '0');
125
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
126
+ expect(props.httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({
127
+ body: JSON.stringify({ amount: '0', currency: 'USD' }),
128
+ }));
129
+ });
130
+ });
131
+ describe('When the "amount" schema has a "minimum" constraint', () => {
132
+ it('should show validation error when submitting an amount below the minimum', async () => {
133
+ const props = getDefaultProps();
134
+ const initialStep = getInitialStep({
135
+ amountSchemaOverrides: {
136
+ type: 'string',
137
+ minimum: '4.25',
138
+ validationMessages: { minimum: 'Amount must be at least 4.25.' },
139
+ },
140
+ });
141
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
142
+ const amountInput = screen.getByRole('textbox', {
143
+ name: 'Money Input (object schema title)',
144
+ });
145
+ await user.type(amountInput, '4.10');
146
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
147
+ await waitFor(() => {
148
+ expect(screen.getByText('Amount must be at least 4.25.')).toBeInTheDocument();
149
+ });
150
+ });
151
+ it('should show validation error when submitting an amount above the maximum', async () => {
152
+ const props = getDefaultProps();
153
+ const initialStep = getInitialStep({
154
+ amountSchemaOverrides: {
155
+ type: 'string',
156
+ maximum: '99.99',
157
+ validationMessages: { maximum: 'Amount must be at most 99.99.' },
158
+ },
159
+ });
160
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
161
+ const amountInput = screen.getByRole('textbox', {
162
+ name: 'Money Input (object schema title)',
163
+ });
164
+ await user.type(amountInput, '100');
165
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
166
+ await waitFor(() => {
167
+ expect(screen.getByText('Amount must be at most 99.99.')).toBeInTheDocument();
168
+ });
169
+ });
170
+ });
171
+ describe('When the "currency" schema has onChange behavior', () => {
172
+ it('should trigger the specified behavior when changing currency', async () => {
173
+ const props = getDefaultProps();
174
+ const initialStep = getInitialStep({
175
+ currencySchemaOverrides: { onChange: { type: 'refresh' } },
176
+ });
177
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
178
+ await user.click(screen.getByRole('combobox', { name: 'Select currency' }));
179
+ await user.click(screen.getByRole('option', { name: 'EUR Euro' }));
180
+ await waitFor(() => {
181
+ expect(props.httpClient).toHaveBeenCalledWith('/refresh', expect.any(Object));
182
+ });
183
+ });
184
+ });
185
+ describe('When the "amount" schema has onChange behavior', () => {
186
+ it('should trigger the specified behavior when changing amount (debounced)', async () => {
187
+ const localUser = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
188
+ const props = getDefaultProps();
189
+ const initialStep = getInitialStep({
190
+ amountSchemaOverrides: { onChange: { type: 'refresh' } },
191
+ });
192
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
193
+ const amountInput = screen.getByRole('textbox', {
194
+ name: 'Money Input (object schema title)',
195
+ });
196
+ await localUser.type(amountInput, '123.45');
197
+ await act(async () => {
198
+ vi.runAllTimers();
199
+ });
200
+ await waitFor(() => {
201
+ expect(props.httpClient).toHaveBeenCalledWith('/refresh', expect.any(Object));
202
+ });
203
+ });
204
+ });
205
+ describe('when the object schema contains other const or hidden properties', () => {
206
+ const props = getDefaultProps();
207
+ const initialStep = Object.assign(Object.assign({}, getInitialStep({
208
+ additionalProperties: {
209
+ hiddenProp: { type: 'string', hidden: true, const: 'hiddenValue' },
210
+ constProp: { const: 'constValue' },
211
+ },
212
+ })), { model: { amount: '99.99', currency: 'USD' } });
213
+ it('should still render the money input correctly', async () => {
214
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
215
+ expect(screen.getByRole('textbox', {
216
+ name: 'Money Input (object schema title)',
217
+ })).toBeInTheDocument();
218
+ });
219
+ it('should submit the correct model including the const and hidden properties', async () => {
220
+ const httpClient = vi.fn();
221
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep, httpClient: httpClient })));
222
+ await user.click(screen.getByRole('button', { name: 'Submit' }));
223
+ expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({
224
+ body: JSON.stringify({
225
+ amount: '99.99',
226
+ currency: 'USD',
227
+ hiddenProp: 'hiddenValue',
228
+ constProp: 'constValue',
229
+ }),
230
+ }));
231
+ });
232
+ });
233
+ describe('Grouped currencies', () => {
234
+ describe('when grouping tags are NOT present in currencies', () => {
235
+ it('should render currencies without grouping', async () => {
236
+ const props = getDefaultProps();
237
+ const initialStep = getInitialStep();
238
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
239
+ await user.click(screen.getByRole('combobox', { name: 'Select currency' }));
240
+ await waitFor(() => {
241
+ expect(screen.getByRole('option', { name: 'USD US Dollar' })).toBeInTheDocument();
242
+ });
243
+ expect(screen.queryByText('All')).not.toBeInTheDocument();
244
+ });
245
+ });
246
+ describe('when grouping tags are present in currencies', () => {
247
+ it('should render currencies grouped by known tags', async () => {
248
+ const props = getDefaultProps();
249
+ const initialStep = getInitialStep();
250
+ const { oneOf } = initialStep.schemas[0].properties
251
+ .currency;
252
+ // Add some grouping tags to the currency options
253
+ oneOf.forEach((option, index) => {
254
+ const constSchemaOption = option;
255
+ constSchemaOption.tags = [];
256
+ if (index % 2 === 0) {
257
+ constSchemaOption.tags.push('popular');
258
+ }
259
+ if (index % 4 === 0) {
260
+ constSchemaOption.tags.push('recent');
261
+ }
262
+ });
263
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
264
+ await user.click(screen.getByRole('combobox', { name: 'Select currency' }));
265
+ await waitFor(() => {
266
+ expect(screen.getByText('All')).toBeInTheDocument();
267
+ });
268
+ expect(screen.getByText('Popular')).toBeInTheDocument();
269
+ expect(screen.getByText('Recent')).toBeInTheDocument();
270
+ });
271
+ });
272
+ });
273
+ describe('When currency codes are not string values', () => {
274
+ it('should throw an error during rendering', async () => {
275
+ const props = getDefaultProps();
276
+ const initialStep = getInitialStep({
277
+ currencySchemaOverrides: { oneOf: [{ const: 123 }, { const: 456 }] },
278
+ });
279
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
280
+ expect(props.onError).toHaveBeenCalledWith(expect.any(Error));
281
+ expect(props.onError.mock.calls[0][0].message).toMatch(/only supports string currency codes/i);
282
+ });
283
+ });
284
+ describe('When the step contains validation errors', () => {
285
+ describe('for the amount schema', () => {
286
+ it('should display the validation errors on the MoneyInput component', async () => {
287
+ const props = getDefaultProps();
288
+ const initialStep = Object.assign(Object.assign({}, getInitialStep()), { errors: { validation: { amount: 'Amount must be greater than zero.' } } });
289
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
290
+ await waitFor(() => {
291
+ expect(screen.getByText('Amount must be greater than zero.')).toBeInTheDocument();
292
+ });
293
+ });
294
+ });
295
+ describe('for the currency schema', () => {
296
+ it('should display the validation errors on the MoneyInput component', async () => {
297
+ const props = getDefaultProps();
298
+ const initialStep = Object.assign(Object.assign({}, getInitialStep()), { errors: { validation: { currency: 'Please select a better currency.' } }, model: { amount: '50', currency: 'USD' } });
299
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
300
+ await waitFor(() => {
301
+ expect(screen.getByText('Please select a better currency.')).toBeInTheDocument();
302
+ });
303
+ });
304
+ });
305
+ describe('for the object schema', () => {
306
+ it('should display the validation errors on the MoneyInput component', async () => {
307
+ const props = getDefaultProps();
308
+ const initialStep = Object.assign(Object.assign({}, getInitialStep()), { errors: { validation: 'The money input is invalid.' }, model: { amount: '50', currency: 'USD' } });
309
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { initialStep: initialStep })));
310
+ await waitFor(() => {
311
+ expect(screen.getByText('The money input is invalid.')).toBeInTheDocument();
312
+ });
313
+ });
314
+ });
315
+ });
316
+ });