@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,9 @@
1
+ import { createFormattedValueComponent } from '../../../components/FormattedValueComponent';
2
+ import { mapCommonSchemaProps } from '../utils/mapCommonSchemaProps';
3
+ import { getSchemaOnChange } from '../../../features/schema-on-change/getSchemaOnChange';
4
+ export const objectSchemaToFormattedValueComponent = (schemaMapperProps, mapperProps) => {
5
+ const { onBehavior } = mapperProps;
6
+ const { schema } = schemaMapperProps;
7
+ const { format } = schema;
8
+ return createFormattedValueComponent(Object.assign(Object.assign({}, mapCommonSchemaProps(schemaMapperProps)), { format, value: schemaMapperProps.model, schemaOnChange: getSchemaOnChange(schema, onBehavior) }), mapperProps.onComponentUpdate);
9
+ };
@@ -0,0 +1,119 @@
1
+ import { isConstSchema, isOneOfSchema, isStringSchema } from '../../../../utils/type-utils';
2
+ import { createMoneyInputComponent } from '../../../components/MoneyInputComponent';
3
+ import { getAboveMaximumMoneyAmountCheck, getBelowMinimumMoneyAmountCheck, } from '../../../features/validation/value-checks';
4
+ import { mapCommonSchemaProps } from '../utils/mapCommonSchemaProps';
5
+ import { createComponentMap } from './objectSchemaToObjectComponent';
6
+ import { isObject } from '../../../../utils/type-validators';
7
+ // Constraints for amount schema
8
+ const isAmountSchema = (schema) => isStringSchema(schema) &&
9
+ !schema.hidden &&
10
+ schema.format !== 'date' &&
11
+ schema.format !== 'base64url' &&
12
+ schema.persistAsync === undefined;
13
+ // Constraints for currency schema
14
+ const isCurrencyConstSchema = (schema) => isConstSchema(schema);
15
+ // A oneOf schema is valid for currency if all its options are const schemas, and there is at least one option
16
+ const isCurrencySelectSchema = (schema) => isOneOfSchema(schema) &&
17
+ schema.oneOf.length > 0 &&
18
+ schema.oneOf.every((option) => isCurrencyConstSchema(option));
19
+ export const objectSchemaToMoneyInputComponent = (schemaMapperProps, mapperProps) => {
20
+ return createMoneyInputComponent(Object.assign(Object.assign({}, mapCommonSchemaProps(schemaMapperProps)), createMoneyInputSubComponents(schemaMapperProps, mapperProps)), mapperProps.onComponentUpdate);
21
+ };
22
+ const createMoneyInputSubComponents = (schemaMapperProps, mapperProps) => {
23
+ const { schema: objectSchema, model } = schemaMapperProps;
24
+ const amountKey = getAmountSchemaKey(objectSchema, mapperProps);
25
+ const currencyKey = getCurrencySchemaKey(objectSchema, mapperProps);
26
+ // This is to ensure that the currency schema is always a oneOf schema,
27
+ // so that it is always mapped to a SelectInputComponent.
28
+ const currencySchema = objectSchema.properties[currencyKey];
29
+ const customSchemaMapperProps = Object.assign(Object.assign({}, schemaMapperProps), { model: getSanitizedModel(amountKey, model), schema: isOneOfSchema(currencySchema)
30
+ ? objectSchema
31
+ : replaceKeyInObjectSchema(objectSchema, currencyKey, { oneOf: [currencySchema] }) });
32
+ const componentMap = createComponentMap(customSchemaMapperProps, mapperProps, 'money');
33
+ const amountComponent = componentMap[amountKey];
34
+ assertAmountComponent(amountComponent, mapperProps.logEvent);
35
+ const currencyComponent = componentMap[currencyKey];
36
+ assertCurrencyComponent(currencyComponent, mapperProps.logEvent);
37
+ const extraValues = getExtraValues(componentMap, [amountKey, currencyKey]);
38
+ const checks = getMinMaxChecks(objectSchema, amountKey, mapperProps);
39
+ return { amountKey, amountComponent, currencyKey, currencyComponent, extraValues, checks };
40
+ };
41
+ const isNumberString = (value) => {
42
+ return (typeof value === 'string' &&
43
+ !Number.isNaN(Number.parseFloat(value)) &&
44
+ /^\d*(?:\.\d*)?$/.test(value));
45
+ };
46
+ const getSanitizedModel = (key, value) => {
47
+ if (isObject(value)) {
48
+ const amount = value[key];
49
+ if (!isNumberString(amount)) {
50
+ return Object.assign(Object.assign({}, value), { [key]: null });
51
+ }
52
+ }
53
+ return value;
54
+ };
55
+ const getAmountSchemaKey = ({ displayOrder, properties }, mapperProps) => {
56
+ var _a;
57
+ const entry = displayOrder
58
+ .map((key) => ({ key, schema: properties[key] }))
59
+ .find(({ schema }) => isAmountSchema(schema));
60
+ if (!entry) {
61
+ const message = `Object schema with format "money" is not properly configured. It must have one schema property of type "string" which is not hidden and has a valid format or no format at all.`;
62
+ (_a = mapperProps.logEvent) === null || _a === void 0 ? void 0 : _a.call(mapperProps, 'error', message);
63
+ throw new Error(message);
64
+ }
65
+ return entry.key;
66
+ };
67
+ const getCurrencySchemaKey = ({ displayOrder, properties }, mapperProps) => {
68
+ var _a;
69
+ const entry = displayOrder
70
+ .map((key) => ({ key, schema: properties[key] }))
71
+ .find(({ schema }) => isCurrencyConstSchema(schema) || isCurrencySelectSchema(schema));
72
+ if (!entry) {
73
+ const message = `Object schema with format "money" is not properly configured. It must have one schema property representing the currency code. This property must be a const schema or a oneOf schema with at least one const schema options.`;
74
+ (_a = mapperProps.logEvent) === null || _a === void 0 ? void 0 : _a.call(mapperProps, 'error', message);
75
+ throw new Error(message);
76
+ }
77
+ return entry.key;
78
+ };
79
+ // Returns a new ObjectSchema with the specified key replaced by the new schema
80
+ const replaceKeyInObjectSchema = (objSchema, key, newSchema) => {
81
+ return Object.assign(Object.assign({}, objSchema), { properties: Object.assign(Object.assign({}, objSchema.properties), { [key]: newSchema }) });
82
+ };
83
+ const getMinMaxChecks = (objectSchema, amountKey, mapperProps) => {
84
+ const amountSchema = objectSchema.properties[amountKey];
85
+ const errorMessageFunctions = mapperProps.getErrorMessageFunctions(amountSchema.validationMessages);
86
+ const checks = [
87
+ getAboveMaximumMoneyAmountCheck(amountSchema, errorMessageFunctions),
88
+ getBelowMinimumMoneyAmountCheck(amountSchema, errorMessageFunctions),
89
+ ];
90
+ return checks;
91
+ };
92
+ // Returns values of schema components that are either const or hidden,
93
+ // excluding the ones with keys in excludeKeys (typically, amount and currency).
94
+ function getExtraValues(componentMap, excludeKeys) {
95
+ return Object.fromEntries(Object.entries(componentMap)
96
+ .filter(([key, component]) => !excludeKeys.includes(key) && (component.type === 'const' || component.hidden))
97
+ .map(([key, component]) => [key, component.getSubmittableValueSync()]));
98
+ }
99
+ function assertAmountComponent(component, logEvent) {
100
+ if (component.type !== 'text') {
101
+ // This should never happen because isAmountSchema ensures that the amount schema
102
+ // is a string schema of without hidden, date or base64url format,
103
+ // and those are always mapped to TextInputComponents.
104
+ const message = 'A string schema was selected for amount but it was not successfully mapped into a TextInputComponent.';
105
+ logEvent === null || logEvent === void 0 ? void 0 : logEvent('error', message);
106
+ throw new Error(message);
107
+ }
108
+ }
109
+ function assertCurrencyComponent(component, logEvent) {
110
+ if (component.type !== 'select') {
111
+ // This should never happen because
112
+ // oneOfSchemaToComponent always returns a SelectInputComponent
113
+ // for oneOf schemas with const options, and
114
+ // we have already asserted that the oneOf schema only has const options.
115
+ const message = 'A schema was selected for currency but it was not successfully mapped into a SelectInputComponent.';
116
+ logEvent === null || logEvent === void 0 ? void 0 : logEvent('error', message);
117
+ throw new Error(message);
118
+ }
119
+ }
@@ -0,0 +1,96 @@
1
+ import { objectSchemaToMoneyInputComponent } from './objectSchemaToMoneyInputComponent';
2
+ describe('objectSchemaToMoneyInputComponent', () => {
3
+ const schemaMapperProps = {
4
+ uid: 'test-money-input',
5
+ model: { amount: '100' },
6
+ localValue: null,
7
+ validationErrors: null,
8
+ };
9
+ const mapperProps = {
10
+ logEvent: () => { },
11
+ getErrorMessageFunctions: () => () => '',
12
+ };
13
+ describe('when the object schema has "format": "money"', () => {
14
+ describe('and it is properly configured', () => {
15
+ it('should map to a MoneyInputComponent', () => {
16
+ const moneyInputComponent = objectSchemaToMoneyInputComponent(Object.assign(Object.assign({}, schemaMapperProps), { schema: {
17
+ type: 'object',
18
+ format: 'money',
19
+ properties: {
20
+ amount: { type: 'string', format: 'numeric' },
21
+ currency: { const: 'USD' },
22
+ },
23
+ displayOrder: ['amount', 'currency'],
24
+ } }), mapperProps);
25
+ expect(moneyInputComponent.type).toBe('money-input');
26
+ });
27
+ });
28
+ describe('but it is not properly configured', () => {
29
+ it('should throw when missing string schema (for amount)', () => {
30
+ expect(() => objectSchemaToMoneyInputComponent(Object.assign(Object.assign({}, schemaMapperProps), { schema: {
31
+ type: 'object',
32
+ format: 'money',
33
+ properties: {
34
+ amount: { type: 'number' },
35
+ currency: { const: 'USD' },
36
+ },
37
+ displayOrder: ['amount', 'currency'],
38
+ } }), mapperProps)).toThrow(/not properly configured.*string/);
39
+ });
40
+ it('should throw when missing a usable string schema (hidden schemas cannot be used)', () => {
41
+ expect(() => objectSchemaToMoneyInputComponent(Object.assign(Object.assign({}, schemaMapperProps), { schema: {
42
+ type: 'object',
43
+ format: 'money',
44
+ properties: {
45
+ amount: { type: 'string', hidden: true },
46
+ currency: { const: 'USD' },
47
+ },
48
+ displayOrder: ['amount', 'currency'],
49
+ } }), mapperProps)).toThrow(/not properly configured.*string/);
50
+ });
51
+ it('should throw when missing a usable string schema (format: "date" cannot be used)', () => {
52
+ expect(() => objectSchemaToMoneyInputComponent(Object.assign(Object.assign({}, schemaMapperProps), { schema: {
53
+ type: 'object',
54
+ format: 'money',
55
+ properties: {
56
+ amount: { type: 'string', format: 'date' },
57
+ currency: { const: 'USD' },
58
+ },
59
+ displayOrder: ['amount', 'currency'],
60
+ } }), mapperProps)).toThrow(/not properly configured.*string/);
61
+ });
62
+ it('should throw when missing a usable string schema (upload schemas cannot be used)', () => {
63
+ expect(() => objectSchemaToMoneyInputComponent(Object.assign(Object.assign({}, schemaMapperProps), { schema: {
64
+ type: 'object',
65
+ format: 'money',
66
+ properties: {
67
+ amount: { type: 'string', format: 'base64url' },
68
+ currency: { const: 'USD' },
69
+ },
70
+ displayOrder: ['amount', 'currency'],
71
+ } }), mapperProps)).toThrow(/not properly configured.*string/);
72
+ });
73
+ it('should throw when missing currency schema', () => {
74
+ expect(() => objectSchemaToMoneyInputComponent(Object.assign(Object.assign({}, schemaMapperProps), { schema: {
75
+ type: 'object',
76
+ format: 'money',
77
+ properties: {
78
+ amount: { type: 'string' },
79
+ },
80
+ displayOrder: ['amount'],
81
+ } }), mapperProps)).toThrow(/not properly configured.*currency code/);
82
+ });
83
+ it('should throw when currency schema oneOf is empty', () => {
84
+ expect(() => objectSchemaToMoneyInputComponent(Object.assign(Object.assign({}, schemaMapperProps), { schema: {
85
+ type: 'object',
86
+ format: 'money',
87
+ properties: {
88
+ amount: { type: 'string', format: 'numeric' },
89
+ currency: { oneOf: [] },
90
+ },
91
+ displayOrder: ['amount', 'currency'],
92
+ } }), mapperProps)).toThrow(/not properly configured.*const.*oneOf/);
93
+ });
94
+ });
95
+ });
96
+ });
@@ -0,0 +1,31 @@
1
+ import { isObjectLocalValue, isObjectModel } from '../../../../utils/type-utils';
2
+ import { createObjectComponent } from '../../../components/ObjectComponent';
3
+ import { mapSchemaToComponent } from '../../mapSchemaToComponent';
4
+ import { isObject } from '../../../../utils/type-validators';
5
+ import { mapCommonSchemaProps } from '../utils/mapCommonSchemaProps';
6
+ import { assertDisplayOrder } from './assertDisplayOrder';
7
+ export const objectSchemaToObjectComponent = (schemaMapperProps, mapperProps) => {
8
+ assertDisplayOrder(schemaMapperProps.schema, mapperProps.logEvent);
9
+ const componentMap = createComponentMap(schemaMapperProps, mapperProps, 'obj');
10
+ return createObjectComponent(Object.assign(Object.assign({}, mapCommonSchemaProps(schemaMapperProps)), { displayOrder: schemaMapperProps.schema.displayOrder, componentMap }));
11
+ };
12
+ export const createComponentMap = (schemaMapperProps, mapperProps, uidPrefix) => {
13
+ const { uid, localValue, schema, model, validationErrors } = schemaMapperProps;
14
+ const { $id, displayOrder, properties, required } = schema;
15
+ const initialReducerValue = {};
16
+ return displayOrder.reduce((acc, propName) => {
17
+ var _a;
18
+ const propSchema = properties[propName];
19
+ if (propSchema === undefined) {
20
+ throw new Error(`Object schema ${$id} has no property named ${propName}`);
21
+ }
22
+ return Object.assign(Object.assign({}, acc), { [propName]: mapSchemaToComponent({
23
+ uid: `${uid}.${uidPrefix}-${propName}`,
24
+ schema: propSchema,
25
+ model: isObjectModel(model) ? model[propName] : null,
26
+ localValue: isObjectLocalValue(localValue) ? localValue[propName] : null,
27
+ required: (_a = required === null || required === void 0 ? void 0 : required.includes(propName)) !== null && _a !== void 0 ? _a : false,
28
+ validationErrors: isObject(validationErrors) ? validationErrors[propName] : undefined,
29
+ }, mapperProps) });
30
+ }, initialReducerValue);
31
+ };
@@ -0,0 +1,99 @@
1
+ import { getMockMapperProps, getMockSchemaMapperProps } from '../tests/test-utils';
2
+ import { objectSchemaToObjectComponent } from './objectSchemaToObjectComponent';
3
+ describe('objectSchemaToObjectComponent', () => {
4
+ const defaultMapperProps = getMockMapperProps();
5
+ describe('initial model handling', () => {
6
+ describe('when the initial model is null', () => {
7
+ it('sets component value to an empty object', async () => {
8
+ const schemaMapperProps = getMockSchemaMapperProps({
9
+ schema: {
10
+ type: 'object',
11
+ properties: {},
12
+ displayOrder: [],
13
+ },
14
+ model: null,
15
+ });
16
+ const component = objectSchemaToObjectComponent(schemaMapperProps, defaultMapperProps);
17
+ expect(await component.getSubmittableValue()).toStrictEqual({});
18
+ });
19
+ it('is still an empty object if child values have not been set', async () => {
20
+ const schemaMapperProps = getMockSchemaMapperProps({
21
+ schema: {
22
+ type: 'object',
23
+ properties: {
24
+ name: {
25
+ type: 'string',
26
+ },
27
+ occupation: {
28
+ type: 'string',
29
+ },
30
+ },
31
+ displayOrder: ['name', 'occupation'],
32
+ },
33
+ model: null,
34
+ });
35
+ const component = objectSchemaToObjectComponent(schemaMapperProps, defaultMapperProps);
36
+ expect(await component.getSubmittableValue()).toStrictEqual({});
37
+ });
38
+ });
39
+ describe('when initial model is defined', () => {
40
+ const schemaMapperProps = getMockSchemaMapperProps({
41
+ schema: {
42
+ type: 'object',
43
+ properties: {
44
+ name: {
45
+ type: 'string',
46
+ },
47
+ occupation: {
48
+ type: 'string',
49
+ },
50
+ },
51
+ displayOrder: ['name', 'occupation'],
52
+ },
53
+ model: {
54
+ name: 'John Smith',
55
+ city: 'New York',
56
+ },
57
+ });
58
+ it('sets the component value to the initial model', async () => {
59
+ const component = objectSchemaToObjectComponent(schemaMapperProps, defaultMapperProps);
60
+ expect(await component.getSubmittableValue()).toStrictEqual(expect.objectContaining({
61
+ name: 'John Smith',
62
+ }));
63
+ });
64
+ });
65
+ });
66
+ describe('initial validation error handling', () => {
67
+ it('sets the initial error if provided', () => {
68
+ const schemaMapperProps = getMockSchemaMapperProps({
69
+ schema: {
70
+ type: 'object',
71
+ properties: {
72
+ name: {
73
+ type: 'string',
74
+ },
75
+ occupation: {
76
+ type: 'string',
77
+ },
78
+ },
79
+ displayOrder: ['name', 'occupation'],
80
+ },
81
+ model: {
82
+ name: 'Jane Doe',
83
+ occupation: 'Programmer',
84
+ },
85
+ validationErrors: {
86
+ name: 'Problem with this name',
87
+ occupation: 'Problem with this occupation',
88
+ },
89
+ });
90
+ const component = objectSchemaToObjectComponent(schemaMapperProps, defaultMapperProps);
91
+ expect(component.componentMap.name.errors).toStrictEqual([
92
+ 'Problem with this name',
93
+ ]);
94
+ expect(component.componentMap.occupation.errors).toStrictEqual([
95
+ 'Problem with this occupation',
96
+ ]);
97
+ });
98
+ });
99
+ });
@@ -0,0 +1,66 @@
1
+ var __rest = (this && this.__rest) || function (s, e) {
2
+ var t = {};
3
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
4
+ t[p] = s[p];
5
+ if (s != null && typeof Object.getOwnPropertySymbols === "function")
6
+ for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
7
+ if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
8
+ t[p[i]] = s[p[i]];
9
+ }
10
+ return t;
11
+ };
12
+ import { isAllOfSchema, isArrayTupleSchema, isObjectSchema } from '../../../../utils/type-utils';
13
+ import { createSelectInputComponent } from '../../../components/SelectInputComponent';
14
+ import { getSchemaOnChange } from '../../../features/schema-on-change/getSchemaOnChange';
15
+ import { getRequiredCheck } from '../../../features/validation/value-checks';
16
+ import { mapSchemaToComponent } from '../../mapSchemaToComponent';
17
+ import { getAutocompleteString } from '../../utils/getAutocompleteString';
18
+ import { getDomainLayerMedia } from '../../utils/media-utils';
19
+ import { mapInlineAlert } from '../../utils/utils';
20
+ import { mapCommonSchemaProps } from '../utils/mapCommonSchemaProps';
21
+ export const oneOfSchemaToComponent = (schemaMapperProps, mapperProps) => {
22
+ var _a;
23
+ const { uid, localValue, schema, model, validationErrors: initialError, required = false, } = schemaMapperProps;
24
+ const initialModel = (_a = model !== null && model !== void 0 ? model : schema.default) !== null && _a !== void 0 ? _a : null;
25
+ const options = schema.oneOf.map((childSchema, index) => {
26
+ var _a;
27
+ const { title = '', description, image, icon, media, keywords = [], tags, additionalText, inlineAlert, supportingValues, } = childSchema;
28
+ return {
29
+ title,
30
+ description,
31
+ additionalText,
32
+ analyticsId: (_a = childSchema.analyticsId) !== null && _a !== void 0 ? _a : childSchema.$id,
33
+ disabled: 'disabled' in childSchema ? Boolean(childSchema.disabled) : false,
34
+ inlineAlert: mapInlineAlert(inlineAlert),
35
+ keywords,
36
+ media: getDomainLayerMedia({ icon, image, media }),
37
+ supportingValues,
38
+ tags,
39
+ component: mapSchemaToComponent({
40
+ uid: `${uid}.oneOf-${index}`,
41
+ schema: isFormSectionSchema(childSchema)
42
+ ? supressSchemaTitleAndDescription(childSchema)
43
+ : childSchema,
44
+ model: initialModel,
45
+ localValue,
46
+ validationErrors: initialError,
47
+ required,
48
+ }, mapperProps),
49
+ };
50
+ });
51
+ const { getErrorMessageFunctions, onComponentUpdate, trackEvent, onBehavior, onValueChange } = mapperProps;
52
+ const { validationMessages } = schema;
53
+ const errorMessageFunctions = getErrorMessageFunctions(validationMessages);
54
+ return createSelectInputComponent(Object.assign(Object.assign({}, mapCommonSchemaProps(schemaMapperProps)), { autoComplete: getAutocompleteString(schema.autocompleteHint), checks: schema.hidden ? [] : [getRequiredCheck(required, errorMessageFunctions)], options,
55
+ initialModel, schemaOnChange: getSchemaOnChange(schema, onBehavior), onValueChange,
56
+ trackEvent }), onComponentUpdate);
57
+ };
58
+ const isFormSectionSchema = (schema) => isObjectSchema(schema) || isArrayTupleSchema(schema) || isAllOfSchema(schema);
59
+ /**
60
+ * When using a schema in a oneOf, the title is not needed, as it is already displayed in the dropdown, radio or tab.
61
+ * We also suppress the description, because it is not displayed on mobile clients.
62
+ */
63
+ const supressSchemaTitleAndDescription = (schema) => {
64
+ const { title, description } = schema, headlessSchema = __rest(schema, ["title", "description"]);
65
+ return headlessSchema;
66
+ };
@@ -0,0 +1,225 @@
1
+ import { getMockMapperProps, getMockSchemaMapperProps } from '../tests/test-utils';
2
+ import { oneOfSchemaToComponent } from './oneOfSchemaToComponent';
3
+ describe('oneOfSchemaToComponent', () => {
4
+ describe('initial model handling', () => {
5
+ describe('when initial model is null', () => {
6
+ it('sets component value to null', async () => {
7
+ const schemaMapperProps = getMockSchemaMapperProps({
8
+ schema: {
9
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
10
+ },
11
+ model: null,
12
+ });
13
+ const component = oneOfSchemaToComponent(schemaMapperProps, getMockMapperProps());
14
+ expect(await component.getSubmittableValue()).toBeNull();
15
+ });
16
+ });
17
+ });
18
+ describe('when a default is provided', () => {
19
+ it('initialises the value to the default', async () => {
20
+ const schemaMapperProps = getMockSchemaMapperProps({
21
+ schema: {
22
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
23
+ default: 'USD',
24
+ },
25
+ model: null,
26
+ });
27
+ const component = oneOfSchemaToComponent(schemaMapperProps, getMockMapperProps());
28
+ expect(await component.getSubmittableValue()).toBe('USD');
29
+ });
30
+ });
31
+ describe('when initial model is defined', () => {
32
+ it('sets the component value to the initial model', async () => {
33
+ const schemaMapperProps = getMockSchemaMapperProps({
34
+ schema: {
35
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
36
+ },
37
+ model: 'USD',
38
+ });
39
+ const component = oneOfSchemaToComponent(schemaMapperProps, getMockMapperProps());
40
+ expect(await component.getSubmittableValue()).toBe('USD');
41
+ });
42
+ });
43
+ describe('when a default and a model are provided', () => {
44
+ it('prefers the model over the default', async () => {
45
+ const schemaMapperProps = getMockSchemaMapperProps({
46
+ schema: {
47
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
48
+ default: 'EUR',
49
+ },
50
+ model: 'USD',
51
+ });
52
+ const component = oneOfSchemaToComponent(schemaMapperProps, getMockMapperProps());
53
+ expect(await component.getSubmittableValue()).toBe('USD');
54
+ });
55
+ });
56
+ describe('initial validation error handling', () => {
57
+ it('sets the initial error if provided', () => {
58
+ const schemaMapperProps = getMockSchemaMapperProps({
59
+ schema: {
60
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
61
+ },
62
+ model: null,
63
+ validationErrors: 'You must select a currency',
64
+ });
65
+ const component = oneOfSchemaToComponent(schemaMapperProps, getMockMapperProps());
66
+ expect(component.errors).toEqual(['You must select a currency']);
67
+ });
68
+ });
69
+ describe('validation checks', () => {
70
+ describe('when required is true but the model is null', () => {
71
+ it('fails validation', () => {
72
+ const schemaMapperProps = Object.assign(Object.assign({}, getMockSchemaMapperProps({
73
+ schema: {
74
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
75
+ },
76
+ model: null,
77
+ })), { required: true });
78
+ const component = oneOfSchemaToComponent(schemaMapperProps, getMockMapperProps());
79
+ expect(component.validate()).toBe(false);
80
+ expect(component.errors).toStrictEqual(['required']);
81
+ });
82
+ });
83
+ describe('when required is true but the model is not null', () => {
84
+ it('passes validation', () => {
85
+ const schemaMapperProps = Object.assign(Object.assign({}, getMockSchemaMapperProps({
86
+ schema: {
87
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
88
+ },
89
+ model: 'EUR',
90
+ })), { required: true });
91
+ const component = oneOfSchemaToComponent(schemaMapperProps, getMockMapperProps());
92
+ expect(component.validate()).toBe(true);
93
+ });
94
+ });
95
+ });
96
+ describe('analytics', () => {
97
+ describe('when selecting an option that is a const schema', () => {
98
+ it('triggers an event', () => {
99
+ const schemaMapperProps = getMockSchemaMapperProps({
100
+ schema: {
101
+ analyticsId: 'currency',
102
+ oneOf: [
103
+ { const: 'EUR', analyticsId: 'EUR' },
104
+ { const: 'USD', analyticsId: 'USD' },
105
+ { const: 'GBP', analyticsId: 'GBP' },
106
+ ],
107
+ },
108
+ model: null,
109
+ });
110
+ const defaultMapperProps = getMockMapperProps();
111
+ const component = oneOfSchemaToComponent(schemaMapperProps, defaultMapperProps);
112
+ component.onSelect(1);
113
+ expect(defaultMapperProps.trackEvent).toHaveBeenCalledWith('OneOf Selected', {
114
+ oneOfId: 'currency',
115
+ schemaId: 'USD',
116
+ });
117
+ });
118
+ it('uses $id in the absence of analyticsId', () => {
119
+ const schemaMapperProps = getMockSchemaMapperProps({
120
+ schema: {
121
+ $id: 'currency',
122
+ oneOf: [
123
+ { const: 'EUR', $id: 'EUR' },
124
+ { const: 'USD', $id: 'USD' },
125
+ { const: 'GBP', $id: 'GBP' },
126
+ ],
127
+ },
128
+ model: null,
129
+ });
130
+ const defaultMapperProps = getMockMapperProps();
131
+ const component = oneOfSchemaToComponent(schemaMapperProps, defaultMapperProps);
132
+ component.onSelect(1);
133
+ expect(defaultMapperProps.trackEvent).toHaveBeenCalledWith('OneOf Selected', {
134
+ oneOfId: 'currency',
135
+ schemaId: 'USD',
136
+ });
137
+ });
138
+ it('favours analyticsId over $id', () => {
139
+ const schemaMapperProps = getMockSchemaMapperProps({
140
+ schema: {
141
+ analyticsId: 'currency',
142
+ $id: 'currency-$id',
143
+ oneOf: [
144
+ { const: 'EUR', analyticsId: 'EUR', $id: 'EUR-$id' },
145
+ { const: 'USD', analyticsId: 'USD', $id: 'USD-$id' },
146
+ { const: 'GBP', analyticsId: 'GBP', $id: 'GBP-$id' },
147
+ ],
148
+ },
149
+ model: null,
150
+ });
151
+ const defaultMapperProps = getMockMapperProps();
152
+ const component = oneOfSchemaToComponent(schemaMapperProps, defaultMapperProps);
153
+ component.onSelect(1);
154
+ expect(defaultMapperProps.trackEvent).toHaveBeenCalledWith('OneOf Selected', {
155
+ oneOfId: 'currency',
156
+ schemaId: 'USD',
157
+ });
158
+ });
159
+ it(`doesn't trigger a 'OneOf Selected' event if the main schema doesn't have an analyticsId or $id, but it triggers 'OneOf Option Selected'`, () => {
160
+ const schemaMapperProps = getMockSchemaMapperProps({
161
+ schema: {
162
+ oneOf: [
163
+ { const: 'EUR', analyticsId: 'EUR', $id: 'EUR-$id' },
164
+ { const: 'USD', analyticsId: 'USD', $id: 'USD-$id' },
165
+ { const: 'GBP', analyticsId: 'GBP', $id: 'GBP-$id' },
166
+ ],
167
+ },
168
+ model: null,
169
+ });
170
+ const defaultMapperProps = getMockMapperProps();
171
+ const component = oneOfSchemaToComponent(schemaMapperProps, defaultMapperProps);
172
+ component.onSelect(1);
173
+ expect(defaultMapperProps.trackEvent).not.toHaveBeenCalledWith('OneOf Selected', expect.objectContaining({ oneOfId: '', schemaId: 'USD' }));
174
+ expect(defaultMapperProps.trackEvent).toHaveBeenCalledWith('OneOf Option Selected', expect.objectContaining({ selectedOption: 'USD', schema: '' }));
175
+ });
176
+ it('does trigger an event if the selected option doesnt have an analyticsId or $id', () => {
177
+ const schemaMapperProps = getMockSchemaMapperProps({
178
+ schema: {
179
+ analyticsId: 'currency',
180
+ oneOf: [{ const: 'EUR' }, { const: 'USD' }, { const: 'GBP' }],
181
+ },
182
+ model: null,
183
+ });
184
+ const defaultMapperProps = getMockMapperProps();
185
+ const component = oneOfSchemaToComponent(schemaMapperProps, defaultMapperProps);
186
+ component.onSelect(1);
187
+ expect(defaultMapperProps.trackEvent).toHaveBeenCalledWith('OneOf Selected', {
188
+ oneOfId: 'currency',
189
+ schemaId: undefined,
190
+ });
191
+ });
192
+ });
193
+ describe('when selecting an option that is an object schema', () => {
194
+ it('triggers an event', () => {
195
+ const schemaMapperProps = getMockSchemaMapperProps({
196
+ schema: {
197
+ analyticsId: 'options-a-or-b',
198
+ oneOf: [
199
+ {
200
+ type: 'object',
201
+ analyticsId: 'option-A',
202
+ displayOrder: ['name'],
203
+ properties: { name: { type: 'string' } },
204
+ },
205
+ {
206
+ type: 'object',
207
+ analyticsId: 'option-B',
208
+ displayOrder: ['name'],
209
+ properties: { name: { type: 'string' } },
210
+ },
211
+ ],
212
+ },
213
+ model: null,
214
+ });
215
+ const defaultMapperProps = getMockMapperProps();
216
+ const component = oneOfSchemaToComponent(schemaMapperProps, defaultMapperProps);
217
+ component.onSelect(1);
218
+ expect(defaultMapperProps.trackEvent).toHaveBeenCalledWith('OneOf Selected', {
219
+ oneOfId: 'options-a-or-b',
220
+ schemaId: 'option-B',
221
+ });
222
+ });
223
+ });
224
+ });
225
+ });