@wise/dynamic-flow-client 5.12.0 → 5.12.1

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 (405) 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 +373 -0
  21. package/build/controller/executePoll.js +64 -0
  22. package/build/controller/executeRefresh.js +45 -0
  23. package/build/controller/executeRequest.js +80 -0
  24. package/build/controller/executeSubmission.js +70 -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 +24 -0
  79. package/build/domain/components/step/StepDomainComponent.js +78 -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 +15 -0
  154. package/build/domain/mappers/mapStepToComponent.js +133 -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 +66 -0
  175. package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.test.js +225 -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/renderers/CoreContainerRenderer.js +5 -0
  208. package/build/renderers/CoreRootRenderer.js +12 -0
  209. package/build/renderers/EmptyLoadingStateRenderer.js +5 -0
  210. package/build/renderers/getRenderFunction.js +24 -0
  211. package/build/renderers/getSchemaErrorMessageFunction.js +97 -0
  212. package/build/renderers/mappers/alertComponentToProps.js +2 -0
  213. package/build/renderers/mappers/allOfComponentToProps.js +6 -0
  214. package/build/renderers/mappers/booleanInputComponentToProps.js +5 -0
  215. package/build/renderers/mappers/boxComponentToProps.js +13 -0
  216. package/build/renderers/mappers/buttonComponentToProps.js +4 -0
  217. package/build/renderers/mappers/columnsComponentToProps.js +11 -0
  218. package/build/renderers/mappers/componentToRendererProps.js +164 -0
  219. package/build/renderers/mappers/constComponentToProps.js +5 -0
  220. package/build/renderers/mappers/containerComponentToProps.js +7 -0
  221. package/build/renderers/mappers/dateInputComponentToProps.js +2 -0
  222. package/build/renderers/mappers/decisionComponentToProps.js +16 -0
  223. package/build/renderers/mappers/dividerComponentToProps.js +2 -0
  224. package/build/renderers/mappers/externalComponentToProps.js +3 -0
  225. package/build/renderers/mappers/formComponentToProps.js +12 -0
  226. package/build/renderers/mappers/formattedValueComponentToProps.js +5 -0
  227. package/build/renderers/mappers/headingComponentToProps.js +2 -0
  228. package/build/renderers/mappers/hiddenComponentToProps.js +4 -0
  229. package/build/renderers/mappers/imageComponentToProps.js +2 -0
  230. package/build/renderers/mappers/instructionsComponentToProps.js +2 -0
  231. package/build/renderers/mappers/integerInputComponentToProps.js +2 -0
  232. package/build/renderers/mappers/listComponentToProps.js +2 -0
  233. package/build/renderers/mappers/loadingIndicatorComponentToProps.js +2 -0
  234. package/build/renderers/mappers/markdownComponentToProps.js +2 -0
  235. package/build/renderers/mappers/mediaComponentToProps.js +2 -0
  236. package/build/renderers/mappers/modalComponentToProps.js +11 -0
  237. package/build/renderers/mappers/modalLayoutComponentToProps.js +16 -0
  238. package/build/renderers/mappers/moneyInputComponentToProps.js +36 -0
  239. package/build/renderers/mappers/multiSelectComponentToProps.js +23 -0
  240. package/build/renderers/mappers/multiUploadInputComponentToProps.js +12 -0
  241. package/build/renderers/mappers/numberInputComponentToProps.js +2 -0
  242. package/build/renderers/mappers/objectComponentToProps.js +8 -0
  243. package/build/renderers/mappers/paragraphComponentToProps.js +2 -0
  244. package/build/renderers/mappers/persistAsyncComponentToProps.js +8 -0
  245. package/build/renderers/mappers/progressComponentToProps.js +2 -0
  246. package/build/renderers/mappers/repeatableComponentToProps.js +30 -0
  247. package/build/renderers/mappers/reviewComponentToProps.js +2 -0
  248. package/build/renderers/mappers/rootComponentToProps.js +21 -0
  249. package/build/renderers/mappers/searchComponentToProps.js +57 -0
  250. package/build/renderers/mappers/sectionComponentToProps.js +6 -0
  251. package/build/renderers/mappers/selectInputComponentToProps.js +34 -0
  252. package/build/renderers/mappers/statusListComponentToProps.js +2 -0
  253. package/build/renderers/mappers/subflowComponentToRendererProps.js +4 -0
  254. package/build/renderers/mappers/tabsComponentToProps.js +14 -0
  255. package/build/renderers/mappers/textInputComponentToProps.js +2 -0
  256. package/build/renderers/mappers/tupleComponentToProps.js +8 -0
  257. package/build/renderers/mappers/uploadInputComponentToProps.js +8 -0
  258. package/build/renderers/mappers/upsellComponentToProps.js +2 -0
  259. package/build/renderers/mappers/utils/getValidationState.js +12 -0
  260. package/build/renderers/mappers/utils/inputComponentToProps.js +26 -0
  261. package/build/renderers/mappers/utils/mapErrorsToValidationState.js +9 -0
  262. package/build/renderers/mappers/utils/pick.js +8 -0
  263. package/build/renderers/mappers/utils/selectInputOptionsToProps.js +11 -0
  264. package/build/renderers/stepComponentToProps.js +32 -0
  265. package/build/renderers/utils.js +69 -0
  266. package/build/renderers/utils.test.js +70 -0
  267. package/build/stories/dev-tools/ContainerQueries.story.js +66 -0
  268. package/build/stories/dev-tools/Debugger.story.js +38 -0
  269. package/build/stories/dev-tools/FixtureSelect.story.js +23 -0
  270. package/build/stories/dev-tools/TestServer.story.js +32 -0
  271. package/build/stories/examples/NativeFlow.story.js +80 -0
  272. package/build/stories/examples/Recipients.story.js +568 -0
  273. package/build/stories/spec/behavior/Copy.story.js +71 -0
  274. package/build/stories/spec/behavior/Link.story.js +40 -0
  275. package/build/stories/spec/behavior/Modal.story.js +79 -0
  276. package/build/stories/spec/behavior/Subflow.story.js +267 -0
  277. package/build/stories/spec/layouts/Decision.story.js +241 -0
  278. package/build/stories/spec/layouts/Image.story.js +42 -0
  279. package/build/stories/spec/layouts/Modal.story.js +81 -0
  280. package/build/stories/spec/layouts/Search.story.js +325 -0
  281. package/build/stories/spec/layouts/Upsell.story.js +55 -0
  282. package/build/stories/spec/layouts/button/Button.story.js +100 -0
  283. package/build/stories/spec/layouts/button/PinnedButton.story.js +81 -0
  284. package/build/stories/spec/response/ActionResponse.story.js +66 -0
  285. package/build/stories/spec/schemas/MultiSelect.story.js +148 -0
  286. package/build/stories/spec/schemas/Upload.story.js +168 -0
  287. package/build/stories/spec/schemas/const/ConstLayout.story.js +159 -0
  288. package/build/stories/spec/schemas/const/ObjectConst.story.js +94 -0
  289. package/build/stories/spec/schemas/features/PersistAsync.story.js +211 -0
  290. package/build/stories/spec/schemas/features/ValidationAsync.story.js +103 -0
  291. package/build/stories/spec/schemas/object/FormattedValue.story.js +92 -0
  292. package/build/stories/spec/schemas/object/MoneyInput.story.js +240 -0
  293. package/build/stories/spec/schemas/oneOf/OneOfInitialisation.story.js +55 -0
  294. package/build/stories/spec/schemas/oneOf/OneOfWithSingleOption.story.js +114 -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/ScrollToBottom.story.js +103 -0
  300. package/build/stories/spec/step/Tags.story.js +39 -0
  301. package/build/stories/spec/step/ToolBar.story.js +60 -0
  302. package/build/stories/spec/step/features/ErrorHandling.story.js +92 -0
  303. package/build/stories/spec/step/features/External.story.js +91 -0
  304. package/build/stories/spec/step/features/Polling.story.js +108 -0
  305. package/build/stories/spec/step/features/RefreshAfter.story.js +92 -0
  306. package/build/stories/spec/step/features/refresh/Refresh.story.js +258 -0
  307. package/build/stories/spec/step/features/refresh/RefreshWithPersistAsync.story.js +958 -0
  308. package/build/stories/types.js +1 -0
  309. package/build/stories/utils/fixtureHttpClient.js +70 -0
  310. package/build/stories/utils/getBasicStep.js +223 -0
  311. package/build/stories/utils/mockSearchHandler.js +71 -0
  312. package/build/stories/utils/render-utils.js +41 -0
  313. package/build/stories/visual-tests/layouts/NotUsingListItem.story.js +17 -0
  314. package/build/test-utils/DynamicFlowWise.js +32 -0
  315. package/build/test-utils/DynamicFlowWiseModal.js +34 -0
  316. package/build/test-utils/NeptuneProviders.js +11 -0
  317. package/build/test-utils/component-utils.js +5 -0
  318. package/build/test-utils/fetch-utils.js +45 -0
  319. package/build/test-utils/getMergedTestRenderers.js +34 -0
  320. package/build/test-utils/getRandomId.js +1 -0
  321. package/build/test-utils/index.js +3 -0
  322. package/build/test-utils/openLinkInNewTab.js +15 -0
  323. package/build/test-utils/rtl-utils.js +7 -0
  324. package/build/test-utils/step-utils.js +6 -0
  325. package/build/test-utils/wait.js +3 -0
  326. package/build/tests/AlertLayout.test.js +78 -0
  327. package/build/tests/ArrayTuple.test.js +118 -0
  328. package/build/tests/ButtonLayout.test.js +308 -0
  329. package/build/tests/ConstLayout.test.js +95 -0
  330. package/build/tests/DateInput.test.js +163 -0
  331. package/build/tests/DecisionLayout.test.js +146 -0
  332. package/build/tests/DynamicFlow.test.js +147 -0
  333. package/build/tests/External.test.js +169 -0
  334. package/build/tests/Flow.test.js +328 -0
  335. package/build/tests/FormLayout.test.js +28 -0
  336. package/build/tests/FormattedValue.test.js +107 -0
  337. package/build/tests/ImageRenderer.test.js +78 -0
  338. package/build/tests/InitialAction.test.js +179 -0
  339. package/build/tests/InitialStep.test.js +168 -0
  340. package/build/tests/InstructionsLayout.test.js +45 -0
  341. package/build/tests/ListLayout.test.js +168 -0
  342. package/build/tests/Logging.test.js +53 -0
  343. package/build/tests/ModalBehavior.test.js +149 -0
  344. package/build/tests/MoneyInput.test.js +316 -0
  345. package/build/tests/MultiUpload.test.js +293 -0
  346. package/build/tests/NativeBack.test.js +267 -0
  347. package/build/tests/NoOp.test.js +194 -0
  348. package/build/tests/OneOfInitialisation.test.js +571 -0
  349. package/build/tests/PersistAsync.test.js +687 -0
  350. package/build/tests/Polling.test.js +702 -0
  351. package/build/tests/Prefetching.test.js +230 -0
  352. package/build/tests/RefreshAfter.test.js +63 -0
  353. package/build/tests/RefreshOnChange.ResponseHandling.test.js +205 -0
  354. package/build/tests/RefreshOnChange.test.js +233 -0
  355. package/build/tests/RefreshOnChange.with.Segmented.test.js +348 -0
  356. package/build/tests/RefreshOnChange.with.Tabs.test.js +358 -0
  357. package/build/tests/RefreshOnChangePreserve.test.js +224 -0
  358. package/build/tests/RendererProps.test.js +342 -0
  359. package/build/tests/Repeatable.test.js +107 -0
  360. package/build/tests/Rerendering.test.js +67 -0
  361. package/build/tests/ReviewLayout.test.js +274 -0
  362. package/build/tests/SchemaOnChange.test.js +133 -0
  363. package/build/tests/SchemaReferences.test.js +88 -0
  364. package/build/tests/ScrollToBottom.test.js +122 -0
  365. package/build/tests/ScrollToError.test.js +217 -0
  366. package/build/tests/SegmentedControl.test.js +49 -0
  367. package/build/tests/SingleFileUpload.test.js +168 -0
  368. package/build/tests/StatusList.test.js +85 -0
  369. package/build/tests/Subflow.test.js +710 -0
  370. package/build/tests/Submission.ResponseHandling.test.js +557 -0
  371. package/build/tests/Submission.merging.test.js +202 -0
  372. package/build/tests/Submission.test.js +748 -0
  373. package/build/tests/Tags.test.js +475 -0
  374. package/build/tests/Upsell.test.js +154 -0
  375. package/build/tests/ValidationAsync.test.js +295 -0
  376. package/build/tests/legacy/Actions.test.js +158 -0
  377. package/build/tests/legacy/BackButton.test.js +114 -0
  378. package/build/tests/legacy/HiddenSchemas.test.js +246 -0
  379. package/build/tests/legacy/MultiSelect.test.js +501 -0
  380. package/build/tests/legacy/MultipleFileUploadSchema.test.js +341 -0
  381. package/build/tests/legacy/PersistAsync.blob-schema.test.js +224 -0
  382. package/build/tests/legacy/PersistAsync.string-schema.test.js +211 -0
  383. package/build/tests/legacy/RefreshStepOnChange.debouncing.test.js +209 -0
  384. package/build/tests/legacy/RefreshStepOnChange.test.js +424 -0
  385. package/build/tests/legacy/Search.test.js +437 -0
  386. package/build/tests/renderers/MultiSelectInputRendererProps.test.js +58 -0
  387. package/build/tests/renderers/SelectInputRendererProps.test.js +43 -0
  388. package/build/tests/renderers/TextInputRenderer.test.js +51 -0
  389. package/build/tsconfig.types.tsbuildinfo +1 -0
  390. package/build/types.js +1 -0
  391. package/build/useDynamicFlow.js +104 -0
  392. package/build/useDynamicFlowModal.js +58 -0
  393. package/build/utils/analyse-step.js +14 -0
  394. package/build/utils/component-utils.js +8 -0
  395. package/build/utils/component-utils.test.js +113 -0
  396. package/build/utils/getScrollToTop.js +12 -0
  397. package/build/utils/normalise-flow-id.js +1 -0
  398. package/build/utils/normalise-flow-id.test.js +24 -0
  399. package/build/utils/recursiveMerge.js +40 -0
  400. package/build/utils/recursiveMerge.test.js +93 -0
  401. package/build/utils/type-utils.js +21 -0
  402. package/build/utils/type-validators.js +11 -0
  403. package/build/utils/type-validators.test.js +180 -0
  404. package/build/utils/useStableCallback.js +15 -0
  405. package/package.json +20 -21
@@ -0,0 +1,274 @@
1
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
+ /* eslint-disable @typescript-eslint/no-unsafe-assignment */
3
+ import { screen, waitFor } from '@testing-library/react';
4
+ import { userEvent } from '@testing-library/user-event';
5
+ import { Fragment } from 'react/jsx-runtime';
6
+ import { renderWithProviders } from '../test-utils';
7
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
8
+ import { vi } from 'vitest';
9
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
10
+ describe('ReviewLayout component', () => {
11
+ const getDefaultProps = () => ({
12
+ flowId: 'flow-id',
13
+ httpClient: vi.fn(),
14
+ onCompletion: vi.fn(),
15
+ onError: vi.fn(),
16
+ onEvent: vi.fn(),
17
+ onLog: vi.fn(),
18
+ });
19
+ const getInitialStep = (behavior) => ({
20
+ id: 'step-id',
21
+ title: 'Step with alert',
22
+ layout: [
23
+ {
24
+ type: 'review',
25
+ title: 'Review Title',
26
+ analyticsId: 'review-layout-aid',
27
+ fields: [
28
+ {
29
+ analyticsId: 'review-field-1-aid',
30
+ label: 'Label 1',
31
+ value: 'Value 1',
32
+ rawValue: 'Raw value 1',
33
+ tag: 'tag1',
34
+ inlineAlert: { content: 'Inline alert 1', context: 'positive' },
35
+ additionalInfo: { text: 'Additional info 1' },
36
+ callToAction: {
37
+ title: 'Call to Action 1',
38
+ behavior: { type: 'action', action: { url: '/call-to-action', method: 'GET' } },
39
+ },
40
+ },
41
+ {
42
+ analyticsId: 'review-field-2-aid',
43
+ label: 'Label 2',
44
+ value: 'Value 2',
45
+ rawValue: 'Raw value 2',
46
+ tag: 'tag2',
47
+ inlineAlert: { content: 'Inline alert 2', context: 'negative' },
48
+ additionalInfo: {
49
+ text: 'Additional info 2',
50
+ behavior: { type: 'action', action: { url: '/additional-info', method: 'GET' } },
51
+ },
52
+ },
53
+ ],
54
+ callToAction: Object.assign({ title: 'Call to Action', action: {} }, (behavior ? { behavior } : {})),
55
+ },
56
+ ],
57
+ schemas: [],
58
+ });
59
+ describe('Behavior / Call to Action', () => {
60
+ describe('Old "container" behavior type', () => {
61
+ describe('Given a review layout component with a link behavior', () => {
62
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep({ link: { url: 'https://google.com' } }) });
63
+ it('should render a link that opens the url on click', async () => {
64
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
65
+ await waitFor(() => {
66
+ expect(screen.getByRole('link')).toHaveTextContent('Call to Action');
67
+ });
68
+ expect(screen.getByRole('link')).toHaveAttribute('href', 'https://google.com');
69
+ });
70
+ });
71
+ describe('Given a review layout component with an action behavior', () => {
72
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep({ action: { url: '/action' } }) });
73
+ it('should trigger the action on click', async () => {
74
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
75
+ await waitFor(() => {
76
+ expect(screen.getByRole('button', { name: 'Call to Action' })).toBeInTheDocument();
77
+ });
78
+ await user.click(screen.getByRole('button', { name: 'Call to Action' }));
79
+ expect(props.httpClient).toHaveBeenCalledWith('/action', expect.any(Object));
80
+ });
81
+ });
82
+ });
83
+ describe('New "union type" behavior type', () => {
84
+ describe('Given a review layout component with a link behavior', () => {
85
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep({ type: 'link', url: 'https://google.com' }) });
86
+ it('should render a link that opens the url on click', async () => {
87
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
88
+ await waitFor(() => {
89
+ expect(screen.getByRole('link')).toHaveTextContent('Call to Action');
90
+ });
91
+ expect(screen.getByRole('link')).toHaveAttribute('href', 'https://google.com');
92
+ });
93
+ });
94
+ describe('Given a review layout component with an action behavior', () => {
95
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep({ type: 'action', action: { url: '/action' } }) });
96
+ it('should trigger the action on click', async () => {
97
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
98
+ await waitFor(() => {
99
+ expect(screen.getByRole('button', { name: 'Call to Action' })).toBeInTheDocument();
100
+ });
101
+ await user.click(screen.getByRole('button', { name: 'Call to Action' }));
102
+ expect(props.httpClient).toHaveBeenCalledWith('/action', expect.any(Object));
103
+ });
104
+ });
105
+ });
106
+ describe('Deprecated "action" support', () => {
107
+ describe('Given a review layout component with an action', () => {
108
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: {
109
+ id: 'step-id',
110
+ title: 'Step with alert',
111
+ layout: [
112
+ {
113
+ type: 'review',
114
+ title: 'Review Title',
115
+ fields: [],
116
+ action: { url: '/action', title: 'Old Action with Title' },
117
+ },
118
+ ],
119
+ schemas: [],
120
+ } });
121
+ it('should trigger the action on click', async () => {
122
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
123
+ await waitFor(() => {
124
+ expect(screen.getByRole('button')).toHaveTextContent('Old Action with Title');
125
+ });
126
+ await user.click(screen.getByRole('button'));
127
+ expect(props.httpClient).toHaveBeenCalledWith('/action', expect.any(Object));
128
+ });
129
+ });
130
+ });
131
+ describe('Deprecated "calltoAction" with "action" support', () => {
132
+ describe('Given a review layout component with an action', () => {
133
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: {
134
+ id: 'step-id',
135
+ title: 'Step with alert',
136
+ layout: [
137
+ {
138
+ type: 'review',
139
+ title: 'Review Title',
140
+ fields: [],
141
+ callToAction: {
142
+ title: 'Call to Action',
143
+ action: { url: '/action' },
144
+ },
145
+ },
146
+ ],
147
+ schemas: [],
148
+ } });
149
+ it('should trigger the action on click', async () => {
150
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
151
+ await waitFor(() => {
152
+ expect(screen.getByRole('button')).toHaveTextContent('Call to Action');
153
+ });
154
+ await user.click(screen.getByRole('button'));
155
+ expect(props.httpClient).toHaveBeenCalledWith('/action', expect.any(Object));
156
+ });
157
+ });
158
+ });
159
+ });
160
+ describe('Extra props', () => {
161
+ describe('Given a review layout component with extra properties in its fields', () => {
162
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep() });
163
+ it('should pass the tag to the the renderer', async () => {
164
+ const customRenderer = {
165
+ canRenderType: 'review',
166
+ render: vi.fn(),
167
+ };
168
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
169
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
170
+ fields: expect.arrayContaining([
171
+ expect.objectContaining({ tag: 'tag1' }),
172
+ expect.objectContaining({ tag: 'tag2' }),
173
+ ]),
174
+ }));
175
+ });
176
+ it('should pass the rawValue to the the renderer', async () => {
177
+ const customRenderer = {
178
+ canRenderType: 'review',
179
+ render: vi.fn(),
180
+ };
181
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
182
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
183
+ fields: expect.arrayContaining([
184
+ expect.objectContaining({ rawValue: 'Raw value 1' }),
185
+ expect.objectContaining({ rawValue: 'Raw value 2' }),
186
+ ]),
187
+ }));
188
+ });
189
+ it('should pass the inlineAlert to the the renderer', async () => {
190
+ const customRenderer = {
191
+ canRenderType: 'review',
192
+ render: vi.fn(),
193
+ };
194
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
195
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
196
+ fields: expect.arrayContaining([
197
+ expect.objectContaining({
198
+ inlineAlert: { content: 'Inline alert 1', context: 'positive' },
199
+ }),
200
+ expect.objectContaining({
201
+ inlineAlert: { content: 'Inline alert 2', context: 'negative' },
202
+ }),
203
+ ]),
204
+ }));
205
+ });
206
+ it('should pass the additionalInfo to the the renderer', async () => {
207
+ const customRenderer = {
208
+ canRenderType: 'review',
209
+ render: vi.fn(),
210
+ };
211
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
212
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
213
+ fields: expect.arrayContaining([
214
+ expect.objectContaining({
215
+ additionalInfo: expect.objectContaining({ text: 'Additional info 1' }),
216
+ }),
217
+ expect.objectContaining({
218
+ additionalInfo: expect.objectContaining({ text: 'Additional info 2' }),
219
+ }),
220
+ ]),
221
+ }));
222
+ });
223
+ it('should pass the callToAction to the the renderer', async () => {
224
+ const customRenderer = {
225
+ canRenderType: 'review',
226
+ render: vi.fn(),
227
+ };
228
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
229
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
230
+ fields: expect.arrayContaining([
231
+ expect.objectContaining({
232
+ callToAction: expect.objectContaining({ title: 'Call to Action 1' }),
233
+ }),
234
+ ]),
235
+ }));
236
+ });
237
+ it('should pass the analyticsId to the renderer', async () => {
238
+ const customRenderer = {
239
+ canRenderType: 'review',
240
+ render: vi.fn(),
241
+ };
242
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
243
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
244
+ analyticsId: 'review-layout-aid',
245
+ fields: expect.arrayContaining([
246
+ expect.objectContaining({ analyticsId: 'review-field-1-aid' }),
247
+ expect.objectContaining({ analyticsId: 'review-field-2-aid' }),
248
+ ]),
249
+ }));
250
+ });
251
+ describe('interactivity', () => {
252
+ const customRenderer = {
253
+ canRenderType: 'review',
254
+ render: (review) => (_jsx(_Fragment, { children: review.fields.map((field) => {
255
+ var _a, _b, _c, _d;
256
+ return (_jsxs(Fragment, { children: [_jsx("button", { type: "button", onClick: (_a = field.additionalInfo) === null || _a === void 0 ? void 0 : _a.onClick, children: (_b = field.additionalInfo) === null || _b === void 0 ? void 0 : _b.text }), _jsx("button", { type: "button", onClick: (_c = field.callToAction) === null || _c === void 0 ? void 0 : _c.onClick, children: (_d = field.callToAction) === null || _d === void 0 ? void 0 : _d.title })] }, field.label));
257
+ }) })),
258
+ };
259
+ it('should call httpClient when clicking on an additionalInfo', async () => {
260
+ const localProps = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep() });
261
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, localProps, { renderers: [customRenderer] })));
262
+ await user.click(screen.getByText('Additional info 2'));
263
+ expect(localProps.httpClient).toHaveBeenCalledWith('/additional-info', expect.any(Object));
264
+ });
265
+ it('should call httpClient when clicking on an items callToAction', async () => {
266
+ const localProps = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep() });
267
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, localProps, { renderers: [customRenderer] })));
268
+ await user.click(screen.getByText('Call to Action 1'));
269
+ expect(localProps.httpClient).toHaveBeenCalledWith('/call-to-action', expect.any(Object));
270
+ });
271
+ });
272
+ });
273
+ });
274
+ });
@@ -0,0 +1,133 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ /* eslint-disable @typescript-eslint/no-non-null-assertion */
3
+ import { act, screen, waitFor } from '@testing-library/react';
4
+ import { userEvent } from '@testing-library/user-event';
5
+ import { renderWithProviders, respondWith } 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('SchemaOnChange', () => {
10
+ const getStep = (schema) => ({
11
+ id: 'step-1',
12
+ title: 'SchemaOnChange Test',
13
+ schemas: [schema],
14
+ layout: [{ type: 'form', schemaId: schema.$id }],
15
+ });
16
+ const renderDF = (props) => {
17
+ var _a, _b, _c;
18
+ const flowId = (_a = props.flowId) !== null && _a !== void 0 ? _a : 'persist-async-test-flow';
19
+ const onError = (_b = props.onError) !== null && _b !== void 0 ? _b : vi.fn();
20
+ const onCompletion = (_c = props.onCompletion) !== null && _c !== void 0 ? _c : vi.fn();
21
+ return renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, Object.assign({ onError, onCompletion, flowId }, props))));
22
+ };
23
+ describe('StringSchema & ActionBehavior', () => {
24
+ it('should submit the step on string schema change', async () => {
25
+ const initialStep = getStep({
26
+ $id: '#name',
27
+ type: 'string',
28
+ title: 'Name',
29
+ onChange: { type: 'action', action: { url: '/submit' } },
30
+ });
31
+ const httpClient = vi.fn();
32
+ renderDF({ initialStep, httpClient });
33
+ const nameInput = screen.getByLabelText('Name');
34
+ await user.type(nameInput, 'John Doe');
35
+ await act(async () => vi.advanceTimersByTimeAsync(1000)); // wait for debounce
36
+ await waitFor(() => {
37
+ expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({
38
+ body: '"John Doe"',
39
+ }));
40
+ });
41
+ });
42
+ });
43
+ describe('OneOfSchema & ModalBehavior', () => {
44
+ it('should open a modal on oneOf schema change', async () => {
45
+ const initialStep = getStep({
46
+ $id: '#name',
47
+ title: 'Currency',
48
+ control: 'radio',
49
+ oneOf: [
50
+ { const: 'USD', title: 'USD' },
51
+ { const: 'EUR', title: 'EUR' },
52
+ { const: 'GBP', title: 'GBP' },
53
+ ],
54
+ onChange: {
55
+ type: 'modal',
56
+ title: 'Currency Selected',
57
+ content: [
58
+ {
59
+ type: 'paragraph',
60
+ text: 'You have selected a currency, but this modal doesnt know which one.',
61
+ },
62
+ ],
63
+ },
64
+ });
65
+ renderDF({ initialStep });
66
+ await user.click(screen.getByText('GBP'));
67
+ await waitFor(() => {
68
+ expect(screen.getByText('Currency Selected')).toBeInTheDocument();
69
+ });
70
+ });
71
+ });
72
+ describe('BooleanSchema & LinkBehavior', () => {
73
+ it('should open a link on boolean schema change', async () => {
74
+ const initialStep = getStep({
75
+ $id: '#terms',
76
+ type: 'boolean',
77
+ title: 'Accept Terms',
78
+ onChange: {
79
+ type: 'link',
80
+ url: 'https://example.com/terms',
81
+ },
82
+ });
83
+ const onLink = vi.fn();
84
+ renderDF({ initialStep, onLink });
85
+ const termsCheckbox = screen.getByLabelText('Accept Terms');
86
+ await user.click(termsCheckbox);
87
+ await waitFor(() => {
88
+ expect(onLink).toHaveBeenCalledWith('https://example.com/terms');
89
+ });
90
+ });
91
+ });
92
+ describe('File Upload Schema with PersistAsync & ActionBehavior', () => {
93
+ it('should submit the step on file upload schema change', async () => {
94
+ const initialStep = getStep({
95
+ $id: '#file-upload',
96
+ type: 'string',
97
+ title: 'Upload File Outer Schema',
98
+ persistAsync: {
99
+ url: '/upload',
100
+ method: 'POST',
101
+ idProperty: 'id',
102
+ param: 'value',
103
+ schema: {
104
+ type: 'blob',
105
+ title: 'Upload File',
106
+ description: 'Upload a file',
107
+ },
108
+ },
109
+ onChange: { type: 'action', action: { url: '/submit' } },
110
+ });
111
+ const httpClient = vi.fn(async (input) => {
112
+ switch (input) {
113
+ case '/upload':
114
+ return respondWith({ id: '###ttoken###' });
115
+ default:
116
+ return respondWith({});
117
+ }
118
+ });
119
+ renderDF({ initialStep, httpClient });
120
+ const file = new File(['file content'], 'test.txt', { type: 'text/plain' });
121
+ await user.upload(screen.getByTestId('uploadInput'), file);
122
+ await waitFor(() => {
123
+ expect(httpClient).toHaveBeenCalledWith('/upload', expect.objectContaining({
124
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
125
+ body: expect.any(FormData),
126
+ }));
127
+ });
128
+ await waitFor(() => {
129
+ expect(httpClient).toHaveBeenLastCalledWith('/submit', expect.objectContaining({}));
130
+ });
131
+ });
132
+ });
133
+ });
@@ -0,0 +1,88 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { screen } from '@testing-library/react';
3
+ import userEvent from '@testing-library/user-event';
4
+ import { vi } from 'vitest';
5
+ import { renderWithProviders } from '../test-utils';
6
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
7
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
8
+ describe('Schema References', () => {
9
+ const getDefaultProps = () => ({
10
+ flowId: 'flow-id',
11
+ httpClient: vi.fn(),
12
+ onCompletion: vi.fn(),
13
+ onError: vi.fn(),
14
+ onEvent: vi.fn(),
15
+ onLog: vi.fn(),
16
+ });
17
+ const getInitialStep = () => ({
18
+ id: 'step-id',
19
+ title: 'Step Title',
20
+ schemas: [
21
+ {
22
+ $id: '#root-schema-id',
23
+ title: 'Select an account',
24
+ oneOf: [
25
+ {
26
+ $id: '#root-schema-id',
27
+ type: 'object',
28
+ title: 'Object Schema Title',
29
+ displayOrder: ['accountId'],
30
+ properties: {
31
+ accountId: {
32
+ title: 'Current account',
33
+ type: 'string',
34
+ },
35
+ },
36
+ },
37
+ ],
38
+ },
39
+ ],
40
+ layout: [
41
+ {
42
+ type: 'form',
43
+ schemaId: '#root-schema-id',
44
+ },
45
+ {
46
+ type: 'button',
47
+ title: 'Submit',
48
+ behavior: { type: 'action', action: { url: '/submit' } },
49
+ },
50
+ ],
51
+ });
52
+ describe('Given a step with a root-level oneOf schema containing only one option', () => {
53
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep: getInitialStep() });
54
+ it('should render without failing', async () => {
55
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
56
+ expect(props.onError).not.toHaveBeenCalled();
57
+ expect(screen.getByText('Step Title')).toBeInTheDocument();
58
+ });
59
+ it('should skip the oneOf schema and show the single-option schema directly', async () => {
60
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
61
+ expect(screen.queryByText('Object Schema Title')).not.toBeInTheDocument();
62
+ expect(screen.getByText('Current account')).toBeInTheDocument();
63
+ });
64
+ it('should trigger an analytics event for auto-selecting the single option', async () => {
65
+ const onEvent = vi.fn();
66
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { onAnalytics: onEvent })));
67
+ expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - OneOf Option Selected - Auto-select single-option', expect.objectContaining({
68
+ integrationId: 'flow-id',
69
+ flowId: 'flow-id',
70
+ stepId: 'step-id',
71
+ }));
72
+ });
73
+ describe('when entering a value and attempting to submit', () => {
74
+ it('should submit the form successfully', async () => {
75
+ const mockHttpClient = vi.fn();
76
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { httpClient: mockHttpClient })));
77
+ const input = screen.getByLabelText('Current account');
78
+ await user.type(input, 'test-account');
79
+ const submitButton = screen.getByRole('button', { name: 'Submit' });
80
+ await user.click(submitButton);
81
+ expect(mockHttpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({
82
+ method: 'POST',
83
+ body: JSON.stringify({ accountId: 'test-account' }),
84
+ }));
85
+ });
86
+ });
87
+ });
88
+ });
@@ -0,0 +1,122 @@
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 { vi } from 'vitest';
5
+ import { renderWithProviders } from '../test-utils';
6
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
7
+ /*
8
+ This test mocks the IntersectionObserver API, because,
9
+ unfortunately, jsdom doesn't implement real layout, scrolling, or visibility.
10
+
11
+ - IntersectionObserver doesn't exist (hence why we mock it)
12
+ - scrollIntoView() does nothing — it's a no-op
13
+ - getBoundingClientRect() always returns all zeros
14
+ - There is no actual viewport or rendering engine
15
+ */
16
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
17
+ const getDefaultProps = () => ({
18
+ flowId: 'flow-id',
19
+ onCompletion: vi.fn(),
20
+ onError: vi.fn(),
21
+ onEvent: vi.fn(),
22
+ onLog: vi.fn(),
23
+ });
24
+ const makeStep = (overrides = {}) => (Object.assign({ id: 'step-id', title: 'Step', schemas: [], layout: [] }, overrides));
25
+ const mockIntersectionObserver = (isIntersecting) => {
26
+ vi.stubGlobal('IntersectionObserver', class {
27
+ get root() {
28
+ return null;
29
+ }
30
+ get rootMargin() {
31
+ return '';
32
+ }
33
+ get thresholds() {
34
+ return [];
35
+ }
36
+ constructor(callback) {
37
+ this.observe = vi.fn().mockImplementation((el) => {
38
+ this.callback([{ isIntersecting, target: el }], this);
39
+ });
40
+ this.unobserve = vi.fn();
41
+ this.disconnect = vi.fn();
42
+ this.takeRecords = vi.fn().mockReturnValue([]);
43
+ this.callback = callback;
44
+ }
45
+ });
46
+ };
47
+ describe('Scroll to bottom', () => {
48
+ afterEach(() => {
49
+ vi.unstubAllGlobals();
50
+ });
51
+ const stepWithoutFooter = makeStep({
52
+ layout: [{ type: 'markdown', content: 'Some very long terms and conditions content.' }],
53
+ tags: ['scroll-to-bottom'],
54
+ });
55
+ const stepWithFooter = makeStep({
56
+ layout: [{ type: 'markdown', content: 'Some very long terms and conditions content.' }],
57
+ footer: [
58
+ {
59
+ type: 'button',
60
+ title: 'Accept',
61
+ behavior: { type: 'action', action: { url: '/accept' } },
62
+ },
63
+ ],
64
+ tags: ['scroll-to-bottom'],
65
+ });
66
+ describe('given a step with a long markdown content without a footer', () => {
67
+ it('shows a scroll to bottom button if the end of the step is not visible', async () => {
68
+ mockIntersectionObserver(false);
69
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ initialStep: stepWithoutFooter, httpClient: vi.fn() }, getDefaultProps())));
70
+ await waitFor(() => {
71
+ expect(screen.getByRole('button', { name: 'Scroll to bottom' })).toBeInTheDocument();
72
+ });
73
+ });
74
+ it('scrolls to the bottom of the step when the scroll to bottom button is clicked', async () => {
75
+ const scrollIntoViewMock = vi.fn();
76
+ window.HTMLElement.prototype.scrollIntoView = scrollIntoViewMock;
77
+ mockIntersectionObserver(false);
78
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ initialStep: stepWithoutFooter, httpClient: vi.fn() }, getDefaultProps())));
79
+ await waitFor(() => {
80
+ expect(screen.getByRole('button', { name: 'Scroll to bottom' })).toBeInTheDocument();
81
+ });
82
+ await user.click(screen.getByRole('button', { name: 'Scroll to bottom' }));
83
+ expect(scrollIntoViewMock).toHaveBeenCalled();
84
+ });
85
+ it('does not show a scroll to bottom button if the end of the step is visible', async () => {
86
+ mockIntersectionObserver(true);
87
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ initialStep: stepWithoutFooter, httpClient: vi.fn() }, getDefaultProps())));
88
+ await waitFor(() => {
89
+ expect(screen.queryByRole('button', { name: 'Scroll to bottom' })).not.toBeInTheDocument();
90
+ });
91
+ });
92
+ });
93
+ describe('given a step with a long markdown content and a footer', () => {
94
+ it('shows a scroll to bottom button in the footer, along with the footer content, if the end of the step is not visible', async () => {
95
+ mockIntersectionObserver(false);
96
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ initialStep: stepWithFooter, httpClient: vi.fn() }, getDefaultProps())));
97
+ await waitFor(() => {
98
+ expect(screen.getByRole('button', { name: 'Scroll to bottom' })).toBeInTheDocument();
99
+ });
100
+ expect(screen.getByRole('button', { name: 'Accept' })).toBeInTheDocument();
101
+ });
102
+ it('scrolls to the bottom of the step when the scroll to bottom button is clicked', async () => {
103
+ const scrollIntoViewMock = vi.fn();
104
+ window.HTMLElement.prototype.scrollIntoView = scrollIntoViewMock;
105
+ mockIntersectionObserver(false);
106
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ initialStep: stepWithFooter, httpClient: vi.fn() }, getDefaultProps())));
107
+ await waitFor(() => {
108
+ expect(screen.getByRole('button', { name: 'Scroll to bottom' })).toBeInTheDocument();
109
+ });
110
+ await user.click(screen.getByRole('button', { name: 'Scroll to bottom' }));
111
+ expect(scrollIntoViewMock).toHaveBeenCalled();
112
+ });
113
+ it('does not show a scroll to bottom button in the footer, if the end of the step is visible', async () => {
114
+ mockIntersectionObserver(true);
115
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ initialStep: stepWithFooter, httpClient: vi.fn() }, getDefaultProps())));
116
+ await waitFor(() => {
117
+ expect(screen.queryByRole('button', { name: 'Scroll to bottom' })).not.toBeInTheDocument();
118
+ });
119
+ expect(screen.getByRole('button', { name: 'Accept' })).toBeInTheDocument();
120
+ });
121
+ });
122
+ });