@wise/dynamic-flow-client 5.9.2-logging-bf0bd22 → 5.9.2

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 (402) 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 +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/main.js +1 -7
  208. package/build/main.mjs +1 -7
  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 +34 -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 +211 -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/schemas/oneOf/OneOfWithSingleOption.story.js +114 -0
  296. package/build/stories/spec/step/Controls.story.js +109 -0
  297. package/build/stories/spec/step/DFModal.story.js +58 -0
  298. package/build/stories/spec/step/Footer.story.js +70 -0
  299. package/build/stories/spec/step/Navigation.story.js +20 -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/rtl-utils.js +7 -0
  323. package/build/test-utils/step-utils.js +6 -0
  324. package/build/test-utils/wait.js +3 -0
  325. package/build/tests/AlertLayout.test.js +78 -0
  326. package/build/tests/ArrayTuple.test.js +118 -0
  327. package/build/tests/ButtonLayout.test.js +308 -0
  328. package/build/tests/ConstLayout.test.js +95 -0
  329. package/build/tests/DateInput.test.js +163 -0
  330. package/build/tests/DecisionLayout.test.js +146 -0
  331. package/build/tests/DynamicFlow.test.js +147 -0
  332. package/build/tests/External.test.js +169 -0
  333. package/build/tests/Flow.test.js +328 -0
  334. package/build/tests/FormLayout.test.js +28 -0
  335. package/build/tests/FormattedValue.test.js +107 -0
  336. package/build/tests/ImageRenderer.test.js +78 -0
  337. package/build/tests/InitialAction.test.js +179 -0
  338. package/build/tests/InitialStep.test.js +168 -0
  339. package/build/tests/InstructionsLayout.test.js +45 -0
  340. package/build/tests/ListLayout.test.js +145 -0
  341. package/build/tests/Logging.test.js +53 -0
  342. package/build/tests/ModalBehavior.test.js +149 -0
  343. package/build/tests/MoneyInput.test.js +316 -0
  344. package/build/tests/MultiUpload.test.js +293 -0
  345. package/build/tests/NativeBack.test.js +267 -0
  346. package/build/tests/OneOfInitialisation.test.js +571 -0
  347. package/build/tests/PersistAsync.test.js +687 -0
  348. package/build/tests/Polling.test.js +617 -0
  349. package/build/tests/Prefetching.test.js +230 -0
  350. package/build/tests/RefreshAfter.test.js +63 -0
  351. package/build/tests/RefreshOnChange.ResponseHandling.test.js +205 -0
  352. package/build/tests/RefreshOnChange.test.js +233 -0
  353. package/build/tests/RefreshOnChange.with.Segmented.test.js +348 -0
  354. package/build/tests/RefreshOnChange.with.Tabs.test.js +358 -0
  355. package/build/tests/RefreshOnChangePreserve.test.js +224 -0
  356. package/build/tests/RendererProps.test.js +342 -0
  357. package/build/tests/Repeatable.test.js +107 -0
  358. package/build/tests/Rerendering.test.js +67 -0
  359. package/build/tests/ReviewLayout.test.js +274 -0
  360. package/build/tests/SchemaOnChange.test.js +133 -0
  361. package/build/tests/SchemaReferences.test.js +88 -0
  362. package/build/tests/ScrollToError.test.js +217 -0
  363. package/build/tests/SegmentedControl.test.js +49 -0
  364. package/build/tests/SingleFileUpload.test.js +88 -0
  365. package/build/tests/StatusList.test.js +85 -0
  366. package/build/tests/Subflow.test.js +710 -0
  367. package/build/tests/Submission.ResponseHandling.test.js +557 -0
  368. package/build/tests/Submission.merging.test.js +202 -0
  369. package/build/tests/Submission.test.js +603 -0
  370. package/build/tests/Tags.test.js +475 -0
  371. package/build/tests/Upsell.test.js +126 -0
  372. package/build/tests/ValidationAsync.test.js +295 -0
  373. package/build/tests/legacy/Actions.test.js +158 -0
  374. package/build/tests/legacy/BackButton.test.js +114 -0
  375. package/build/tests/legacy/HiddenSchemas.test.js +246 -0
  376. package/build/tests/legacy/MultiSelect.test.js +501 -0
  377. package/build/tests/legacy/MultipleFileUploadSchema.test.js +341 -0
  378. package/build/tests/legacy/PersistAsync.blob-schema.test.js +224 -0
  379. package/build/tests/legacy/PersistAsync.string-schema.test.js +211 -0
  380. package/build/tests/legacy/RefreshStepOnChange.debouncing.test.js +209 -0
  381. package/build/tests/legacy/RefreshStepOnChange.test.js +424 -0
  382. package/build/tests/legacy/Search.test.js +437 -0
  383. package/build/tests/renderers/MultiSelectInputRendererProps.test.js +58 -0
  384. package/build/tests/renderers/SelectInputRendererProps.test.js +43 -0
  385. package/build/tests/renderers/TextInputRenderer.test.js +51 -0
  386. package/build/types.js +1 -0
  387. package/build/useDynamicFlow.js +104 -0
  388. package/build/useDynamicFlowModal.js +58 -0
  389. package/build/utils/analyse-step.js +14 -0
  390. package/build/utils/component-utils.js +8 -0
  391. package/build/utils/component-utils.test.js +113 -0
  392. package/build/utils/getScrollToTop.js +12 -0
  393. package/build/utils/normalise-flow-id.js +1 -0
  394. package/build/utils/normalise-flow-id.test.js +24 -0
  395. package/build/utils/openLinkInNewTab.js +10 -0
  396. package/build/utils/recursiveMerge.js +40 -0
  397. package/build/utils/recursiveMerge.test.js +93 -0
  398. package/build/utils/type-utils.js +21 -0
  399. package/build/utils/type-validators.js +11 -0
  400. package/build/utils/type-validators.test.js +180 -0
  401. package/build/utils/useStableCallback.js +15 -0
  402. package/package.json +10 -10
@@ -0,0 +1,163 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { renderWithProviders } from '../test-utils';
3
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
4
+ import userEvent from '@testing-library/user-event';
5
+ import { screen, waitFor } from '@testing-library/react';
6
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
7
+ describe('given a step with date input', () => {
8
+ const step = (required, defaultValue) => ({
9
+ id: 'step',
10
+ title: 'step title',
11
+ schemas: [
12
+ {
13
+ properties: {
14
+ date: {
15
+ format: 'date',
16
+ title: 'This is a date',
17
+ type: 'string',
18
+ default: defaultValue,
19
+ },
20
+ },
21
+ required: required ? ['date'] : undefined,
22
+ displayOrder: ['date'],
23
+ $id: '#schema',
24
+ type: 'object',
25
+ },
26
+ ],
27
+ layout: [
28
+ {
29
+ schemaId: '#schema',
30
+ type: 'form',
31
+ },
32
+ {
33
+ type: 'button',
34
+ title: 'Submit',
35
+ action: { url: '/submit' },
36
+ },
37
+ ],
38
+ });
39
+ describe('when a full date is provided', () => {
40
+ it('submits successfully', async () => {
41
+ vi.setSystemTime(new Date(2026, 1, 1, 12, 0));
42
+ const httpClient = vi.fn();
43
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: step(true), onCompletion: vi.fn(), onError: vi.fn() }));
44
+ await user.click(screen.getAllByText('Month')[0]);
45
+ await waitFor(() => {
46
+ expect(screen.getByText('January')).toBeInTheDocument();
47
+ });
48
+ await user.click(screen.getByText('January'));
49
+ await user.type(screen.getByLabelText('Day'), '1');
50
+ await user.type(screen.getByLabelText('Year'), '2024');
51
+ await user.click(screen.getByText('Submit'));
52
+ await waitFor(() => {
53
+ expect(httpClient).toHaveBeenCalled();
54
+ });
55
+ expect(screen.queryByText('Please fill out this field.')).not.toBeInTheDocument();
56
+ });
57
+ });
58
+ describe('when a partial date (YYYY-MM) is filled in', () => {
59
+ describe('and the field is required', () => {
60
+ it('blocks submission', async () => {
61
+ const httpClient = vi.fn();
62
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: step(true), onCompletion: vi.fn(), onError: vi.fn() }));
63
+ await waitFor(() => {
64
+ expect(screen.getByText('step title')).toBeInTheDocument();
65
+ });
66
+ await user.click(screen.getAllByText('Month')[0]);
67
+ await waitFor(() => {
68
+ expect(screen.getByText('January')).toBeInTheDocument();
69
+ });
70
+ await user.click(screen.getByText('January'));
71
+ await user.type(screen.getByLabelText('Year'), '2024');
72
+ await user.click(screen.getByText('Submit'));
73
+ await waitFor(() => {
74
+ expect(httpClient).not.toHaveBeenCalled();
75
+ });
76
+ expect(screen.getByText('Please fill out this field.')).toBeInTheDocument();
77
+ });
78
+ });
79
+ describe('and the field is optional', () => {
80
+ it('submits null', async () => {
81
+ const httpClient = vi.fn();
82
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: step(false), onCompletion: vi.fn(), onError: vi.fn() }));
83
+ await waitFor(() => {
84
+ expect(screen.getByText('step title')).toBeInTheDocument();
85
+ });
86
+ await user.click(screen.getAllByText('Month')[0]);
87
+ await waitFor(() => {
88
+ expect(screen.getByText('January')).toBeInTheDocument();
89
+ });
90
+ await user.click(screen.getByText('January'));
91
+ await user.type(screen.getByLabelText('Year'), '2024');
92
+ await user.click(screen.getByText('Submit'));
93
+ await waitFor(() => {
94
+ expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({ body: '{}' }));
95
+ });
96
+ expect(screen.queryByText('Please fill out this field.')).not.toBeInTheDocument();
97
+ });
98
+ });
99
+ });
100
+ describe('when a partial date (YYYY-MM) is provided in the model', () => {
101
+ describe('and the field is required', () => {
102
+ it('blocks submission', async () => {
103
+ const httpClient = vi.fn();
104
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: Object.assign(Object.assign({}, step(true)), { model: { date: '2024-07' } }), onCompletion: vi.fn(), onError: vi.fn() }));
105
+ await user.click(screen.getByText('Submit'));
106
+ await waitFor(() => {
107
+ expect(httpClient).not.toHaveBeenCalled();
108
+ });
109
+ expect(screen.getByText('Please fill out this field.')).toBeInTheDocument();
110
+ });
111
+ });
112
+ describe('and the field is optional', () => {
113
+ it('submits null', async () => {
114
+ const httpClient = vi.fn();
115
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: Object.assign(Object.assign({}, step(false)), { model: { date: '2024-07' } }), onCompletion: vi.fn(), onError: vi.fn() }));
116
+ await user.click(screen.getByText('Submit'));
117
+ await waitFor(() => {
118
+ expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({ body: '{}' }));
119
+ });
120
+ expect(screen.queryByText('Please fill out this field.')).not.toBeInTheDocument();
121
+ });
122
+ });
123
+ });
124
+ describe('when an invalid date is provided', () => {
125
+ const cases = [
126
+ { model: 'abcd-ef-gh', description: 'nonsense string in model' },
127
+ { model: '25-10-1985', description: 'wrong format in model' },
128
+ { default: 'abcd-ef-gh', description: 'nonsense string in default' },
129
+ { default: '25-10-1985', description: 'wrong format in default' },
130
+ ];
131
+ cases.forEach(({ model, default: defaultValue, description }) => {
132
+ it(`blocks submission for ${model !== null && model !== void 0 ? model : defaultValue} because "${description}"`, async () => {
133
+ const httpClient = vi.fn();
134
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: Object.assign(Object.assign({}, step(true, defaultValue)), { model: { date: model !== null && model !== void 0 ? model : null } }), onCompletion: vi.fn(), onError: vi.fn() }));
135
+ await user.click(screen.getByText('Submit'));
136
+ await waitFor(() => {
137
+ expect(httpClient).not.toHaveBeenCalled();
138
+ });
139
+ expect(screen.getByText('Please fill out this field.')).toBeInTheDocument();
140
+ });
141
+ });
142
+ });
143
+ describe('when a date-time value is provided in the model', () => {
144
+ it('extracts and uses only the date portion', async () => {
145
+ const httpClient = vi.fn();
146
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: Object.assign(Object.assign({}, step(true)), { model: { date: '1985-10-25T01:18:00.000-08:00' } }), onCompletion: vi.fn(), onError: vi.fn() }));
147
+ await user.click(screen.getByText('Submit'));
148
+ await waitFor(() => {
149
+ expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({ body: '{"date":"1985-10-25"}' }));
150
+ });
151
+ });
152
+ });
153
+ describe('when a date-time value is provided in the schema default', () => {
154
+ it('extracts and uses only the date portion', async () => {
155
+ const httpClient = vi.fn();
156
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", httpClient: httpClient, initialStep: Object.assign({}, step(true, '1985-10-25T01:18:00.000-08:00')), onCompletion: vi.fn(), onError: vi.fn() }));
157
+ await user.click(screen.getByText('Submit'));
158
+ await waitFor(() => {
159
+ expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({ body: '{"date":"1985-10-25"}' }));
160
+ });
161
+ });
162
+ });
163
+ });
@@ -0,0 +1,146 @@
1
+ import { jsx as _jsx } 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 { vi } from 'vitest';
6
+ import { renderWithProviders } from '../test-utils';
7
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
8
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
9
+ describe('DecisionLayout 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 initialStep = {
19
+ id: 'step-id',
20
+ title: 'Step with decision',
21
+ layout: [
22
+ {
23
+ type: 'decision',
24
+ analyticsId: 'decision-layout-aid',
25
+ options: [
26
+ {
27
+ analyticsId: 'action-option-1-aid',
28
+ title: 'Action Option',
29
+ action: { url: '/action-option' },
30
+ tag: 'action-option',
31
+ additionalText: 'Additional text',
32
+ inlineAlert: {
33
+ context: 'warning',
34
+ content: 'This is an inline alert',
35
+ },
36
+ },
37
+ {
38
+ analyticsId: 'action-option-2-aid',
39
+ title: 'Behavior Action Option',
40
+ behavior: {
41
+ type: 'action',
42
+ action: { url: '/behavior-action' },
43
+ },
44
+ tag: 'behavior-action-option',
45
+ },
46
+ {
47
+ analyticsId: 'action-option-3-aid',
48
+ title: 'Behavior Link Option',
49
+ behavior: {
50
+ type: 'link',
51
+ url: '/behavior-link',
52
+ },
53
+ tag: 'behavior-link-option',
54
+ },
55
+ ],
56
+ },
57
+ ],
58
+ schemas: [],
59
+ };
60
+ describe('Legacy action', () => {
61
+ describe('Given a decision layout component with an action', () => {
62
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep });
63
+ it('should render a button that triggers the action on click', async () => {
64
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
65
+ await waitFor(() => {
66
+ expect(screen.getByRole('button', { name: 'Action Option' })).toBeInTheDocument();
67
+ });
68
+ await user.click(screen.getByRole('button', { name: 'Action Option' }));
69
+ expect(props.httpClient).toHaveBeenCalledWith('/action-option', expect.any(Object));
70
+ });
71
+ });
72
+ });
73
+ describe('New "union type" behavior type', () => {
74
+ describe('Given a decision layout component with a link behavior', () => {
75
+ const onLink = vi.fn();
76
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { onLink,
77
+ initialStep });
78
+ it('should render a link that opens the url on click', async () => {
79
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
80
+ await waitFor(() => {
81
+ expect(screen.getByRole('link', { name: 'Behavior Link Option' })).toBeInTheDocument();
82
+ });
83
+ expect(screen.getByRole('link', { name: 'Behavior Link Option' })).toHaveAttribute('href', '/behavior-link');
84
+ });
85
+ });
86
+ describe('Given a decision layout component with an action behavior', () => {
87
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep });
88
+ it('should trigger the action on click', async () => {
89
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
90
+ await waitFor(() => {
91
+ expect(screen.getByRole('button', { name: 'Behavior Action Option' })).toBeInTheDocument();
92
+ });
93
+ await user.click(screen.getByRole('button', { name: 'Behavior Action Option' }));
94
+ expect(props.httpClient).toHaveBeenCalledWith('/behavior-action', expect.any(Object));
95
+ });
96
+ });
97
+ });
98
+ describe('Extra props', () => {
99
+ const customRenderer = {
100
+ canRenderType: 'decision',
101
+ render: vi.fn(),
102
+ };
103
+ it('should pass the tag in each option to the renderer', () => {
104
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep });
105
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
106
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
107
+ options: expect.arrayContaining([
108
+ expect.objectContaining({ tag: 'action-option' }),
109
+ expect.objectContaining({ tag: 'behavior-action-option' }),
110
+ expect.objectContaining({ tag: 'behavior-link-option' }),
111
+ ]),
112
+ }));
113
+ });
114
+ it('should pass the additional text, supporting values, and inline alert in each option to the renderer', () => {
115
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep });
116
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
117
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
118
+ options: expect.arrayContaining([
119
+ expect.objectContaining({ additionalText: 'Additional text' }),
120
+ ]),
121
+ }));
122
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
123
+ options: expect.arrayContaining([
124
+ expect.objectContaining({
125
+ inlineAlert: {
126
+ context: 'warning',
127
+ content: 'This is an inline alert',
128
+ },
129
+ }),
130
+ ]),
131
+ }));
132
+ });
133
+ it('should pass the analyticsId in each option to the renderer', () => {
134
+ const props = Object.assign(Object.assign({}, getDefaultProps()), { initialStep });
135
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props, { renderers: [customRenderer] })));
136
+ expect(customRenderer.render).toHaveBeenCalledWith(expect.objectContaining({
137
+ analyticsId: 'decision-layout-aid',
138
+ options: expect.arrayContaining([
139
+ expect.objectContaining({ analyticsId: 'action-option-1-aid' }),
140
+ expect.objectContaining({ analyticsId: 'action-option-2-aid' }),
141
+ expect.objectContaining({ analyticsId: 'action-option-3-aid' }),
142
+ ]),
143
+ }));
144
+ });
145
+ });
146
+ });
@@ -0,0 +1,147 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { screen, waitFor } from '@testing-library/react';
3
+ import { vi, beforeEach } from 'vitest';
4
+ import { getMockHttpClient, renderWithProviders } from '../test-utils';
5
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
6
+ import userEvent from '@testing-library/user-event';
7
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
8
+ describe('DynamicFlow', () => {
9
+ const makeDefaultProps = () => ({
10
+ httpClient: getMockHttpClient({}),
11
+ onCompletion: vi.fn(),
12
+ onError: vi.fn(),
13
+ onEvent: vi.fn(),
14
+ onLog: vi.fn(),
15
+ });
16
+ beforeEach(() => {
17
+ vi.clearAllMocks();
18
+ });
19
+ describe('initial step with initial model', () => {
20
+ it('renders the step with relevant fields filled in', () => {
21
+ const step = {
22
+ id: 'new-df-initial-step',
23
+ title: 'Dynamic Flow',
24
+ schemas: [
25
+ {
26
+ $id: '#schema',
27
+ type: 'object',
28
+ title: 'Object schema title',
29
+ description: 'Object schema description',
30
+ displayOrder: ['name', 'occupation'],
31
+ properties: {
32
+ name: {
33
+ title: 'Full name',
34
+ type: 'string',
35
+ },
36
+ occupation: {
37
+ title: 'Occupation',
38
+ type: 'string',
39
+ },
40
+ },
41
+ },
42
+ ],
43
+ layout: [{ type: 'form', schemaId: '#schema' }],
44
+ model: { name: 'John Smith' },
45
+ };
46
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ flowId: "new-df-flow-id" }, makeDefaultProps(), { initialStep: step })));
47
+ expect(screen.getByDisplayValue('John Smith')).toBeInTheDocument();
48
+ });
49
+ });
50
+ describe('events', () => {
51
+ describe('when the flow starts with flowId and stepId', () => {
52
+ it('emits a flow started event containing those values', () => {
53
+ const props = Object.assign(Object.assign({}, makeDefaultProps()), { initialStep: {
54
+ id: 'ny-initial-step-id',
55
+ title: 'Dynamic Flow',
56
+ schemas: [],
57
+ layout: [],
58
+ } });
59
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ flowId: "my-flow-id" }, props)));
60
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Initiated', expect.objectContaining({
61
+ flowId: 'my-flow-id',
62
+ stepId: 'ny-initial-step-id',
63
+ }));
64
+ });
65
+ it('emits a step shown event containing those values', () => {
66
+ const props = Object.assign(Object.assign({}, makeDefaultProps()), { initialStep: {
67
+ id: 'ny-initial-step-id',
68
+ title: 'Dynamic Flow',
69
+ schemas: [],
70
+ layout: [],
71
+ } });
72
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ flowId: "my-flow-id" }, props)));
73
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Step Shown', expect.objectContaining({
74
+ flowId: 'my-flow-id',
75
+ stepId: 'ny-initial-step-id',
76
+ }));
77
+ });
78
+ });
79
+ });
80
+ describe('onCopy', () => {
81
+ const textToCopy = 'Sample text';
82
+ const step = {
83
+ id: 'new-df-initial-step',
84
+ title: 'Dynamic Flow',
85
+ schemas: [],
86
+ layout: [
87
+ {
88
+ type: 'button',
89
+ title: 'Click to copy',
90
+ control: 'primary',
91
+ behavior: {
92
+ type: 'copy',
93
+ content: textToCopy,
94
+ },
95
+ },
96
+ ],
97
+ };
98
+ it('calls onCopy when content is copied', async () => {
99
+ const props = makeDefaultProps();
100
+ const onCopy = vi.fn();
101
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ flowId: "my-flow-id" }, props, { initialStep: step, onCopy: onCopy })));
102
+ const copyButton = screen.getByText('Click to copy');
103
+ await user.click(copyButton);
104
+ await waitFor(() => {
105
+ expect(onCopy).toHaveBeenCalledWith(textToCopy);
106
+ });
107
+ });
108
+ it('calls onCopy with null when copy fails', async () => {
109
+ const props = makeDefaultProps();
110
+ const onCopy = vi.fn();
111
+ Object.defineProperty(navigator, 'clipboard', {
112
+ value: {
113
+ writeText: vi.fn(async () => Promise.reject(new Error('Copy failed'))),
114
+ },
115
+ writable: true,
116
+ });
117
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ flowId: "my-flow-id" }, props, { initialStep: step, onCopy: onCopy })));
118
+ const copyButton = screen.getByText('Click to copy');
119
+ await user.click(copyButton);
120
+ await waitFor(() => {
121
+ expect(onCopy).toHaveBeenCalledWith(null);
122
+ });
123
+ });
124
+ });
125
+ describe('hideStepTitle feature flag', () => {
126
+ it('hides the step title when the feature flag is enabled', () => {
127
+ const step = {
128
+ id: 'step-with-title',
129
+ title: 'Hidden Title',
130
+ schemas: [],
131
+ layout: [],
132
+ };
133
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ flowId: "flow-with-hide-title" }, makeDefaultProps(), { initialStep: step, features: { hideStepTitle: true } })));
134
+ expect(screen.queryByText('Hidden Title')).not.toBeInTheDocument();
135
+ });
136
+ it('shows the step title when the feature flag is disabled', () => {
137
+ const step = {
138
+ id: 'step-with-title',
139
+ title: 'Visible Title',
140
+ schemas: [],
141
+ layout: [],
142
+ };
143
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ flowId: "flow-with-show-title" }, makeDefaultProps(), { initialStep: step, features: { hideStepTitle: false } })));
144
+ expect(screen.getByText('Visible Title')).toBeInTheDocument();
145
+ });
146
+ });
147
+ });
@@ -0,0 +1,169 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { fireEvent, screen, waitFor } from '@testing-library/react';
3
+ import { vi } from 'vitest';
4
+ import { act } from 'react';
5
+ import { getMockHttpClient, renderWithProviders, respondWith } from '../test-utils';
6
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
7
+ const mockWindowOpenSuccess = () => vi.mocked(window.open).mockImplementation(() => ({}));
8
+ const mockWindowOpenFail = () => vi.mocked(window.open).mockImplementation(() => null);
9
+ describe('External', () => {
10
+ const mockHttpClient = vi.fn(async () => Promise.resolve(new Response('')));
11
+ const onError = vi.fn();
12
+ const onCompletion = vi.fn();
13
+ const renderComponent = (step) => renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", initialStep: step, httpClient: mockHttpClient, onCompletion: onCompletion, onError: onError }));
14
+ afterEach(() => {
15
+ mockHttpClient.mockClear();
16
+ vi.restoreAllMocks();
17
+ });
18
+ describe('Given a step with an "external" property', () => {
19
+ const step = {
20
+ type: 'form',
21
+ id: 'external',
22
+ title: 'Step Title',
23
+ external: { url: 'https://third-party.website' },
24
+ layout: [{ type: 'paragraph', text: 'This is a paragraph in the step layout' }],
25
+ schemas: [],
26
+ };
27
+ it('attempts to open the specified URL, but ONLY ONCE', async () => {
28
+ vi.spyOn(window, 'open');
29
+ mockWindowOpenSuccess();
30
+ renderComponent(step);
31
+ expect(window.open).toHaveBeenCalledTimes(1);
32
+ expect(window.open).toHaveBeenCalledWith('https://third-party.website', '_blank');
33
+ await waitFor(() => {
34
+ expect(window.open).toHaveBeenCalledTimes(1);
35
+ });
36
+ });
37
+ describe('when the window/tab fails to open', () => {
38
+ it('displays a confirmation UI (ExternalConfirmationStep)', async () => {
39
+ vi.spyOn(window, 'open');
40
+ mockWindowOpenFail();
41
+ renderComponent(step);
42
+ await expect(screen.findByText('Please confirm')).resolves.toBeInTheDocument();
43
+ });
44
+ describe('when the confirmation button is clicked', () => {
45
+ it('opens the external url and displays the step layout', () => {
46
+ vi.spyOn(window, 'open');
47
+ mockWindowOpenFail();
48
+ renderComponent(step);
49
+ expect(screen.getByText('Please confirm')).toBeInTheDocument();
50
+ const confirmButton = screen.getByText('Open in new tab');
51
+ vi.mocked(window.open).mockClear();
52
+ fireEvent.click(confirmButton);
53
+ expect(window.open).toHaveBeenCalledTimes(1);
54
+ expect(screen.getByText('This is a paragraph in the step layout')).toBeInTheDocument();
55
+ });
56
+ });
57
+ describe('when the cancel button is clicked', () => {
58
+ it('does not open the external url and displays the step layout', () => {
59
+ vi.spyOn(window, 'open');
60
+ mockWindowOpenFail();
61
+ renderComponent(step);
62
+ expect(screen.getByText('Please confirm')).toBeInTheDocument();
63
+ const confirmButton = screen.getByText('Cancel');
64
+ vi.mocked(window.open).mockClear();
65
+ fireEvent.click(confirmButton);
66
+ expect(window.open).toHaveBeenCalledTimes(0);
67
+ expect(screen.getByText('This is a paragraph in the step layout')).toBeInTheDocument();
68
+ });
69
+ });
70
+ describe('when a subsequent step with external is loaded using the same URL', () => {
71
+ const mockClient = getMockHttpClient({
72
+ '/external': async () => respondWith({
73
+ id: 'external',
74
+ title: 'External Step',
75
+ external: { url: 'https://third-party.website' },
76
+ layout: [
77
+ { type: 'paragraph', text: 'This is a paragraph in the external step layout' },
78
+ { type: 'button', action: { url: '/layout', title: 'Continue to layout step' } },
79
+ ],
80
+ schemas: [],
81
+ }),
82
+ '/layout': async () => respondWith({
83
+ id: 'layout',
84
+ title: 'Layout Step',
85
+ layout: [
86
+ { type: 'paragraph', text: 'This is a paragraph in the layout step layout' },
87
+ {
88
+ type: 'button',
89
+ action: { url: '/external', title: 'Continue to external step' },
90
+ },
91
+ ],
92
+ schemas: [],
93
+ }),
94
+ });
95
+ it('displays the confirmation UI (ExternalConfirmationStep) a second time', async () => {
96
+ vi.spyOn(window, 'open');
97
+ mockWindowOpenFail();
98
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", initialAction: { method: 'GET', url: '/external' }, httpClient: mockClient, onCompletion: onCompletion, onError: onError }));
99
+ await expect(screen.findByText('Please confirm')).resolves.toBeInTheDocument();
100
+ const confirmButton = screen.getByText('Cancel');
101
+ vi.mocked(window.open).mockClear();
102
+ fireEvent.click(confirmButton);
103
+ expect(window.open).toHaveBeenCalledTimes(0);
104
+ expect(screen.getByText('This is a paragraph in the external step layout')).toBeInTheDocument();
105
+ fireEvent.click(screen.getByText(/continue to layout step/i));
106
+ await expect(screen.findByText('Layout Step')).resolves.toBeInTheDocument();
107
+ fireEvent.click(screen.getByText(/continue to external step/i));
108
+ await expect(screen.findByText('Please confirm')).resolves.toBeInTheDocument();
109
+ });
110
+ });
111
+ });
112
+ describe('when the window/tab opens successfully', () => {
113
+ it('displays the step normally', async () => {
114
+ vi.spyOn(window, 'open');
115
+ mockWindowOpenSuccess();
116
+ renderComponent(step);
117
+ await expect(screen.findByText('This is a paragraph in the step layout')).resolves.toBeInTheDocument();
118
+ expect(screen.queryByText('Please confirm')).not.toBeInTheDocument();
119
+ });
120
+ });
121
+ });
122
+ describe('Given a step with an "external" property and "refreshAfter"', () => {
123
+ const step = {
124
+ type: 'form',
125
+ id: 'external',
126
+ title: 'Step Title',
127
+ external: { url: 'https://third-party.website' },
128
+ layout: [{ type: 'paragraph', text: 'This is a paragraph in the step layout' }],
129
+ schemas: [],
130
+ refreshAfter: new Date(Date.now() + 1000).toISOString(),
131
+ refreshUrl: '/refresh',
132
+ };
133
+ it('attempts to open the specified URL, but ONLY ONCE, even after the refresh', async () => {
134
+ vi.spyOn(window, 'open');
135
+ mockWindowOpenSuccess();
136
+ const httpClient = vi.fn(async (input) => {
137
+ return input === '/refresh'
138
+ ? respondWith(Object.assign(Object.assign({}, step), { title: 'Refreshed Title' }), { status: 200 })
139
+ : respondWith(null, { status: 404 });
140
+ });
141
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", initialStep: step, httpClient: httpClient, onCompletion: onCompletion, onError: onError }));
142
+ expect(window.open).toHaveBeenCalledTimes(1);
143
+ expect(window.open).toHaveBeenCalledWith('https://third-party.website', '_blank');
144
+ expect(screen.getByText('Step Title')).toBeInTheDocument();
145
+ await act(async () => vi.advanceTimersByTimeAsync(1100));
146
+ expect(httpClient).toHaveBeenCalledWith('/refresh', expect.any(Object));
147
+ expect(screen.getByText('Refreshed Title')).toBeInTheDocument();
148
+ expect(window.open).toHaveBeenCalledTimes(1);
149
+ });
150
+ });
151
+ describe('Given a step with a malformed "external" property', () => {
152
+ const step = {
153
+ type: 'form',
154
+ id: 'external',
155
+ title: 'Step Title',
156
+ external: { wrong: 'banana' },
157
+ layout: [{ type: 'paragraph', text: 'This is a paragraph in the step layout' }],
158
+ schemas: [],
159
+ };
160
+ it('does not attempt to open a new tab', async () => {
161
+ vi.spyOn(window, 'open');
162
+ mockWindowOpenSuccess();
163
+ renderComponent(step);
164
+ await waitFor(() => {
165
+ expect(window.open).not.toHaveBeenCalled();
166
+ });
167
+ });
168
+ });
169
+ });