@wise/dynamic-flow-client 5.8.1 → 5.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/DynamicFlowCore.js +5 -0
- package/build/common/errorBoundary/ErrorBoundary.js +26 -0
- package/build/common/errorBoundary/ErrorBoundary.test.js +29 -0
- package/build/common/errorBoundary/ErrorBoundaryAlert.js +11 -0
- package/build/common/makeHttpClient/api-utils.js +3 -0
- package/build/common/makeHttpClient/index.js +1 -0
- package/build/common/makeHttpClient/makeHttpClient.js +10 -0
- package/build/common/makeHttpClient/makeHttpClient.test.js +186 -0
- package/build/common/messages/external-confirmation.messages.js +23 -0
- package/build/common/messages/file-upload.messages.js +13 -0
- package/build/common/messages/generic-error.messages.js +18 -0
- package/build/common/messages/help.messages.js +8 -0
- package/build/common/messages/multi-file-upload.messages.js +18 -0
- package/build/common/messages/multi-select.messages.js +8 -0
- package/build/common/messages/paragraph.messages.js +13 -0
- package/build/common/messages/repeatable.messages.js +23 -0
- package/build/common/messages/search.messages.js +8 -0
- package/build/common/messages/validation.array.messages.js +13 -0
- package/build/common/messages/validation.messages.js +53 -0
- package/build/controller/FlowController.js +368 -0
- package/build/controller/executePoll.js +49 -0
- package/build/controller/executeRefresh.js +39 -0
- package/build/controller/executeRequest.js +77 -0
- package/build/controller/executeSubmission.js +69 -0
- package/build/controller/getErrorMessage.js +7 -0
- package/build/controller/getRequestAbortController.js +13 -0
- package/build/controller/getResponseType.js +35 -0
- package/build/controller/getSafeHttpClient.js +8 -0
- package/build/controller/getStepCounter.js +16 -0
- package/build/controller/handleErrorResponse.js +26 -0
- package/build/controller/makeSafeHttpClient.js +8 -0
- package/build/controller/response-utils.js +72 -0
- package/build/domain/components/AlertComponent.js +1 -0
- package/build/domain/components/AllOfComponent.js +40 -0
- package/build/domain/components/BooleanInputComponent.js +50 -0
- package/build/domain/components/BoxComponent.js +3 -0
- package/build/domain/components/ButtonComponent.js +1 -0
- package/build/domain/components/ColumnsComponent.js +3 -0
- package/build/domain/components/ConstComponent.js +18 -0
- package/build/domain/components/ContainerComponent.js +3 -0
- package/build/domain/components/DateInputComponent.js +75 -0
- package/build/domain/components/DecisionComponent.js +1 -0
- package/build/domain/components/DividerComponent.js +1 -0
- package/build/domain/components/FormComponent.js +3 -0
- package/build/domain/components/FormattedValueComponent.js +44 -0
- package/build/domain/components/HeadingComponent.js +1 -0
- package/build/domain/components/ImageComponent.js +1 -0
- package/build/domain/components/InstructionsComponent.js +1 -0
- package/build/domain/components/IntegerInputComponent.js +74 -0
- package/build/domain/components/ListComponent.js +1 -0
- package/build/domain/components/LoadingIndicatorComponent.js +1 -0
- package/build/domain/components/MarkdownComponent.js +1 -0
- package/build/domain/components/MediaComponent.js +1 -0
- package/build/domain/components/ModalComponent.js +16 -0
- package/build/domain/components/ModalLayoutComponent.js +3 -0
- package/build/domain/components/MoneyInputComponent.js +57 -0
- package/build/domain/components/MultiSelectInputComponent.js +81 -0
- package/build/domain/components/MultiUploadInputComponent.js +88 -0
- package/build/domain/components/NumberInputComponent.js +73 -0
- package/build/domain/components/ObjectComponent.js +45 -0
- package/build/domain/components/ParagraphComponent.js +1 -0
- package/build/domain/components/PersistAsyncComponent.js +92 -0
- package/build/domain/components/ProgressComponent.js +1 -0
- package/build/domain/components/RepeatableComponent.js +103 -0
- package/build/domain/components/ReviewComponent.js +1 -0
- package/build/domain/components/RootDomainComponent.js +173 -0
- package/build/domain/components/SectionComponent.js +5 -0
- package/build/domain/components/SelectInputComponent.js +88 -0
- package/build/domain/components/StatusListComponent.js +1 -0
- package/build/domain/components/SubflowDomainComponent.js +9 -0
- package/build/domain/components/TabsComponent.js +1 -0
- package/build/domain/components/TextInputComponent.js +76 -0
- package/build/domain/components/TupleComponent.js +41 -0
- package/build/domain/components/UploadInputComponent.js +83 -0
- package/build/domain/components/UpsellComponent.js +25 -0
- package/build/domain/components/searchComponent/SearchComponent.js +92 -0
- package/build/domain/components/searchComponent/SearchComponent.test.js +190 -0
- package/build/domain/components/step/ExternalConfirmationComponent.js +28 -0
- package/build/domain/components/step/StepDomainComponent.js +73 -0
- package/build/domain/components/step/ToolbarComponent.js +1 -0
- package/build/domain/components/utils/WithUpdate.js +1 -0
- package/build/domain/components/utils/component-utils.js +12 -0
- package/build/domain/components/utils/debounce.js +34 -0
- package/build/domain/components/utils/debounce.test.js +67 -0
- package/build/domain/components/utils/file-utils.js +21 -0
- package/build/domain/components/utils/file-utils.test.js +27 -0
- package/build/domain/components/utils/getRandomId.js +1 -0
- package/build/domain/components/utils/isExactLocalValueMatch.js +14 -0
- package/build/domain/components/utils/isOrWasValid.js +5 -0
- package/build/domain/components/utils/isPartialModelMatch.js +18 -0
- package/build/domain/components/utils/isPartialModelMatch.test.js +74 -0
- package/build/domain/features/eventNames.js +24 -0
- package/build/domain/features/events.js +1 -0
- package/build/domain/features/persistAsync/getComponentMultiPersistAsync.js +50 -0
- package/build/domain/features/persistAsync/getInitialPersistedState.js +7 -0
- package/build/domain/features/persistAsync/getPerformPersistAsync.js +43 -0
- package/build/domain/features/persistAsync/getPerformPersistAsync.test.js +139 -0
- package/build/domain/features/polling/getStepPolling.js +43 -0
- package/build/domain/features/polling/getStepPolling.test.js +90 -0
- package/build/domain/features/prefetch/getStepPrefetch.js +43 -0
- package/build/domain/features/prefetch/request-cache.js +49 -0
- package/build/domain/features/prefetch/request-cache.test.js +70 -0
- package/build/domain/features/refreshAfter/getStepRefreshAfter.js +24 -0
- package/build/domain/features/refreshAfter/getStepRefreshAfter.test.js +40 -0
- package/build/domain/features/schema-on-change/getDebouncedSchemaOnChange.js +50 -0
- package/build/domain/features/schema-on-change/getSchemaOnChange.js +34 -0
- package/build/domain/features/search/getPerformSearchFunction.js +75 -0
- package/build/domain/features/search/getPerformSearchFunction.test.js +301 -0
- package/build/domain/features/summary/summary-utils.js +40 -0
- package/build/domain/features/summary/summary-utils.test.js +125 -0
- package/build/domain/features/utils/http-utils.js +21 -0
- package/build/domain/features/utils/response-utils.js +9 -0
- package/build/domain/features/validation/spec-utils.js +19 -0
- package/build/domain/features/validation/validateStringPattern.js +24 -0
- package/build/domain/features/validation/validation-functions.js +6 -0
- package/build/domain/features/validation/validation-functions.test.js +108 -0
- package/build/domain/features/validation/value-checks.js +125 -0
- package/build/domain/features/validation/value-checks.test.js +262 -0
- package/build/domain/features/validationAsync/getComponentValidationAsync.js +53 -0
- package/build/domain/features/validationAsync/getComponentValidationAsync.test.js +67 -0
- package/build/domain/features/validationAsync/getInitialValidationAsyncState.js +5 -0
- package/build/domain/features/validationAsync/getPerformValidationAsync.js +45 -0
- package/build/domain/features/validationAsync/getPerformValidationAsync.test.js +143 -0
- package/build/domain/mappers/layout/alertLayoutToComponent.js +16 -0
- package/build/domain/mappers/layout/boxLayoutToComponent.js +13 -0
- package/build/domain/mappers/layout/buttonLayoutToComponent.js +77 -0
- package/build/domain/mappers/layout/columnsLayoutToComponent.js +13 -0
- package/build/domain/mappers/layout/decisionLayoutToComponent.js +22 -0
- package/build/domain/mappers/layout/deprecatedListLayoutToComponent.js +30 -0
- package/build/domain/mappers/layout/dividerLayoutToComponent.js +2 -0
- package/build/domain/mappers/layout/formLayoutToComponent.js +19 -0
- package/build/domain/mappers/layout/headingLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/imageLayoutToComponent.js +20 -0
- package/build/domain/mappers/layout/infoLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/instructionsLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/listLayoutToComponent.js +39 -0
- package/build/domain/mappers/layout/loadingIndicatorLayoutToComponent.js +9 -0
- package/build/domain/mappers/layout/markdownLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/mediaLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/modalLayoutToComponent.js +17 -0
- package/build/domain/mappers/layout/modalToComponent.js +8 -0
- package/build/domain/mappers/layout/paragraphLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/progressLayoutToComponent.js +15 -0
- package/build/domain/mappers/layout/reviewLayoutToComponent.js +48 -0
- package/build/domain/mappers/layout/searchLayoutToComponent.js +44 -0
- package/build/domain/mappers/layout/sectionLayoutToComponent.js +15 -0
- package/build/domain/mappers/layout/statusListLayoutToComponent.js +15 -0
- package/build/domain/mappers/layout/tabsLayoutToComponent.js +16 -0
- package/build/domain/mappers/layout/upsellLayoutToComponent.js +25 -0
- package/build/domain/mappers/mapLayoutToComponent.js +81 -0
- package/build/domain/mappers/mapSchemaToComponent.js +61 -0
- package/build/domain/mappers/mapSchemaToComponent.test.js +112 -0
- package/build/domain/mappers/mapStepSchemas.js +17 -0
- package/build/domain/mappers/mapStepToComponent.js +132 -0
- package/build/domain/mappers/mapStepToComponent.test.js +221 -0
- package/build/domain/mappers/mapToolbarToComponent.js +15 -0
- package/build/domain/mappers/schema/allOfSchemaToComponent.js +16 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToComponent.js +26 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiSelectComponent.js +55 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiUploadComponent.js +67 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToRepeatableComponent.js +57 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToTupleComponent.js +20 -0
- package/build/domain/mappers/schema/blobSchemaToComponent.js +15 -0
- package/build/domain/mappers/schema/booleanSchemaToComponent.js +29 -0
- package/build/domain/mappers/schema/constSchemaToComponent.js +23 -0
- package/build/domain/mappers/schema/integerSchemaToComponent.js +28 -0
- package/build/domain/mappers/schema/numberSchemaToComponent.js +26 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/assertDisplayOrder.js +23 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToFormattedValueComponent.js +9 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.js +119 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.test.js +96 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.js +31 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.test.js +99 -0
- package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.js +76 -0
- package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.test.js +265 -0
- package/build/domain/mappers/schema/persistAsyncSchemaToComponent.js +29 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.js +18 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.test.js +133 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToDateInputComponent.js +48 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToTextInputComponent.js +37 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToUploadInputComponent.js +28 -0
- package/build/domain/mappers/schema/tests/test-utils.js +16 -0
- package/build/domain/mappers/schema/types.js +1 -0
- package/build/domain/mappers/schema/utils/getPerformPersistAsyncFn.js +19 -0
- package/build/domain/mappers/schema/utils/getValidationAsyncInitialState.js +23 -0
- package/build/domain/mappers/schema/utils/mapCommonSchemaProps.js +16 -0
- package/build/domain/mappers/types.js +1 -0
- package/build/domain/mappers/utils/FeatureFlags.js +22 -0
- package/build/domain/mappers/utils/behavior-utils.js +44 -0
- package/build/domain/mappers/utils/call-to-action-utils.js +21 -0
- package/build/domain/mappers/utils/getAutocompleteString.js +76 -0
- package/build/domain/mappers/utils/getAutocompleteString.test.js +21 -0
- package/build/domain/mappers/utils/groupLayoutByPinned.js +38 -0
- package/build/domain/mappers/utils/groupLayoutByPinned.test.js +166 -0
- package/build/domain/mappers/utils/image.js +9 -0
- package/build/domain/mappers/utils/layout-utils.js +11 -0
- package/build/domain/mappers/utils/legacy-utils.js +49 -0
- package/build/domain/mappers/utils/media-utils.js +14 -0
- package/build/domain/mappers/utils/suggestions-utils.js +26 -0
- package/build/domain/mappers/utils/suggestions-utils.test.js +36 -0
- package/build/domain/mappers/utils/tags-utils.js +1 -0
- package/build/domain/mappers/utils/utils.js +35 -0
- package/build/domain/types.js +1 -0
- package/build/getSubflowCallbacks.js +38 -0
- package/build/i18n/index.js +40 -0
- package/build/index.js +8 -0
- package/build/main.js +112 -54
- package/build/main.mjs +111 -53
- package/build/renderers/CoreContainerRenderer.js +5 -0
- package/build/renderers/CoreRootRenderer.js +12 -0
- package/build/renderers/EmptyLoadingStateRenderer.js +5 -0
- package/build/renderers/getRenderFunction.js +24 -0
- package/build/renderers/getSchemaErrorMessageFunction.js +97 -0
- package/build/renderers/mappers/alertComponentToProps.js +2 -0
- package/build/renderers/mappers/allOfComponentToProps.js +6 -0
- package/build/renderers/mappers/booleanInputComponentToProps.js +5 -0
- package/build/renderers/mappers/boxComponentToProps.js +13 -0
- package/build/renderers/mappers/buttonComponentToProps.js +4 -0
- package/build/renderers/mappers/columnsComponentToProps.js +11 -0
- package/build/renderers/mappers/componentToRendererProps.js +164 -0
- package/build/renderers/mappers/constComponentToProps.js +5 -0
- package/build/renderers/mappers/containerComponentToProps.js +7 -0
- package/build/renderers/mappers/dateInputComponentToProps.js +2 -0
- package/build/renderers/mappers/decisionComponentToProps.js +16 -0
- package/build/renderers/mappers/dividerComponentToProps.js +2 -0
- package/build/renderers/mappers/externalComponentToProps.js +3 -0
- package/build/renderers/mappers/formComponentToProps.js +12 -0
- package/build/renderers/mappers/formattedValueComponentToProps.js +5 -0
- package/build/renderers/mappers/headingComponentToProps.js +2 -0
- package/build/renderers/mappers/hiddenComponentToProps.js +4 -0
- package/build/renderers/mappers/imageComponentToProps.js +2 -0
- package/build/renderers/mappers/instructionsComponentToProps.js +2 -0
- package/build/renderers/mappers/integerInputComponentToProps.js +2 -0
- package/build/renderers/mappers/listComponentToProps.js +2 -0
- package/build/renderers/mappers/loadingIndicatorComponentToProps.js +2 -0
- package/build/renderers/mappers/markdownComponentToProps.js +2 -0
- package/build/renderers/mappers/mediaComponentToProps.js +2 -0
- package/build/renderers/mappers/modalComponentToProps.js +11 -0
- package/build/renderers/mappers/modalLayoutComponentToProps.js +16 -0
- package/build/renderers/mappers/moneyInputComponentToProps.js +36 -0
- package/build/renderers/mappers/multiSelectComponentToProps.js +23 -0
- package/build/renderers/mappers/multiUploadInputComponentToProps.js +12 -0
- package/build/renderers/mappers/numberInputComponentToProps.js +2 -0
- package/build/renderers/mappers/objectComponentToProps.js +8 -0
- package/build/renderers/mappers/paragraphComponentToProps.js +2 -0
- package/build/renderers/mappers/persistAsyncComponentToProps.js +8 -0
- package/build/renderers/mappers/progressComponentToProps.js +2 -0
- package/build/renderers/mappers/repeatableComponentToProps.js +30 -0
- package/build/renderers/mappers/reviewComponentToProps.js +2 -0
- package/build/renderers/mappers/rootComponentToProps.js +21 -0
- package/build/renderers/mappers/searchComponentToProps.js +57 -0
- package/build/renderers/mappers/sectionComponentToProps.js +6 -0
- package/build/renderers/mappers/selectInputComponentToProps.js +26 -0
- package/build/renderers/mappers/statusListComponentToProps.js +2 -0
- package/build/renderers/mappers/subflowComponentToRendererProps.js +4 -0
- package/build/renderers/mappers/tabsComponentToProps.js +14 -0
- package/build/renderers/mappers/textInputComponentToProps.js +2 -0
- package/build/renderers/mappers/tupleComponentToProps.js +8 -0
- package/build/renderers/mappers/uploadInputComponentToProps.js +8 -0
- package/build/renderers/mappers/upsellComponentToProps.js +2 -0
- package/build/renderers/mappers/utils/getValidationState.js +12 -0
- package/build/renderers/mappers/utils/inputComponentToProps.js +26 -0
- package/build/renderers/mappers/utils/mapErrorsToValidationState.js +9 -0
- package/build/renderers/mappers/utils/pick.js +8 -0
- package/build/renderers/mappers/utils/selectInputOptionsToProps.js +11 -0
- package/build/renderers/stepComponentToProps.js +32 -0
- package/build/renderers/utils.js +69 -0
- package/build/renderers/utils.test.js +70 -0
- package/build/stories/dev-tools/ContainerQueries.story.js +66 -0
- package/build/stories/dev-tools/Debugger.story.js +38 -0
- package/build/stories/dev-tools/FixtureSelect.story.js +23 -0
- package/build/stories/dev-tools/TestServer.story.js +32 -0
- package/build/stories/examples/NativeFlow.story.js +80 -0
- package/build/stories/examples/Recipients.story.js +568 -0
- package/build/stories/spec/behavior/Copy.story.js +59 -0
- package/build/stories/spec/behavior/Modal.story.js +76 -0
- package/build/stories/spec/behavior/Subflow.story.js +267 -0
- package/build/stories/spec/layouts/Decision.story.js +241 -0
- package/build/stories/spec/layouts/Image.story.js +42 -0
- package/build/stories/spec/layouts/Modal.story.js +81 -0
- package/build/stories/spec/layouts/Search.story.js +325 -0
- package/build/stories/spec/layouts/Upsell.story.js +55 -0
- package/build/stories/spec/layouts/button/Button.story.js +100 -0
- package/build/stories/spec/layouts/button/PinnedButton.story.js +81 -0
- package/build/stories/spec/response/ActionResponse.story.js +66 -0
- package/build/stories/spec/schemas/MultiSelect.story.js +148 -0
- package/build/stories/spec/schemas/Upload.story.js +168 -0
- package/build/stories/spec/schemas/const/ConstLayout.story.js +159 -0
- package/build/stories/spec/schemas/const/ObjectConst.story.js +94 -0
- package/build/stories/spec/schemas/features/PersistAsync.story.js +176 -0
- package/build/stories/spec/schemas/features/ValidationAsync.story.js +103 -0
- package/build/stories/spec/schemas/object/FormattedValue.story.js +92 -0
- package/build/stories/spec/schemas/object/MoneyInput.story.js +240 -0
- package/build/stories/spec/schemas/oneOf/OneOfInitialisation.story.js +55 -0
- package/build/stories/spec/step/Controls.story.js +109 -0
- package/build/stories/spec/step/DFModal.story.js +58 -0
- package/build/stories/spec/step/Footer.story.js +70 -0
- package/build/stories/spec/step/Navigation.story.js +20 -0
- package/build/stories/spec/step/Tags.story.js +39 -0
- package/build/stories/spec/step/ToolBar.story.js +60 -0
- package/build/stories/spec/step/features/ErrorHandling.story.js +92 -0
- package/build/stories/spec/step/features/External.story.js +44 -0
- package/build/stories/spec/step/features/Polling.story.js +108 -0
- package/build/stories/spec/step/features/RefreshAfter.story.js +92 -0
- package/build/stories/spec/step/features/refresh/Refresh.story.js +258 -0
- package/build/stories/spec/step/features/refresh/RefreshWithPersistAsync.story.js +958 -0
- package/build/stories/types.js +1 -0
- package/build/stories/utils/fixtureHttpClient.js +70 -0
- package/build/stories/utils/getBasicStep.js +223 -0
- package/build/stories/utils/mockSearchHandler.js +71 -0
- package/build/stories/utils/render-utils.js +41 -0
- package/build/stories/visual-tests/layouts/NotUsingListItem.story.js +17 -0
- package/build/test-utils/DynamicFlowWise.js +32 -0
- package/build/test-utils/DynamicFlowWiseModal.js +34 -0
- package/build/test-utils/NeptuneProviders.js +11 -0
- package/build/test-utils/component-utils.js +5 -0
- package/build/test-utils/fetch-utils.js +45 -0
- package/build/test-utils/getMergedTestRenderers.js +34 -0
- package/build/test-utils/getRandomId.js +1 -0
- package/build/test-utils/index.js +3 -0
- package/build/test-utils/rtl-utils.js +7 -0
- package/build/test-utils/step-utils.js +6 -0
- package/build/test-utils/wait.js +3 -0
- package/build/tests/AlertLayout.test.js +78 -0
- package/build/tests/ArrayTuple.test.js +118 -0
- package/build/tests/ButtonLayout.test.js +308 -0
- package/build/tests/ConstLayout.test.js +95 -0
- package/build/tests/DateInput.test.js +163 -0
- package/build/tests/DecisionLayout.test.js +146 -0
- package/build/tests/DynamicFlow.test.js +147 -0
- package/build/tests/External.test.js +169 -0
- package/build/tests/Flow.test.js +328 -0
- package/build/tests/FormLayout.test.js +28 -0
- package/build/tests/FormattedValue.test.js +107 -0
- package/build/tests/ImageRenderer.test.js +78 -0
- package/build/tests/InitialAction.test.js +179 -0
- package/build/tests/InitialStep.test.js +168 -0
- package/build/tests/InstructionsLayout.test.js +45 -0
- package/build/tests/ListLayout.test.js +145 -0
- package/build/tests/Logging.test.js +53 -0
- package/build/tests/ModalBehavior.test.js +149 -0
- package/build/tests/MoneyInput.test.js +316 -0
- package/build/tests/MultiUpload.test.js +293 -0
- package/build/tests/NativeBack.test.js +267 -0
- package/build/tests/OneOfInitialisation.test.js +571 -0
- package/build/tests/PersistAsync.test.js +653 -0
- package/build/tests/Polling.test.js +617 -0
- package/build/tests/Prefetching.test.js +230 -0
- package/build/tests/RefreshAfter.test.js +63 -0
- package/build/tests/RefreshOnChange.ResponseHandling.test.js +205 -0
- package/build/tests/RefreshOnChange.test.js +233 -0
- package/build/tests/RefreshOnChange.with.Segmented.test.js +348 -0
- package/build/tests/RefreshOnChange.with.Tabs.test.js +358 -0
- package/build/tests/RefreshOnChangePreserve.test.js +224 -0
- package/build/tests/RendererProps.test.js +342 -0
- package/build/tests/Repeatable.test.js +107 -0
- package/build/tests/Rerendering.test.js +67 -0
- package/build/tests/ReviewLayout.test.js +274 -0
- package/build/tests/SchemaOnChange.test.js +133 -0
- package/build/tests/ScrollToError.test.js +217 -0
- package/build/tests/SegmentedControl.test.js +49 -0
- package/build/tests/SingleFileUpload.test.js +88 -0
- package/build/tests/StatusList.test.js +85 -0
- package/build/tests/Subflow.test.js +710 -0
- package/build/tests/Submission.ResponseHandling.test.js +557 -0
- package/build/tests/Submission.merging.test.js +202 -0
- package/build/tests/Submission.test.js +523 -0
- package/build/tests/Tags.test.js +475 -0
- package/build/tests/Upsell.test.js +126 -0
- package/build/tests/ValidationAsync.test.js +295 -0
- package/build/tests/legacy/Actions.test.js +158 -0
- package/build/tests/legacy/BackButton.test.js +114 -0
- package/build/tests/legacy/HiddenSchemas.test.js +246 -0
- package/build/tests/legacy/MultiSelect.test.js +497 -0
- package/build/tests/legacy/MultipleFileUploadSchema.test.js +341 -0
- package/build/tests/legacy/PersistAsync.blob-schema.test.js +224 -0
- package/build/tests/legacy/PersistAsync.string-schema.test.js +211 -0
- package/build/tests/legacy/RefreshStepOnChange.debouncing.test.js +209 -0
- package/build/tests/legacy/RefreshStepOnChange.test.js +424 -0
- package/build/tests/legacy/Search.test.js +437 -0
- package/build/tests/renderers/MultiSelectInputRendererProps.test.js +58 -0
- package/build/tests/renderers/SelectInputRendererProps.test.js +42 -0
- package/build/tests/renderers/TextInputRenderer.test.js +51 -0
- package/build/types/domain/components/UpsellComponent.d.ts +2 -3
- package/build/types/domain/components/UpsellComponent.d.ts.map +1 -1
- package/build/types/domain/components/searchComponent/SearchComponent.d.ts +5 -4
- package/build/types/domain/components/searchComponent/SearchComponent.d.ts.map +1 -1
- package/build/types/domain/features/search/getPerformSearchFunction.d.ts +1 -1
- package/build/types/domain/features/search/getPerformSearchFunction.d.ts.map +1 -1
- package/build/types/domain/mappers/layout/searchLayoutToComponent.d.ts +2 -1
- package/build/types/domain/mappers/layout/searchLayoutToComponent.d.ts.map +1 -1
- package/build/types/domain/mappers/layout/upsellLayoutToComponent.d.ts +1 -1
- package/build/types/domain/mappers/layout/upsellLayoutToComponent.d.ts.map +1 -1
- package/build/types/renderers/EmptyLoadingStateRenderer.d.ts +3 -0
- package/build/types/renderers/EmptyLoadingStateRenderer.d.ts.map +1 -0
- package/build/types/renderers/mappers/rootComponentToProps.d.ts.map +1 -1
- package/build/types/renderers/mappers/searchComponentToProps.d.ts +1 -1
- package/build/types/renderers/mappers/searchComponentToProps.d.ts.map +1 -1
- package/build/types/renderers/mappers/upsellComponentToProps.d.ts +2 -2
- package/build/types/renderers/mappers/upsellComponentToProps.d.ts.map +1 -1
- package/build/types/test-utils/getMergedTestRenderers.d.ts +1 -1
- package/build/types/test-utils/getMergedTestRenderers.d.ts.map +1 -1
- package/build/types/useDynamicFlow.d.ts.map +1 -1
- package/build/types.js +1 -0
- package/build/useDynamicFlow.js +104 -0
- package/build/useDynamicFlowModal.js +58 -0
- package/build/utils/analyse-step.js +14 -0
- package/build/utils/component-utils.js +8 -0
- package/build/utils/component-utils.test.js +113 -0
- package/build/utils/getScrollToTop.js +12 -0
- package/build/utils/normalise-flow-id.js +1 -0
- package/build/utils/normalise-flow-id.test.js +24 -0
- package/build/utils/openLinkInNewTab.js +10 -0
- package/build/utils/recursiveMerge.js +40 -0
- package/build/utils/recursiveMerge.test.js +93 -0
- package/build/utils/type-utils.js +21 -0
- package/build/utils/type-validators.js +11 -0
- package/build/utils/type-validators.test.js +180 -0
- package/build/utils/useStableCallback.js +15 -0
- package/package.json +4 -4
|
@@ -0,0 +1,437 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { act, screen, waitFor } from '@testing-library/react';
|
|
3
|
+
import { userEvent } from '@testing-library/user-event';
|
|
4
|
+
import { renderWithProviders } from '../../test-utils';
|
|
5
|
+
import { abortableDelay, getMockHttpClient } from '../../test-utils/fetch-utils';
|
|
6
|
+
import DynamicFlowWise from '../../test-utils/DynamicFlowWise';
|
|
7
|
+
import { vi } from 'vitest';
|
|
8
|
+
const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
|
|
9
|
+
const getSearchStep = (search = {}) => ({
|
|
10
|
+
id: 'Features/Search',
|
|
11
|
+
title: 'Search Layout',
|
|
12
|
+
schemas: [],
|
|
13
|
+
layout: [
|
|
14
|
+
Object.assign({ type: 'search', analyticsId: 'search-layout-aid', title: 'Title', url: '/search', method: 'GET', param: 'query', emptyMessage: 'No results (sorry!)' }, search),
|
|
15
|
+
],
|
|
16
|
+
});
|
|
17
|
+
const getSearchResults = (query) => [
|
|
18
|
+
{
|
|
19
|
+
type: 'action',
|
|
20
|
+
title: `Title ${query} #1`,
|
|
21
|
+
description: `This is an action search result.`,
|
|
22
|
+
image: {
|
|
23
|
+
accessibilityDescription: 'image from URL',
|
|
24
|
+
url: 'https://example.com/image.png',
|
|
25
|
+
},
|
|
26
|
+
value: {
|
|
27
|
+
id: 'search-result-action-id-01',
|
|
28
|
+
url: `/next-step`,
|
|
29
|
+
method: 'GET',
|
|
30
|
+
},
|
|
31
|
+
},
|
|
32
|
+
{
|
|
33
|
+
type: 'search',
|
|
34
|
+
title: `Title ${query} #2`,
|
|
35
|
+
description: `This is a search search result.`,
|
|
36
|
+
image: {
|
|
37
|
+
accessibilityDescription: 'image from URI',
|
|
38
|
+
uri: 'https://example.com/image.png',
|
|
39
|
+
},
|
|
40
|
+
value: {
|
|
41
|
+
url: `/new-search`,
|
|
42
|
+
method: 'GET',
|
|
43
|
+
param: 'new-query',
|
|
44
|
+
query: 'new query with special characters (!@£$%^&*)',
|
|
45
|
+
},
|
|
46
|
+
},
|
|
47
|
+
];
|
|
48
|
+
const getLegacySearchResponse = (query) => new Response(JSON.stringify({ results: getSearchResults(query) }));
|
|
49
|
+
const getResultsResponse = (query) => new Response(JSON.stringify({ type: 'results', results: getSearchResults(query) }));
|
|
50
|
+
const getLayoutResponse = (query) => new Response(JSON.stringify({
|
|
51
|
+
type: 'layout',
|
|
52
|
+
content: [
|
|
53
|
+
{
|
|
54
|
+
type: 'markdown',
|
|
55
|
+
content: `You've searched for ${query}, but we didn't find anything, so here is an empty state`,
|
|
56
|
+
},
|
|
57
|
+
],
|
|
58
|
+
}));
|
|
59
|
+
describe('DynamicFlow / Search Layout Component', () => {
|
|
60
|
+
const onCompletion = vi.fn();
|
|
61
|
+
const onError = vi.fn();
|
|
62
|
+
const httpClient = vi.fn();
|
|
63
|
+
beforeEach(() => {
|
|
64
|
+
vi.clearAllMocks();
|
|
65
|
+
});
|
|
66
|
+
describe('user interface', () => {
|
|
67
|
+
it('renders a search input with the title as a label', () => {
|
|
68
|
+
const title = 'Search for something';
|
|
69
|
+
const searchStep = getSearchStep({ title });
|
|
70
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: httpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
71
|
+
expect(screen.getByLabelText(title)).toBeInTheDocument();
|
|
72
|
+
});
|
|
73
|
+
it('renders the initialState when it is a layout type', () => {
|
|
74
|
+
const title = 'Search for something';
|
|
75
|
+
const initialState = {
|
|
76
|
+
type: 'layout',
|
|
77
|
+
content: [
|
|
78
|
+
{
|
|
79
|
+
type: 'markdown',
|
|
80
|
+
content: 'This is the initial state',
|
|
81
|
+
},
|
|
82
|
+
],
|
|
83
|
+
};
|
|
84
|
+
const searchStep = getSearchStep({ title, initialState });
|
|
85
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: httpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
86
|
+
expect(screen.getByText('This is the initial state')).toBeInTheDocument();
|
|
87
|
+
});
|
|
88
|
+
it('renders the initialState when it is a results type', () => {
|
|
89
|
+
const title = 'Search for something';
|
|
90
|
+
const initialState = {
|
|
91
|
+
type: 'results',
|
|
92
|
+
results: getSearchResults('initial query'),
|
|
93
|
+
};
|
|
94
|
+
const searchStep = getSearchStep({ title, initialState });
|
|
95
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: httpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
96
|
+
expect(screen.getByText('Title initial query #1')).toBeInTheDocument();
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
// TODO we haven't added search events yet
|
|
100
|
+
describe('when user focuses the text field', () => {
|
|
101
|
+
it('calls onEvent with the correct analytics event', async () => {
|
|
102
|
+
const title = 'Search for something';
|
|
103
|
+
const searchStep = getSearchStep({ title });
|
|
104
|
+
const onEvent = vi.fn();
|
|
105
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: httpClient, initialStep: searchStep, onEvent: onEvent, onCompletion: onCompletion, onError: onError }));
|
|
106
|
+
const input = screen.getByLabelText(title);
|
|
107
|
+
// await user.click(input); // the revamp version trigger on the first character, not on focus
|
|
108
|
+
await user.type(input, 'abcd');
|
|
109
|
+
await waitFor(() => expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - Search Started', expect.objectContaining({
|
|
110
|
+
flowId: 'the-flow-id',
|
|
111
|
+
stepId: 'Features/Search',
|
|
112
|
+
stepCount: 0,
|
|
113
|
+
layoutId: 'search-layout-aid',
|
|
114
|
+
})));
|
|
115
|
+
});
|
|
116
|
+
});
|
|
117
|
+
describe('when user types into text field', () => {
|
|
118
|
+
it('makes a GET request to the search endpoint', async () => {
|
|
119
|
+
const title = 'Search for something';
|
|
120
|
+
const query = 'test';
|
|
121
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(new Response(null, { status: 404 })));
|
|
122
|
+
const searchStep = getSearchStep({ title, url: '/search', method: 'GET' });
|
|
123
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
124
|
+
const input = screen.getByLabelText(title);
|
|
125
|
+
await user.type(input, query);
|
|
126
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
127
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(1);
|
|
128
|
+
expect(mockHttpClient).toHaveBeenCalledWith(`/search?query=${query}`, expect.anything());
|
|
129
|
+
});
|
|
130
|
+
it('makes a GET request to the search endpoint, with the correct query parameters', async () => {
|
|
131
|
+
const title = 'Search for something';
|
|
132
|
+
const query = 'test';
|
|
133
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(new Response(null, { status: 404 })));
|
|
134
|
+
const searchStep = getSearchStep({ title, url: '/search?extra=stuff', method: 'GET' });
|
|
135
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
136
|
+
const input = screen.getByLabelText(title);
|
|
137
|
+
await user.type(input, query);
|
|
138
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
139
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(1);
|
|
140
|
+
expect(mockHttpClient).toHaveBeenCalledWith(`/search?extra=stuff&query=${query}`, expect.anything());
|
|
141
|
+
});
|
|
142
|
+
it('makes a POST request to the search endpoint', async () => {
|
|
143
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(new Response(null, { status: 404 })));
|
|
144
|
+
const title = 'Search for something';
|
|
145
|
+
const method = 'POST';
|
|
146
|
+
const query = 'test';
|
|
147
|
+
const searchStep = getSearchStep({ title, method });
|
|
148
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
149
|
+
const input = screen.getByLabelText(title);
|
|
150
|
+
await user.type(input, query);
|
|
151
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
152
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(1);
|
|
153
|
+
expect(mockHttpClient).toHaveBeenCalledWith('/search', expect.objectContaining({
|
|
154
|
+
method: 'POST',
|
|
155
|
+
body: JSON.stringify({ query }),
|
|
156
|
+
}));
|
|
157
|
+
});
|
|
158
|
+
describe('when the legacy results type is provided', () => {
|
|
159
|
+
it('renders the search results', async () => {
|
|
160
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(getLegacySearchResponse('abcd')));
|
|
161
|
+
const title = 'Search for something';
|
|
162
|
+
const searchStep = getSearchStep({ title });
|
|
163
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
164
|
+
const input = screen.getByLabelText(title);
|
|
165
|
+
await user.type(input, 'abcd');
|
|
166
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
167
|
+
expect(screen.getByText('Title abcd #1')).toBeInTheDocument();
|
|
168
|
+
expect(screen.getByText('Title abcd #2')).toBeInTheDocument();
|
|
169
|
+
});
|
|
170
|
+
});
|
|
171
|
+
describe('when the modern results type is provided', () => {
|
|
172
|
+
it('renders the search results', async () => {
|
|
173
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(getResultsResponse('abcd')));
|
|
174
|
+
const title = 'Search for something';
|
|
175
|
+
const searchStep = getSearchStep({ title });
|
|
176
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
177
|
+
const input = screen.getByLabelText(title);
|
|
178
|
+
await user.type(input, 'abcd');
|
|
179
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
180
|
+
expect(screen.getByText('Title abcd #1')).toBeInTheDocument();
|
|
181
|
+
expect(screen.getByText('Title abcd #2')).toBeInTheDocument();
|
|
182
|
+
});
|
|
183
|
+
});
|
|
184
|
+
describe('when the layout response type is provided', () => {
|
|
185
|
+
it('renders the search results', async () => {
|
|
186
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(getLayoutResponse('abcd')));
|
|
187
|
+
const title = 'Search for something';
|
|
188
|
+
const searchStep = getSearchStep({ title });
|
|
189
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
190
|
+
const input = screen.getByLabelText(title);
|
|
191
|
+
await user.type(input, 'abcd');
|
|
192
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
193
|
+
expect(screen.getByText("You've searched for abcd, but we didn't find anything, so here is an empty state")).toBeInTheDocument();
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
it('renders the images for each result', async () => {
|
|
197
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(getLegacySearchResponse('abcd')));
|
|
198
|
+
const title = 'Search for something';
|
|
199
|
+
const searchStep = getSearchStep({ title });
|
|
200
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
201
|
+
const input = screen.getByLabelText(title);
|
|
202
|
+
await user.type(input, 'abcd');
|
|
203
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
204
|
+
expect(screen.getByAltText('image from URL')).toHaveAttribute('src', 'https://example.com/image.png');
|
|
205
|
+
expect(screen.getByAltText('image from URI')).toHaveAttribute('src', 'https://example.com/image.png');
|
|
206
|
+
});
|
|
207
|
+
describe('debouncing', () => {
|
|
208
|
+
describe('when the user resumes typing before the first request is made', () => {
|
|
209
|
+
it('only makes 1 request with the last query string, rather than every time the search query changes', async () => {
|
|
210
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(new Response(null, { status: 404 })));
|
|
211
|
+
const title = 'Search for something';
|
|
212
|
+
const searchStep = getSearchStep({ title });
|
|
213
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
214
|
+
const input = screen.getByLabelText(title);
|
|
215
|
+
await user.type(input, 'abcd');
|
|
216
|
+
act(() => {
|
|
217
|
+
vi.advanceTimersByTime(200); // half the debounce time
|
|
218
|
+
});
|
|
219
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(0); // no request yet
|
|
220
|
+
await user.type(input, 'efgh'); // continue typing on the text field
|
|
221
|
+
act(() => {
|
|
222
|
+
vi.advanceTimersByTime(200); // half the debounce time
|
|
223
|
+
});
|
|
224
|
+
await waitFor(() => expect(mockHttpClient).toHaveBeenCalledTimes(1));
|
|
225
|
+
expect(mockHttpClient).toHaveBeenCalledWith(`/search?query=abcdefgh`, expect.anything());
|
|
226
|
+
});
|
|
227
|
+
});
|
|
228
|
+
});
|
|
229
|
+
describe('de-duplication', () => {
|
|
230
|
+
describe('when the user adds a trailing space to a successful query', () => {
|
|
231
|
+
it('only makes 1 request with the original query string, rather than a second one', async () => {
|
|
232
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(getLegacySearchResponse('abcd')));
|
|
233
|
+
const title = 'Search for something';
|
|
234
|
+
const searchStep = getSearchStep({ title });
|
|
235
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
236
|
+
const input = screen.getByLabelText(title);
|
|
237
|
+
await user.type(input, 'abcd');
|
|
238
|
+
await act(async () => vi.advanceTimersByTime(500)); // debounce delay
|
|
239
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(1);
|
|
240
|
+
await user.type(input, ' ');
|
|
241
|
+
await act(async () => vi.advanceTimersByTime(500)); // debounce delay
|
|
242
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(1); // no new requests
|
|
243
|
+
});
|
|
244
|
+
});
|
|
245
|
+
describe('when the user adds a trailing space to a failed query', () => {
|
|
246
|
+
it('makes a second request', async () => {
|
|
247
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(new Response(null, { status: 404 })));
|
|
248
|
+
const title = 'Search for something';
|
|
249
|
+
const searchStep = getSearchStep({ title });
|
|
250
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
251
|
+
const input = screen.getByLabelText(title);
|
|
252
|
+
await user.type(input, 'abcd');
|
|
253
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
254
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(1);
|
|
255
|
+
await user.type(input, ' '); // add a space to the text input field
|
|
256
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
257
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(2); // it tries again
|
|
258
|
+
});
|
|
259
|
+
});
|
|
260
|
+
});
|
|
261
|
+
describe('aborting requests', () => {
|
|
262
|
+
describe('when the user types into the text field, then clears it before the request is made', () => {
|
|
263
|
+
it('aborts the request', async () => {
|
|
264
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(new Response(null, { status: 404 })));
|
|
265
|
+
const title = 'Search for something';
|
|
266
|
+
const searchStep = getSearchStep({ title });
|
|
267
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
268
|
+
const input = screen.getByLabelText(title);
|
|
269
|
+
await user.type(input, 'abcd');
|
|
270
|
+
await user.clear(input);
|
|
271
|
+
act(() => {
|
|
272
|
+
vi.advanceTimersByTime(500); // more than the debounce time
|
|
273
|
+
});
|
|
274
|
+
await waitFor(() => expect(mockHttpClient).toHaveBeenCalledTimes(0));
|
|
275
|
+
});
|
|
276
|
+
});
|
|
277
|
+
describe('when the user types into the text field, and a first request is made, then they type again before the request resolves', () => {
|
|
278
|
+
it('aborts the first request and makes a new one, only displaying the results for the second request', async () => {
|
|
279
|
+
const mockHttpClient = vi.fn(async (url, init) => {
|
|
280
|
+
const query = String(url).split('=')[1];
|
|
281
|
+
await abortableDelay(query === 'abcd' ? 2000 : 500, init === null || init === void 0 ? void 0 : init.signal);
|
|
282
|
+
return getLegacySearchResponse(query);
|
|
283
|
+
});
|
|
284
|
+
const title = 'Search for something';
|
|
285
|
+
const searchStep = getSearchStep({ title });
|
|
286
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
287
|
+
const input = screen.getByLabelText(title);
|
|
288
|
+
await user.type(input, 'abcd');
|
|
289
|
+
act(() => {
|
|
290
|
+
vi.advanceTimersByTime(1000);
|
|
291
|
+
});
|
|
292
|
+
await user.type(input, 'efgh');
|
|
293
|
+
act(() => {
|
|
294
|
+
vi.advanceTimersByTime(1000);
|
|
295
|
+
});
|
|
296
|
+
await waitFor(() => {
|
|
297
|
+
expect(screen.getByText('Title abcdefgh #1')).toBeInTheDocument();
|
|
298
|
+
});
|
|
299
|
+
await waitFor(() => expect(mockHttpClient).toHaveBeenCalledTimes(2));
|
|
300
|
+
await act(async () => {
|
|
301
|
+
vi.advanceTimersByTime(1500);
|
|
302
|
+
return Promise.resolve();
|
|
303
|
+
});
|
|
304
|
+
expect(screen.queryByText('Title abcd #1')).not.toBeInTheDocument();
|
|
305
|
+
});
|
|
306
|
+
});
|
|
307
|
+
});
|
|
308
|
+
describe('when the user clears the search input after some results were displayed', () => {
|
|
309
|
+
it('clears the results and displays the initial state', async () => {
|
|
310
|
+
const title = 'Search for something';
|
|
311
|
+
const initialState = {
|
|
312
|
+
type: 'layout',
|
|
313
|
+
content: [
|
|
314
|
+
{
|
|
315
|
+
type: 'markdown',
|
|
316
|
+
content: 'This is the initial state',
|
|
317
|
+
},
|
|
318
|
+
],
|
|
319
|
+
};
|
|
320
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(getResultsResponse('abcd')));
|
|
321
|
+
const searchStep = getSearchStep({ title, initialState });
|
|
322
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
323
|
+
expect(screen.getByText('This is the initial state')).toBeInTheDocument();
|
|
324
|
+
await user.type(screen.getByLabelText(title), 'abcd');
|
|
325
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
326
|
+
expect(screen.getByText('Title abcd #1')).toBeInTheDocument();
|
|
327
|
+
await user.clear(screen.getByLabelText(title));
|
|
328
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
329
|
+
expect(screen.getByText('This is the initial state')).toBeInTheDocument();
|
|
330
|
+
});
|
|
331
|
+
});
|
|
332
|
+
});
|
|
333
|
+
describe('when the user clicks on an action search result', () => {
|
|
334
|
+
it('loads a new step and triggers analytics event', async () => {
|
|
335
|
+
const mockHttpClient = vi.fn(getMockHttpClient({
|
|
336
|
+
'/search?query=abcd': async () => Promise.resolve(getLegacySearchResponse('abcd')),
|
|
337
|
+
'/next-step': async () => Promise.resolve(new Response(JSON.stringify(nextStep))),
|
|
338
|
+
}));
|
|
339
|
+
const onEvent = vi.fn();
|
|
340
|
+
const title = 'Search for something';
|
|
341
|
+
const searchStep = getSearchStep({ title, url: '/search' });
|
|
342
|
+
const nextStep = {
|
|
343
|
+
id: 'next-step',
|
|
344
|
+
title: 'Next Step',
|
|
345
|
+
schemas: [],
|
|
346
|
+
layout: [
|
|
347
|
+
{
|
|
348
|
+
type: 'paragraph',
|
|
349
|
+
text: 'This is the next step.',
|
|
350
|
+
},
|
|
351
|
+
],
|
|
352
|
+
};
|
|
353
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError, onEvent: onEvent }));
|
|
354
|
+
const input = screen.getByLabelText(title);
|
|
355
|
+
await user.type(input, 'abcd');
|
|
356
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
357
|
+
expect(screen.getByText('Title abcd #1')).toBeInTheDocument();
|
|
358
|
+
await user.click(screen.getByText('Title abcd #1'));
|
|
359
|
+
await expect(screen.findByText('Next Step')).resolves.toBeInTheDocument();
|
|
360
|
+
expect(screen.queryByText('Title abcd #1')).not.toBeInTheDocument();
|
|
361
|
+
expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - Search Result Selected', expect.objectContaining({
|
|
362
|
+
flowId: 'the-flow-id',
|
|
363
|
+
stepId: 'Features/Search',
|
|
364
|
+
stepCount: 0,
|
|
365
|
+
layoutId: 'search-layout-aid',
|
|
366
|
+
type: 'action',
|
|
367
|
+
actionId: 'search-result-action-id-01',
|
|
368
|
+
}));
|
|
369
|
+
});
|
|
370
|
+
});
|
|
371
|
+
describe('when the user clicks on a search search result', () => {
|
|
372
|
+
it('makes a new search request, displays the results, and updates the search input, and triggers analytics event', async () => {
|
|
373
|
+
const mockHttpClient = vi.fn(getMockHttpClient({
|
|
374
|
+
'/search?query=abcd': async () => Promise.resolve(getLegacySearchResponse('abcd')),
|
|
375
|
+
'/new-search?new-query=new+query+with+special+characters+%28%21%40%C2%A3%24%25%5E%26*%29': async () => Promise.resolve(getLegacySearchResponse('new query results')),
|
|
376
|
+
}));
|
|
377
|
+
const onEvent = vi.fn();
|
|
378
|
+
const title = 'Search for something';
|
|
379
|
+
const searchStep = getSearchStep({ title, url: '/search' });
|
|
380
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError, onEvent: onEvent }));
|
|
381
|
+
const input = screen.getByLabelText(title);
|
|
382
|
+
await user.type(input, 'abcd');
|
|
383
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
384
|
+
expect(screen.getByText('Title abcd #1')).toBeInTheDocument();
|
|
385
|
+
await user.click(screen.getByText('Title abcd #2'));
|
|
386
|
+
expect(mockHttpClient).toHaveBeenCalledWith(`/new-search?new-query=new+query+with+special+characters+%28%21%40%C2%A3%24%25%5E%26*%29`, expect.anything());
|
|
387
|
+
expect(screen.getByText('Title new query results #1')).toBeInTheDocument();
|
|
388
|
+
expect(screen.getByLabelText(title)).toHaveValue('new query with special characters (!@£$%^&*)');
|
|
389
|
+
expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - Search Result Selected', expect.objectContaining({
|
|
390
|
+
flowId: 'the-flow-id',
|
|
391
|
+
stepId: 'Features/Search',
|
|
392
|
+
stepCount: 0,
|
|
393
|
+
layoutId: 'search-layout-aid',
|
|
394
|
+
type: 'search',
|
|
395
|
+
}));
|
|
396
|
+
});
|
|
397
|
+
});
|
|
398
|
+
describe('when the search request fails', () => {
|
|
399
|
+
it('displays an error message and allows to retry', async () => {
|
|
400
|
+
const responses = [
|
|
401
|
+
Promise.resolve(new Response(null, { status: 404 })),
|
|
402
|
+
Promise.resolve(getLegacySearchResponse('abcd')),
|
|
403
|
+
];
|
|
404
|
+
let callCount = 0;
|
|
405
|
+
const mockHttpClient = vi.fn(async () => {
|
|
406
|
+
const response = responses[Math.min(callCount, responses.length - 1)];
|
|
407
|
+
callCount += 1;
|
|
408
|
+
return response;
|
|
409
|
+
});
|
|
410
|
+
const title = 'Search for something';
|
|
411
|
+
const searchStep = getSearchStep({ title, url: '/search' });
|
|
412
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
413
|
+
const input = screen.getByLabelText(title);
|
|
414
|
+
await user.type(input, 'abcd');
|
|
415
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
416
|
+
expect(screen.getByText('Something went wrong.')).toBeInTheDocument();
|
|
417
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(1);
|
|
418
|
+
const link = screen.getByText('Retry');
|
|
419
|
+
await user.click(link);
|
|
420
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
421
|
+
expect(mockHttpClient).toHaveBeenCalledTimes(2);
|
|
422
|
+
expect(screen.getByText('Title abcd #1')).toBeInTheDocument();
|
|
423
|
+
});
|
|
424
|
+
});
|
|
425
|
+
describe('when a search returns no results', () => {
|
|
426
|
+
it('displays the empty message from the search configuration', async () => {
|
|
427
|
+
const mockHttpClient = vi.fn(async () => Promise.resolve(new Response(JSON.stringify({ type: 'results', results: [] }))));
|
|
428
|
+
const title = 'Search for something';
|
|
429
|
+
const searchStep = getSearchStep({ title });
|
|
430
|
+
renderWithProviders(_jsx(DynamicFlowWise, { flowId: "the-flow-id", httpClient: mockHttpClient, initialStep: searchStep, onCompletion: onCompletion, onError: onError }));
|
|
431
|
+
const input = screen.getByLabelText(title);
|
|
432
|
+
await user.type(input, 'abcd');
|
|
433
|
+
await act(async () => vi.advanceTimersByTimeAsync(500)); // debounce delay
|
|
434
|
+
expect(screen.getByText('No results (sorry!)')).toBeInTheDocument();
|
|
435
|
+
});
|
|
436
|
+
});
|
|
437
|
+
});
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { mapSchemaToComponent } from '../../domain/mappers/mapSchemaToComponent';
|
|
2
|
+
import { getMockMapperProps, getMockRendererMapperProps, getMockSchemaMapperProps, } from '../../domain/mappers/schema/tests/test-utils';
|
|
3
|
+
import { multiSelectInputComponentToProps } from '../../renderers/mappers/multiSelectComponentToProps';
|
|
4
|
+
describe('MultiSelectInputRendererProps mapper', () => {
|
|
5
|
+
describe('given a MultiSelectInputComponent in which each option has a layout', () => {
|
|
6
|
+
it('should map options including their children and childrenProps', () => {
|
|
7
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j;
|
|
8
|
+
const schema = {
|
|
9
|
+
type: 'array',
|
|
10
|
+
addItemTitle: '',
|
|
11
|
+
editItemTitle: '',
|
|
12
|
+
items: {
|
|
13
|
+
oneOf: [
|
|
14
|
+
{
|
|
15
|
+
const: 'option-1',
|
|
16
|
+
title: 'Option 1',
|
|
17
|
+
layout: [{ type: 'paragraph', text: 'This is option 1' }],
|
|
18
|
+
},
|
|
19
|
+
{
|
|
20
|
+
const: 'option-2',
|
|
21
|
+
title: 'Option 2',
|
|
22
|
+
layout: [{ type: 'paragraph', text: 'This is option 2' }],
|
|
23
|
+
},
|
|
24
|
+
{
|
|
25
|
+
const: 'option-3',
|
|
26
|
+
title: 'Option 3',
|
|
27
|
+
layout: [{ type: 'paragraph', text: 'This is option 3' }],
|
|
28
|
+
},
|
|
29
|
+
],
|
|
30
|
+
},
|
|
31
|
+
};
|
|
32
|
+
const schemaMapperProps = getMockSchemaMapperProps({ schema });
|
|
33
|
+
const mapperProps = getMockMapperProps();
|
|
34
|
+
const component = mapSchemaToComponent(schemaMapperProps, mapperProps);
|
|
35
|
+
const rendererMapperProps = Object.assign(Object.assign({}, getMockRendererMapperProps()), { render: () => null });
|
|
36
|
+
const props = multiSelectInputComponentToProps(component, rendererMapperProps);
|
|
37
|
+
expect(props.options).toHaveLength(3);
|
|
38
|
+
expect(props.options[0].children).toBeDefined();
|
|
39
|
+
expect(props.options[1].children).toBeDefined();
|
|
40
|
+
expect(props.options[2].children).toBeDefined();
|
|
41
|
+
expect(props.options[0].childrenProps).toBeDefined();
|
|
42
|
+
expect((_a = props.options[0].childrenProps) === null || _a === void 0 ? void 0 : _a.type).toBe('container');
|
|
43
|
+
expect((_b = props.options[0].childrenProps.childrenProps) === null || _b === void 0 ? void 0 : _b[0].type).toBe('paragraph');
|
|
44
|
+
expect(((_c = props.options[0].childrenProps
|
|
45
|
+
.childrenProps) === null || _c === void 0 ? void 0 : _c[0]).text).toBe('This is option 1');
|
|
46
|
+
expect(props.options[1].childrenProps).toBeDefined();
|
|
47
|
+
expect((_d = props.options[1].childrenProps) === null || _d === void 0 ? void 0 : _d.type).toBe('container');
|
|
48
|
+
expect((_e = props.options[1].childrenProps.childrenProps) === null || _e === void 0 ? void 0 : _e[0].type).toBe('paragraph');
|
|
49
|
+
expect(((_f = props.options[1].childrenProps
|
|
50
|
+
.childrenProps) === null || _f === void 0 ? void 0 : _f[0]).text).toBe('This is option 2');
|
|
51
|
+
expect(props.options[2].childrenProps).toBeDefined();
|
|
52
|
+
expect((_g = props.options[2].childrenProps) === null || _g === void 0 ? void 0 : _g.type).toBe('container');
|
|
53
|
+
expect((_h = props.options[2].childrenProps.childrenProps) === null || _h === void 0 ? void 0 : _h[0].type).toBe('paragraph');
|
|
54
|
+
expect(((_j = props.options[2].childrenProps
|
|
55
|
+
.childrenProps) === null || _j === void 0 ? void 0 : _j[0]).text).toBe('This is option 3');
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
});
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { mapSchemaToComponent } from '../../domain/mappers/mapSchemaToComponent';
|
|
2
|
+
import { getMockMapperProps, getMockRendererMapperProps, getMockSchemaMapperProps, } from '../../domain/mappers/schema/tests/test-utils';
|
|
3
|
+
import { selectInputComponentToProps } from '../../renderers/mappers/selectInputComponentToProps';
|
|
4
|
+
describe('SelectInputRendererProps mapper', () => {
|
|
5
|
+
describe('given a SelectInputComponent in which each option has a layout', () => {
|
|
6
|
+
it('should map options including their children and childrenProps', () => {
|
|
7
|
+
var _a, _b, _c, _d, _e, _f;
|
|
8
|
+
const schema = {
|
|
9
|
+
oneOf: [
|
|
10
|
+
{
|
|
11
|
+
title: 'Option 1',
|
|
12
|
+
const: 'option-1',
|
|
13
|
+
layout: [{ type: 'paragraph', text: 'This is option 1' }],
|
|
14
|
+
},
|
|
15
|
+
{
|
|
16
|
+
title: 'Option 2',
|
|
17
|
+
const: 'option-2',
|
|
18
|
+
layout: [{ type: 'paragraph', text: 'This is option 2' }],
|
|
19
|
+
},
|
|
20
|
+
],
|
|
21
|
+
};
|
|
22
|
+
const schemaMapperProps = getMockSchemaMapperProps({ schema });
|
|
23
|
+
const mapperProps = getMockMapperProps();
|
|
24
|
+
const component = mapSchemaToComponent(schemaMapperProps, mapperProps);
|
|
25
|
+
const rendererMapperProps = Object.assign(Object.assign({}, getMockRendererMapperProps()), { render: () => null });
|
|
26
|
+
const props = selectInputComponentToProps(component, rendererMapperProps);
|
|
27
|
+
expect(props.options).toHaveLength(2);
|
|
28
|
+
expect(props.options[0].children).toBeDefined();
|
|
29
|
+
expect(props.options[1].children).toBeDefined();
|
|
30
|
+
expect(props.options[0].childrenProps).toBeDefined();
|
|
31
|
+
expect((_a = props.options[0].childrenProps) === null || _a === void 0 ? void 0 : _a.type).toBe('container');
|
|
32
|
+
expect((_b = props.options[0].childrenProps.childrenProps) === null || _b === void 0 ? void 0 : _b[0].type).toBe('paragraph');
|
|
33
|
+
expect(((_c = props.options[0].childrenProps
|
|
34
|
+
.childrenProps) === null || _c === void 0 ? void 0 : _c[0]).text).toBe('This is option 1');
|
|
35
|
+
expect(props.options[1].childrenProps).toBeDefined();
|
|
36
|
+
expect((_d = props.options[1].childrenProps) === null || _d === void 0 ? void 0 : _d.type).toBe('container');
|
|
37
|
+
expect((_e = props.options[1].childrenProps.childrenProps) === null || _e === void 0 ? void 0 : _e[0].type).toBe('paragraph');
|
|
38
|
+
expect(((_f = props.options[1].childrenProps
|
|
39
|
+
.childrenProps) === null || _f === void 0 ? void 0 : _f[0]).text).toBe('This is option 2');
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
});
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { screen, waitFor } from '@testing-library/react';
|
|
3
|
+
import { userEvent } from '@testing-library/user-event';
|
|
4
|
+
import { vi } from 'vitest';
|
|
5
|
+
import { renderWithProviders } from '../../test-utils';
|
|
6
|
+
import DynamicFlowWise from '../../test-utils/DynamicFlowWise';
|
|
7
|
+
const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
|
|
8
|
+
const getDefaultProps = () => ({
|
|
9
|
+
flowId: 'flow-id',
|
|
10
|
+
onCompletion: vi.fn(),
|
|
11
|
+
onError: vi.fn(),
|
|
12
|
+
onEvent: vi.fn(),
|
|
13
|
+
onLog: vi.fn(),
|
|
14
|
+
});
|
|
15
|
+
const getStep = (model, schema) => {
|
|
16
|
+
var _a;
|
|
17
|
+
return ({
|
|
18
|
+
id: 'step-id',
|
|
19
|
+
title: 'Submission test',
|
|
20
|
+
layout: [
|
|
21
|
+
{ type: 'form', schemaId: (_a = schema.$id) !== null && _a !== void 0 ? _a : '' },
|
|
22
|
+
{
|
|
23
|
+
type: 'button',
|
|
24
|
+
title: 'Submit',
|
|
25
|
+
action: { id: 'action-id', url: '/submit', method: 'POST' },
|
|
26
|
+
},
|
|
27
|
+
],
|
|
28
|
+
schemas: [schema],
|
|
29
|
+
model,
|
|
30
|
+
});
|
|
31
|
+
};
|
|
32
|
+
describe('TextInputRenderer', () => {
|
|
33
|
+
const variants = [
|
|
34
|
+
['without control nor format', { $id: '#schema', type: 'string', title: 'Name' }],
|
|
35
|
+
['with numeric format', { $id: '#schema', type: 'string', title: 'Name', format: 'numeric' }],
|
|
36
|
+
['with numeric control', { $id: '#schema', type: 'string', title: 'Name', control: 'numeric' }],
|
|
37
|
+
];
|
|
38
|
+
describe.each(variants)('%s', (_, schema) => {
|
|
39
|
+
it('should submit the user-entered data to the server', async () => {
|
|
40
|
+
const step = getStep(null, schema);
|
|
41
|
+
const httpClient = vi.fn();
|
|
42
|
+
const props = getDefaultProps();
|
|
43
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({ httpClient: httpClient, initialStep: step }, props)));
|
|
44
|
+
await user.type(screen.getByLabelText('Name'), '12345.6789');
|
|
45
|
+
await user.click(screen.getByText('Submit'));
|
|
46
|
+
await waitFor(() => {
|
|
47
|
+
expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({ body: JSON.stringify('12345.6789') }));
|
|
48
|
+
});
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
});
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import type { BaseLayoutComponent, CallToAction,
|
|
1
|
+
import type { BaseLayoutComponent, CallToAction, Media, OnComponentUpdate } from '../types';
|
|
2
2
|
export type UpsellComponent = BaseLayoutComponent & {
|
|
3
3
|
type: 'upsell';
|
|
4
4
|
kind: 'layout';
|
|
@@ -7,7 +7,6 @@ export type UpsellComponent = BaseLayoutComponent & {
|
|
|
7
7
|
media?: Media;
|
|
8
8
|
visible: boolean;
|
|
9
9
|
onDismiss: (() => void) | undefined;
|
|
10
|
-
context: Context;
|
|
11
10
|
};
|
|
12
|
-
export declare const createUpsellComponent: (upsellProps: Pick<UpsellComponent, "uid" | "analyticsId" | "callToAction" | "
|
|
11
|
+
export declare const createUpsellComponent: (upsellProps: Pick<UpsellComponent, "uid" | "analyticsId" | "callToAction" | "control" | "margin" | "media" | "tags" | "text" | "onDismiss">, onComponentUpdate: OnComponentUpdate) => UpsellComponent;
|
|
13
12
|
//# sourceMappingURL=UpsellComponent.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"UpsellComponent.d.ts","sourceRoot":"","sources":["../../../../src/domain/components/UpsellComponent.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,
|
|
1
|
+
{"version":3,"file":"UpsellComponent.d.ts","sourceRoot":"","sources":["../../../../src/domain/components/UpsellComponent.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,mBAAmB,EAAE,YAAY,EAAE,KAAK,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAG5F,MAAM,MAAM,eAAe,GAAG,mBAAmB,GAAG;IAClD,IAAI,EAAE,QAAQ,CAAC;IACf,IAAI,EAAE,QAAQ,CAAC;IACf,IAAI,EAAE,MAAM,CAAC;IACb,YAAY,EAAE,YAAY,CAAC;IAC3B,KAAK,CAAC,EAAE,KAAK,CAAC;IACd,OAAO,EAAE,OAAO,CAAC;IACjB,SAAS,EAAE,CAAC,MAAM,IAAI,CAAC,GAAG,SAAS,CAAC;CACrC,CAAC;AAEF,eAAO,MAAM,qBAAqB,GAChC,aAAa,IAAI,CACf,eAAe,EACb,KAAK,GACL,aAAa,GACb,cAAc,GACd,SAAS,GACT,QAAQ,GACR,OAAO,GACP,MAAM,GACN,MAAM,GACN,WAAW,CACd,EACD,mBAAmB,iBAAiB,KACnC,eAmBF,CAAC"}
|
|
@@ -10,26 +10,27 @@ export type SearchProps = {
|
|
|
10
10
|
query: string;
|
|
11
11
|
signal: AbortSignal;
|
|
12
12
|
};
|
|
13
|
-
export type
|
|
13
|
+
export type SearchResultsState = {
|
|
14
14
|
type: 'layout';
|
|
15
15
|
content: LayoutComponent[];
|
|
16
16
|
} | {
|
|
17
17
|
type: 'results';
|
|
18
18
|
results: SearchResult[];
|
|
19
19
|
};
|
|
20
|
-
export type PerformSearch = (props: SearchProps) => Promise<
|
|
20
|
+
export type PerformSearch = (props: SearchProps) => Promise<SearchResultsState | null>;
|
|
21
21
|
export type SearchComponent = BaseLayoutComponent & {
|
|
22
22
|
type: 'search';
|
|
23
23
|
kind: 'layout';
|
|
24
24
|
emptyMessage: string;
|
|
25
|
+
initialState: SearchResultsState;
|
|
25
26
|
hint?: string;
|
|
26
27
|
isLoading: boolean;
|
|
27
28
|
query: string;
|
|
28
29
|
title: string;
|
|
29
|
-
state:
|
|
30
|
+
state: SearchResultsState;
|
|
30
31
|
error?: string;
|
|
31
32
|
onChange: (query: string) => void;
|
|
32
33
|
onSelect: (result: SearchResult) => void;
|
|
33
34
|
};
|
|
34
|
-
export declare const createSearchComponent: (searchProps: Pick<SearchComponent, "uid" | "analyticsId" | "control" | "emptyMessage" | "hint" | "margin" | "tags" | "title">, performSearch: PerformSearch, onBehavior: OnBehavior, onComponentUpdate: OnComponentUpdate) => SearchComponent;
|
|
35
|
+
export declare const createSearchComponent: (searchProps: Pick<SearchComponent, "uid" | "analyticsId" | "control" | "emptyMessage" | "initialState" | "hint" | "margin" | "tags" | "title">, performSearch: PerformSearch, onBehavior: OnBehavior, onComponentUpdate: OnComponentUpdate) => SearchComponent;
|
|
35
36
|
//# sourceMappingURL=SearchComponent.d.ts.map
|