From 029dde92aade11ab2615aa6dfc8f988c6d164972 Mon Sep 17 00:00:00 2001 From: "Chris K.Y. FUNG" <8746768+chriskyfung@users.noreply.github.com> Date: Mon, 9 Oct 2023 22:37:44 +0800 Subject: [PATCH 1/2] Files from original version 2.3.0 Unzipped from https://downloads.wordpress.org/plugin/wp-search-with-algolia.2.3.0.zip --- wp-search-with-algolia/README.txt | 16 +- wp-search-with-algolia/algolia.php | 4 +- .../class-algolia-admin-page-autocomplete.php | 4 +- .../includes/admin/css/algolia-admin.css | 18 - .../includes/class-algolia-cli.php | 30 +- .../includes/class-algolia-scripts.php | 3 - .../includes/class-algolia-settings.php | 26 +- .../class-algolia-template-loader.php | 2 +- .../algoliasearch-client-php/src/Algolia.php | 2 +- .../src/Http/Psr7/Request.php | 57 + .../src/Http/Psr7/Response.php | 42 + .../src/Http/Psr7/Stream.php | 42 + .../src/Http/Psr7/Uri.php | 45 + .../src/InsightsClient.php | 5 +- .../src/PersonalizationClient.php | 5 +- .../src/RecommendClient.php | 5 +- .../src/RecommendationClient.php | 5 +- .../src/Response/AbstractResponse.php | 8 + .../src/Response/BatchIndexingResponse.php | 15 + .../src/Response/MultiResponse.php | 18 + .../dist/algoliasearch-lite.esm.browser.js | 2 +- .../dist/algoliasearch-lite.umd.js | 4 +- .../dist/algoliasearch.esm.browser.js | 2 +- .../algoliasearch/dist/algoliasearch.umd.js | 4 +- .../js/instantsearch.js/CHANGELOG.md | 196 + .../cjs/components/Answers/Answers.js | 8 +- .../cjs/components/Breadcrumb/Breadcrumb.js | 10 +- .../ClearRefinements/ClearRefinements.js | 6 +- .../CurrentRefinements/CurrentRefinements.js | 8 +- .../GeoSearchControls/GeoSearchButton.js | 1 - .../GeoSearchControls/GeoSearchControls.js | 8 +- .../GeoSearchControls/GeoSearchToggle.js | 1 - .../cjs/components/Hits/Hits.js | 8 +- .../components/InfiniteHits/InfiniteHits.js | 12 +- .../cjs/components/MenuSelect/MenuSelect.js | 6 +- .../cjs/components/Pagination/Pagination.js | 10 +- .../cjs/components/Panel/Panel.js | 8 +- .../cjs/components/PoweredBy/PoweredBy.js | 23 +- .../QueryRuleCustomData.js | 1 - .../cjs/components/RangeInput/RangeInput.js | 8 +- .../RefinementList/RefinementList.js | 10 +- .../components/RelevantSort/RelevantSort.js | 1 - .../cjs/components/Selector/Selector.js | 9 +- .../cjs/components/Slider/Pit.js | 8 +- .../cjs/components/Slider/Slider.js | 9 +- .../cjs/components/Stats/Stats.js | 4 +- .../cjs/components/Template/Template.js | 20 +- .../cjs/components/VoiceSearch/VoiceSearch.js | 1 - .../connectCurrentRefinements.js | 2 + .../connectHierarchicalMenu.js | 5 +- .../hits-per-page/connectHitsPerPage.js | 4 +- .../numeric-menu/connectNumericMenu.js | 74 +- .../cjs/connectors/range/connectRange.js | 53 +- .../rating-menu/connectRatingMenu.js | 16 +- .../cjs/connectors/sort-by/connectSortBy.js | 4 +- .../connectToggleRefinement.js | 4 +- .../instantsearch.js/cjs/lib/InstantSearch.js | 145 +- .../instantsearch.js/cjs/lib/createHelpers.js | 4 +- .../cjs/lib/insights/listener.js | 1 - .../cjs/lib/utils/capitalize.js | 7 +- .../cjs/lib/utils/checkIndexUiState.js | 8 +- .../cjs/lib/utils/checkRendering.js | 13 +- .../cjs/lib/utils/clearRefinements.js | 12 +- .../cjs/lib/utils/concatHighlightedParts.js | 2 +- .../cjs/lib/utils/createSendEventForFacet.js | 9 +- .../cjs/lib/utils/createSendEventForHits.js | 13 +- .../instantsearch.js/cjs/lib/utils/defer.js | 9 +- .../cjs/lib/utils/detect-insights-client.js | 2 +- .../cjs/lib/utils/documentation.js | 15 +- .../cjs/lib/utils/escape-highlight.js | 10 +- .../js/instantsearch.js/cjs/lib/utils/find.js | 7 +- .../cjs/lib/utils/findIndex.js | 7 +- .../cjs/lib/utils/getContainerNode.js | 13 +- .../cjs/lib/utils/getHighlightFromSiblings.js | 8 +- .../cjs/lib/utils/getHighlightedParts.js | 2 +- .../cjs/lib/utils/getObjectType.js | 7 +- .../cjs/lib/utils/getPropertyByPath.js | 7 +- .../cjs/lib/utils/getRefinements.js | 10 +- .../instantsearch.js/cjs/lib/utils/index.js | 969 +- .../cjs/lib/utils/isDomElement.js | 7 +- .../instantsearch.js/cjs/lib/utils/isEqual.js | 7 +- .../cjs/lib/utils/isFacetRefined.js | 2 +- .../cjs/lib/utils/isFiniteNumber.js | 8 +- .../cjs/lib/utils/isPlainObject.js | 7 +- .../cjs/lib/utils/isSpecialClick.js | 7 +- .../instantsearch.js/cjs/lib/utils/logger.js | 8 +- .../cjs/lib/utils/mergeSearchParameters.js | 17 +- .../js/instantsearch.js/cjs/lib/utils/noop.js | 7 +- .../instantsearch.js/cjs/lib/utils/range.js | 7 +- .../cjs/lib/utils/resolveSearchParameters.js | 9 +- .../cjs/lib/utils/reverseHighlightedParts.js | 8 +- .../instantsearch.js/cjs/lib/utils/toArray.js | 7 +- .../js/instantsearch.js/cjs/lib/utils/uniq.js | 7 +- .../js/instantsearch.js/cjs/lib/version.js | 2 +- .../middlewares/createInsightsMiddleware.js | 61 +- .../middlewares/createMetadataMiddleware.js | 17 +- .../cjs/middlewares/createRouterMiddleware.js | 1 + .../cjs/types/algoliasearch.js | 19 +- .../cjs/widgets/answers/answers.js | 18 +- .../cjs/widgets/answers/defaultTemplates.js | 8 +- .../cjs/widgets/breadcrumb/breadcrumb.js | 20 +- .../widgets/breadcrumb/defaultTemplates.js | 8 +- .../clear-refinements/clear-refinements.js | 16 +- .../clear-refinements/defaultTemplates.js | 4 +- .../current-refinements.js | 25 +- .../widgets/geo-search/GeoSearchRenderer.js | 2 +- .../widgets/geo-search/createHTMLMarker.js | 14 +- .../widgets/geo-search/defaultTemplates.js | 21 +- .../cjs/widgets/geo-search/geo-search.js | 26 +- .../hierarchical-menu/defaultTemplates.js | 28 +- .../hierarchical-menu/hierarchical-menu.js | 34 +- .../widgets/hits-per-page/hits-per-page.js | 8 +- .../cjs/widgets/hits/defaultTemplates.js | 4 +- .../instantsearch.js/cjs/widgets/hits/hits.js | 20 +- .../cjs/widgets/index/index.js | 112 +- .../widgets/infinite-hits/defaultTemplates.js | 12 +- .../widgets/infinite-hits/infinite-hits.js | 28 +- .../widgets/menu-select/defaultTemplates.js | 13 +- .../cjs/widgets/menu-select/menu-select.js | 14 +- .../cjs/widgets/menu/defaultTemplates.js | 27 +- .../instantsearch.js/cjs/widgets/menu/menu.js | 26 +- .../widgets/numeric-menu/defaultTemplates.js | 20 +- .../cjs/widgets/numeric-menu/numeric-menu.js | 22 +- .../cjs/widgets/pagination/pagination.js | 26 +- .../cjs/widgets/panel/panel.js | 20 +- .../cjs/widgets/powered-by/powered-by.js | 8 +- .../query-rule-custom-data.js | 4 +- .../cjs/widgets/range-input/range-input.js | 32 +- .../cjs/widgets/range-slider/range-slider.js | 6 +- .../widgets/rating-menu/defaultTemplates.js | 57 +- .../cjs/widgets/rating-menu/rating-menu.js | 30 +- .../refinement-list/defaultTemplates.js | 40 +- .../refinement-list/refinement-list.js | 52 +- .../widgets/relevant-sort/defaultTemplates.js | 4 +- .../widgets/relevant-sort/relevant-sort.js | 8 +- .../widgets/search-box/defaultTemplates.js | 64 +- .../cjs/widgets/search-box/search-box.js | 20 +- .../cjs/widgets/sort-by/sort-by.js | 8 +- .../cjs/widgets/stats/stats.js | 102 +- .../toggle-refinement/defaultTemplates.js | 5 +- .../toggle-refinement/toggle-refinement.js | 14 +- .../widgets/voice-search/defaultTemplates.js | 89 +- .../cjs/widgets/voice-search/voice-search.js | 8 +- .../dist/instantsearch.development.js | 14311 ++++++++-------- .../dist/instantsearch.development.js.map | 2 +- .../dist/instantsearch.production.min.js | 4 +- .../dist/instantsearch.production.min.js.map | 2 +- .../es/components/Answers/Answers.js | 9 +- .../es/components/Breadcrumb/Breadcrumb.js | 11 +- .../ClearRefinements/ClearRefinements.js | 7 +- .../CurrentRefinements/CurrentRefinements.js | 7 +- .../GeoSearchControls/GeoSearchButton.js | 1 - .../GeoSearchControls/GeoSearchControls.js | 9 +- .../GeoSearchControls/GeoSearchToggle.js | 1 - .../es/components/Hits/Hits.js | 7 +- .../components/InfiniteHits/InfiniteHits.js | 11 +- .../es/components/MenuSelect/MenuSelect.js | 7 +- .../es/components/Pagination/Pagination.js | 7 +- .../es/components/Panel/Panel.js | 9 +- .../es/components/PoweredBy/PoweredBy.js | 23 +- .../QueryRuleCustomData.js | 1 - .../es/components/RangeInput/RangeInput.js | 5 +- .../RefinementList/RefinementList.js | 11 +- .../RefinementList/RefinementListItem.js | 1 - .../components/RelevantSort/RelevantSort.js | 1 - .../es/components/SearchBox/SearchBox.js | 1 - .../es/components/Selector/Selector.js | 3 +- .../es/components/Slider/Pit.js | 7 +- .../es/components/Slider/Rheostat.js | 2 - .../es/components/Slider/Slider.js | 10 +- .../es/components/Stats/Stats.js | 3 +- .../es/components/Template/Template.js | 20 +- .../ToggleRefinement/ToggleRefinement.js | 5 - .../es/components/VoiceSearch/VoiceSearch.js | 1 - .../connectCurrentRefinements.js | 2 + .../connectHierarchicalMenu.js | 5 +- .../hits-per-page/connectHitsPerPage.js | 4 +- .../numeric-menu/connectNumericMenu.js | 76 +- .../es/connectors/range/connectRange.js | 55 +- .../rating-menu/connectRatingMenu.js | 16 +- .../es/connectors/sort-by/connectSortBy.js | 4 +- .../connectToggleRefinement.js | 4 +- .../js/instantsearch.js/es/index.js | 2 +- .../instantsearch.js/es/lib/InstantSearch.js | 143 +- .../instantsearch.js/es/lib/createHelpers.js | 3 +- .../es/lib/insights/listener.js | 1 - .../es/lib/utils/capitalize.js | 6 +- .../es/lib/utils/checkIndexUiState.js | 2 +- .../es/lib/utils/checkRendering.js | 9 +- .../es/lib/utils/clearRefinements.js | 11 +- .../es/lib/utils/concatHighlightedParts.js | 2 +- .../es/lib/utils/createSendEventForFacet.js | 5 +- .../es/lib/utils/createSendEventForHits.js | 16 +- .../js/instantsearch.js/es/lib/utils/defer.js | 7 +- .../es/lib/utils/detect-insights-client.js | 2 +- .../es/lib/utils/documentation.js | 8 +- .../es/lib/utils/escape-highlight.js | 4 +- .../js/instantsearch.js/es/lib/utils/find.js | 6 +- .../es/lib/utils/findIndex.js | 6 +- .../es/lib/utils/getContainerNode.js | 8 +- .../es/lib/utils/getHighlightFromSiblings.js | 4 +- .../es/lib/utils/getHighlightedParts.js | 2 +- .../es/lib/utils/getObjectType.js | 6 +- .../es/lib/utils/getPropertyByPath.js | 6 +- .../es/lib/utils/getRefinements.js | 4 +- .../js/instantsearch.js/es/lib/utils/index.js | 89 +- .../es/lib/utils/isDomElement.js | 6 +- .../instantsearch.js/es/lib/utils/isEqual.js | 6 +- .../es/lib/utils/isFacetRefined.js | 2 +- .../es/lib/utils/isFiniteNumber.js | 7 +- .../es/lib/utils/isPlainObject.js | 6 +- .../es/lib/utils/isSpecialClick.js | 6 +- .../instantsearch.js/es/lib/utils/logger.js | 2 +- .../es/lib/utils/mergeSearchParameters.js | 10 +- .../js/instantsearch.js/es/lib/utils/noop.js | 4 +- .../js/instantsearch.js/es/lib/utils/range.js | 6 +- .../es/lib/utils/resolveSearchParameters.js | 6 +- .../es/lib/utils/reverseHighlightedParts.js | 4 +- .../instantsearch.js/es/lib/utils/toArray.js | 6 +- .../js/instantsearch.js/es/lib/utils/uniq.js | 6 +- .../js/instantsearch.js/es/lib/version.js | 2 +- .../middlewares/createInsightsMiddleware.js | 54 +- .../middlewares/createMetadataMiddleware.js | 19 +- .../es/middlewares/createRouterMiddleware.js | 1 + .../es/types/algoliasearch.js | 2 + .../js/instantsearch.js/es/types/index.js | 1 + .../es/widgets/answers/answers.js | 6 +- .../es/widgets/answers/defaultTemplates.js | 8 +- .../es/widgets/breadcrumb/breadcrumb.js | 6 +- .../es/widgets/breadcrumb/defaultTemplates.js | 8 +- .../clear-refinements/clear-refinements.js | 10 +- .../clear-refinements/defaultTemplates.js | 4 +- .../current-refinements.js | 12 +- .../widgets/geo-search/GeoSearchRenderer.js | 3 +- .../es/widgets/geo-search/createHTMLMarker.js | 9 +- .../es/widgets/geo-search/defaultTemplates.js | 20 +- .../es/widgets/geo-search/geo-search.js | 5 +- .../hierarchical-menu/defaultTemplates.js | 24 +- .../hierarchical-menu/hierarchical-menu.js | 10 +- .../es/widgets/hits-per-page/hits-per-page.js | 3 +- .../es/widgets/hits/defaultTemplates.js | 4 +- .../instantsearch.js/es/widgets/hits/hits.js | 6 +- .../es/widgets/index/index.js | 103 +- .../widgets/infinite-hits/defaultTemplates.js | 12 +- .../es/widgets/infinite-hits/infinite-hits.js | 6 +- .../widgets/menu-select/defaultTemplates.js | 11 +- .../es/widgets/menu-select/menu-select.js | 6 +- .../es/widgets/menu/defaultTemplates.js | 23 +- .../instantsearch.js/es/widgets/menu/menu.js | 6 +- .../widgets/numeric-menu/defaultTemplates.js | 18 +- .../es/widgets/numeric-menu/numeric-menu.js | 6 +- .../es/widgets/pagination/pagination.js | 3 +- .../es/widgets/panel/panel.js | 3 +- .../es/widgets/powered-by/powered-by.js | 3 +- .../query-rule-custom-data.js | 3 +- .../es/widgets/range-input/range-input.js | 14 +- .../es/widgets/range-slider/range-slider.js | 3 +- .../widgets/rating-menu/defaultTemplates.js | 54 +- .../es/widgets/rating-menu/rating-menu.js | 6 +- .../refinement-list/defaultTemplates.js | 36 +- .../refinement-list/refinement-list.js | 6 +- .../widgets/relevant-sort/defaultTemplates.js | 4 +- .../es/widgets/relevant-sort/relevant-sort.js | 3 +- .../es/widgets/search-box/defaultTemplates.js | 63 +- .../es/widgets/search-box/search-box.js | 3 +- .../es/widgets/sort-by/sort-by.js | 3 +- .../es/widgets/stats/stats.js | 97 +- .../toggle-refinement/defaultTemplates.js | 5 +- .../toggle-refinement/toggle-refinement.js | 6 +- .../widgets/voice-search/defaultTemplates.js | 89 +- .../es/widgets/voice-search/voice-search.js | 3 +- .../templates/autocomplete.php | 49 +- .../templates/instantsearch.php | 10 +- 273 files changed, 10611 insertions(+), 8874 deletions(-) diff --git a/wp-search-with-algolia/README.txt b/wp-search-with-algolia/README.txt index 4ba3ae4..b679564 100644 --- a/wp-search-with-algolia/README.txt +++ b/wp-search-with-algolia/README.txt @@ -2,9 +2,9 @@ Contributors: WebDevStudios, williamsba1, gregrickaby, tw2113, richaber, mrasharirfan Tags: Search, Algolia, Autocomplete, instant-search, relevant search, search highlight, faceted search, find-as-you-type search, suggest, search by category, ajax search, better search, custom search Requires at least: 5.0 -Tested up to: 6.0 +Tested up to: 6.1 Requires PHP: 7.2 -Stable tag: 2.2.0 +Stable tag: 2.3.0 License: GNU General Public License v2.0, MIT License Improve search on your site. Autocomplete is included, along with full control over look, feel and relevance. @@ -106,6 +106,16 @@ WebDevStudios provides end-to-end WordPress opportunities from strategy and plan Follow along with the changelog on [Github](https://github.com/WebDevStudios/wp-search-with-algolia/releases). += 2.3.0 = +* Add algolia_should_override_autocomplete filter to override enable/disable status of Autocomplete +* Add from_batch argument to the re-index WP-CLI command +* Update excluded custom post types and taxonomies to include Core WordPress' internal CPTs and taxonomies +* Update Algolia logos to match the latest version +* Remove jQuery usage and dependency from templates +* Update Algolia JavaScript API Client to 4.14.2 +* Update Algolia InstantSearch.js to 4.49.0 +* Update Algolia PHP API Client to 3.3.2 + = 2.2.0 = * Add alert to Push Settings button on the Search Page. * Replace attributesToIndex index setting with searchableAttributes. @@ -113,7 +123,7 @@ Follow along with the changelog on [Github](https://github.com/WebDevStudios/wp- * Improve drag and drop column description text on the Autocomplete page. * Remove inline CSS for Max. Suggestions input. * Update Algolia JavaScript API Client to 4.13.0 -* Update Algolia InstantSearch.js to 4.50.5 +* Update Algolia InstantSearch.js to 4.40.5 * Update Algolia Autocomplete.js to 0.38.1 * Update Algolia PHP API Client to 3.2.0 diff --git a/wp-search-with-algolia/algolia.php b/wp-search-with-algolia/algolia.php index f5c4818..4a6deee 100644 --- a/wp-search-with-algolia/algolia.php +++ b/wp-search-with-algolia/algolia.php @@ -3,7 +3,7 @@ * Plugin Name: WP Search with Algolia * Plugin URI: https://github.com/WebDevStudios/wp-search-with-algolia * Description: Integrate the powerful Algolia search service with WordPress - * Version: 2.2.0 + * Version: 2.3.0 * Requires at least: 5.0 * Requires PHP: 7.2 * Author: WebDevStudios @@ -26,7 +26,7 @@ } // The Algolia Search plugin version. -define( 'ALGOLIA_VERSION', '2.2.0' ); +define( 'ALGOLIA_VERSION', '2.3.0' ); // The minmum required PHP version. define( 'ALGOLIA_MIN_PHP_VERSION', '7.2' ); diff --git a/wp-search-with-algolia/includes/admin/class-algolia-admin-page-autocomplete.php b/wp-search-with-algolia/includes/admin/class-algolia-admin-page-autocomplete.php index a0ecbd7..b1f969f 100644 --- a/wp-search-with-algolia/includes/admin/class-algolia-admin-page-autocomplete.php +++ b/wp-search-with-algolia/includes/admin/class-algolia-admin-page-autocomplete.php @@ -102,12 +102,12 @@ public function __construct( Algolia_Settings $settings, Algolia_Autocomplete_Co */ public function add_page() { add_menu_page( - 'Algolia Search', + esc_html__( 'Algolia Search', 'wp-search-with-algolia' ), esc_html__( 'Algolia Search', 'wp-search-with-algolia' ), 'manage_options', 'algolia', array( $this, 'display_page' ), - '' + '' ); add_submenu_page( 'algolia', diff --git a/wp-search-with-algolia/includes/admin/css/algolia-admin.css b/wp-search-with-algolia/includes/admin/css/algolia-admin.css index 5974914..1c7a13d 100644 --- a/wp-search-with-algolia/includes/admin/css/algolia-admin.css +++ b/wp-search-with-algolia/includes/admin/css/algolia-admin.css @@ -15,24 +15,6 @@ cursor: pointer; } -/* Override admin menu styles for the algolia icon */ -@font-face { - font-family: 'algolia'; - src: url('../fonts/algolia.eot?nqrd9q'); - src: url('../fonts/algolia.eot?nqrd9q#iefix') format('embedded-opentype'), - url('../fonts/algolia.ttf?nqrd9q') format('truetype'), - url('../fonts/algolia.woff?nqrd9q') format('woff'), - url('../fonts/algolia.svg?nqrd9q#algolia') format('svg'); - font-weight: normal; - font-style: normal; -} - -#adminmenu #toplevel_page_algolia .menu-icon-generic div.wp-menu-image:before, -#adminmenu #toplevel_page_algolia-account-settings .menu-icon-generic div.wp-menu-image:before { - font-family: algolia !important; - content: "\e900"; -} - /* Logs page */ .log-details { background-color: #32373c; diff --git a/wp-search-with-algolia/includes/class-algolia-cli.php b/wp-search-with-algolia/includes/class-algolia-cli.php index b0c835a..cb2cba0 100644 --- a/wp-search-with-algolia/includes/class-algolia-cli.php +++ b/wp-search-with-algolia/includes/class-algolia-cli.php @@ -51,6 +51,9 @@ public function __construct() { * [--all] * : Re-indexes all the enabled indices. * + * [--from_batch=] + * : Re-index starting from the provided batch (instead of the first page). + * * ## EXAMPLES * * wp algolia re-index @@ -68,9 +71,10 @@ public function reindex( $args, $assoc_args ) { WP_CLI::error( 'The configuration for this website does not allow to contact the Algolia API.' ); } - $index_id = isset( $args[0] ) ? $args[0] : null; - $clear = WP_CLI\Utils\get_flag_value( $assoc_args, 'clear' ); - $all = WP_CLI\Utils\get_flag_value( $assoc_args, 'all' ); + $index_id = isset( $args[0] ) ? $args[0] : null; + $clear = WP_CLI\Utils\get_flag_value( $assoc_args, 'clear' ); + $all = WP_CLI\Utils\get_flag_value( $assoc_args, 'all' ); + $from_batch = intval( WP_CLI\Utils\get_flag_value( $assoc_args, 'from_batch', 1 ) ); if ( ! $index_id && ! $all ) { WP_CLI::error( 'You need to either provide an index name or specify the --all argument to re-index all enabled indices.' ); @@ -95,7 +99,7 @@ public function reindex( $args, $assoc_args ) { } foreach ( $indices as $index ) { - $this->do_reindex( $index, $clear ); + $this->do_reindex( $index, $clear, $from_batch ); } } @@ -107,10 +111,11 @@ public function reindex( $args, $assoc_args ) { * * @param Algolia_Index $index Algolia_Index instance. * @param bool $clear Clear all existing records prior to pushing the records. + * @param int $from_batch The batch to start indexing from. * * @return void */ - private function do_reindex( Algolia_Index $index, $clear ) { + private function do_reindex( Algolia_Index $index, $clear, $from_batch ) { if ( $clear ) { /* translators: the placeholder will contain the name of the index. */ @@ -120,7 +125,12 @@ private function do_reindex( Algolia_Index $index, $clear ) { WP_CLI::success( sprintf( __( 'Correctly cleared index "%s".', 'wp-search-with-algolia' ), $index->get_name() ) ); } - $total_pages = $index->get_re_index_max_num_pages(); + $total_pages = $index->get_re_index_max_num_pages() - ( $from_batch - 1 ); + + if ( 0 > $total_pages ) { + WP_CLI::error( 'from_batch value for re-indexing is out of bounds.' ); + return; + } if ( 0 === $total_pages ) { $index->re_index( 1 ); @@ -129,16 +139,18 @@ private function do_reindex( Algolia_Index $index, $clear ) { return; } - $progress = WP_CLI\Utils\make_progress_bar( sprintf( 'Processing %s pages of results.', $total_pages ), $total_pages ); + $progress = WP_CLI\Utils\make_progress_bar( sprintf( 'Processing %s batches of results.', $total_pages ), $total_pages ); - $page = 1; + $page = $from_batch; do { + WP_CLI::log( sprintf( 'Indexing batch %s.', $page ) ); $index->re_index( $page++ ); + WP_CLI::log( sprintf( 'Indexed batch %s.', ( $page - 1 ) ) ); $progress->tick(); } while ( $page <= $total_pages ); $progress->finish(); - WP_CLI::success( sprintf( 'Indexed "%s" pages of results inside index "%s"', $total_pages, $index->get_name() ) ); + WP_CLI::success( sprintf( 'Indexed "%s" batches of results inside index "%s"', $total_pages, $index->get_name() ) ); } } diff --git a/wp-search-with-algolia/includes/class-algolia-scripts.php b/wp-search-with-algolia/includes/class-algolia-scripts.php index 52394a4..5817239 100644 --- a/wp-search-with-algolia/includes/class-algolia-scripts.php +++ b/wp-search-with-algolia/includes/class-algolia-scripts.php @@ -45,7 +45,6 @@ public function register_scripts() { 'algolia-search', ALGOLIA_PLUGIN_URL . 'js/algoliasearch/dist/algoliasearch-lite.umd.js', [ - 'jquery', 'underscore', 'wp-util', ], @@ -57,7 +56,6 @@ public function register_scripts() { 'algolia-autocomplete', ALGOLIA_PLUGIN_URL . 'js/autocomplete.js/dist/autocomplete' . $suffix . '.js', [ - 'jquery', 'underscore', 'wp-util', 'algolia-search', @@ -80,7 +78,6 @@ public function register_scripts() { 'algolia-instantsearch', ALGOLIA_PLUGIN_URL . 'js/instantsearch.js/dist/instantsearch' . $ais_suffix . $suffix . '.js', [ - 'jquery', 'underscore', 'wp-util', 'algolia-search', diff --git a/wp-search-with-algolia/includes/class-algolia-settings.php b/wp-search-with-algolia/includes/class-algolia-settings.php index 0984ee2..3c9ba60 100644 --- a/wp-search-with-algolia/includes/class-algolia-settings.php +++ b/wp-search-with-algolia/includes/class-algolia-settings.php @@ -150,10 +150,15 @@ public function get_excluded_post_types() { // Native WordPress. $excluded[] = 'revision'; - - // Native to Algolia Search plugin. - $excluded[] = 'algolia_task'; - $excluded[] = 'algolia_log'; + $excluded[] = 'custom_css'; + $excluded[] = 'customize_changeset'; + $excluded[] = 'oembed_cache'; + $excluded[] = 'user_request'; + $excluded[] = 'wp_block'; + $excluded[] = 'wp_global_styles'; + $excluded[] = 'wp_navigation'; + $excluded[] = 'wp_template'; + $excluded[] = 'wp_template_part'; // Native to WordPress VIP platform. $excluded[] = 'kr_request_token'; @@ -228,6 +233,8 @@ public function get_excluded_taxonomies() { 'nav_menu', 'link_category', 'post_format', + 'wp_theme', + 'wp_template_part_area', ]; /** @@ -259,7 +266,16 @@ public function get_excluded_taxonomies() { * @return string Can be 'yes' or 'no'. */ public function get_autocomplete_enabled() { - return get_option( 'algolia_autocomplete_enabled', 'no' ); + $enabled = get_option( 'algolia_autocomplete_enabled', 'no' ); + + /** + * Filters the autocomplete enabled option for algolia search. + * + * @since 2.3.0 + * + * @param string $enabled Can be 'yes' or 'no'. + */ + return apply_filters( 'algolia_should_override_autocomplete', $enabled ); } /** diff --git a/wp-search-with-algolia/includes/class-algolia-template-loader.php b/wp-search-with-algolia/includes/class-algolia-template-loader.php index a12230f..17d1d5f 100644 --- a/wp-search-with-algolia/includes/class-algolia-template-loader.php +++ b/wp-search-with-algolia/includes/class-algolia-template-loader.php @@ -83,7 +83,7 @@ public function load_algolia_config() { 'query' => get_search_query(), 'autocomplete' => array( 'sources' => $autocomplete_config->get_config(), - 'input_selector' => (string) apply_filters( 'algolia_autocomplete_input_selector', "input[name='s']:not('.no-autocomplete')" ), + 'input_selector' => (string) apply_filters( 'algolia_autocomplete_input_selector', "input[name='s']:not(.no-autocomplete):not(#adminbar-search)" ), ), 'indices' => array(), ); diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Algolia.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Algolia.php index 8033b23..832f328 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Algolia.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Algolia.php @@ -10,7 +10,7 @@ final class Algolia { - const VERSION = '3.2.0'; + const VERSION = '3.3.2'; /** * Holds an instance of the simple cache repository (PSR-16). diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Request.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Request.php index b4cbc16..d863695 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Request.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Request.php @@ -67,6 +67,9 @@ public function __construct( } } + /** + * @return string|null + */ public function getRequestTarget() { if (null !== $this->requestTarget) { @@ -84,6 +87,9 @@ public function getRequestTarget() return $target; } + /** + * @return Request + */ public function withRequestTarget($requestTarget) { if (preg_match('#\s#', $requestTarget)) { @@ -96,11 +102,17 @@ public function withRequestTarget($requestTarget) return $new; } + /** + * @return string + */ public function getMethod() { return $this->method; } + /** + * @return Request + */ public function withMethod($method) { $new = clone $this; @@ -109,11 +121,17 @@ public function withMethod($method) return $new; } + /** + * @return Uri|UriInterface|string + */ public function getUri() { return $this->uri; } + /** + * @return Request + */ public function withUri(UriInterface $uri, $preserveHost = false) { if ($uri === $this->uri) { @@ -130,6 +148,9 @@ public function withUri(UriInterface $uri, $preserveHost = false) return $new; } + /** + * @return void + */ private function updateHostFromUri() { $host = $this->uri->getHost(); @@ -153,11 +174,17 @@ private function updateHostFromUri() $this->headers = [$header => [$host]] + $this->headers; } + /** + * @return string + */ public function getProtocolVersion() { return $this->protocol; } + /** + * @return Request + */ public function withProtocolVersion($version) { if ($this->protocol === $version) { @@ -169,16 +196,25 @@ public function withProtocolVersion($version) return $new; } + /** + * @return array + */ public function getHeaders() { return $this->headers; } + /** + * @return bool + */ public function hasHeader($header) { return isset($this->headerNames[strtolower($header)]); } + /** + * @return array|mixed + */ public function getHeader($header) { $header = strtolower($header); @@ -190,11 +226,17 @@ public function getHeader($header) return $this->headers[$header]; } + /** + * @return string + */ public function getHeaderLine($header) { return implode(', ', $this->getHeader($header)); } + /** + * @return Request + */ public function withHeader($header, $value) { if (!is_array($value)) { @@ -212,6 +254,9 @@ public function withHeader($header, $value) return $new; } + /** + * @return Request + */ public function withAddedHeader($header, $value) { if (!is_array($value)) { @@ -231,6 +276,9 @@ public function withAddedHeader($header, $value) return $new; } + /** + * @return Request + */ public function withoutHeader($header) { $normalized = strtolower($header); @@ -244,6 +292,9 @@ public function withoutHeader($header) return $new; } + /** + * @return PumpStream|Stream|StreamInterface + */ public function getBody() { if (!$this->stream) { @@ -253,6 +304,9 @@ public function getBody() return $this->stream; } + /** + * @return Request + */ public function withBody(StreamInterface $body) { if ($body === $this->stream) { @@ -264,6 +318,9 @@ public function withBody(StreamInterface $body) return $new; } + /** + * @return void + */ private function setHeaders(array $headers) { $this->headerNames = $this->headers = []; diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Response.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Response.php index b16f702..b804605 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Response.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Response.php @@ -125,16 +125,25 @@ public function __construct( $this->protocol = $version; } + /** + * @return int + */ public function getStatusCode() { return $this->statusCode; } + /** + * @return string + */ public function getReasonPhrase() { return $this->reasonPhrase; } + /** + * @return static + */ public function withStatus($code, $reasonPhrase = '') { $new = clone $this; @@ -147,11 +156,17 @@ public function withStatus($code, $reasonPhrase = '') return $new; } + /** + * @return string + */ public function getProtocolVersion() { return $this->protocol; } + /** + * @return static + */ public function withProtocolVersion($version) { if ($this->protocol === $version) { @@ -164,16 +179,25 @@ public function withProtocolVersion($version) return $new; } + /** + * @return array + */ public function getHeaders() { return $this->headers; } + /** + * @return bool + */ public function hasHeader($header) { return isset($this->headerNames[strtolower($header)]); } + /** + * @return array + */ public function getHeader($header) { $header = strtolower($header); @@ -187,11 +211,17 @@ public function getHeader($header) return $this->headers[$header]; } + /** + * @return string + */ public function getHeaderLine($header) { return implode(', ', $this->getHeader($header)); } + /** + * @return static + */ public function withHeader($header, $value) { if (!is_array($value)) { @@ -211,6 +241,9 @@ public function withHeader($header, $value) return $new; } + /** + * @return static + */ public function withAddedHeader($header, $value) { if (!is_array($value)) { @@ -232,6 +265,9 @@ public function withAddedHeader($header, $value) return $new; } + /** + * @return static + */ public function withoutHeader($header) { $normalized = strtolower($header); @@ -248,6 +284,9 @@ public function withoutHeader($header) return $new; } + /** + * @return \StreamInterface + */ public function getBody() { if (!$this->stream) { @@ -257,6 +296,9 @@ public function getBody() return $this->stream; } + /** + * @return static + */ public function withBody(StreamInterface $body) { if ($body === $this->stream) { diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Stream.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Stream.php index 818864e..77a0ea4 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Stream.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Stream.php @@ -96,6 +96,9 @@ public function __toString() } } + /** + * @return string + */ public function getContents() { if (!isset($this->stream)) { @@ -111,6 +114,9 @@ public function getContents() return $contents; } + /** + * @return void + */ public function close() { if (isset($this->stream)) { @@ -121,6 +127,9 @@ public function close() } } + /** + * @return resource|null + */ public function detach() { if (!isset($this->stream)) { @@ -135,6 +144,9 @@ public function detach() return $result; } + /** + * @return mixed|null + */ public function getSize() { if (null !== $this->size) { @@ -160,21 +172,33 @@ public function getSize() return null; } + /** + * @return bool + */ public function isReadable() { return $this->readable; } + /** + * @return bool + */ public function isWritable() { return $this->writable; } + /** + * @return bool|mixed + */ public function isSeekable() { return $this->seekable; } + /** + * @return bool + */ public function eof() { if (!isset($this->stream)) { @@ -184,6 +208,9 @@ public function eof() return feof($this->stream); } + /** + * @return int + */ public function tell() { if (!isset($this->stream)) { @@ -199,11 +226,17 @@ public function tell() return $result; } + /** + * @return void + */ public function rewind() { $this->seek(0); } + /** + * @return void + */ public function seek($offset, $whence = SEEK_SET) { if (!isset($this->stream)) { @@ -217,6 +250,9 @@ public function seek($offset, $whence = SEEK_SET) } } + /** + * @return string + */ public function read($length) { if (!isset($this->stream)) { @@ -241,6 +277,9 @@ public function read($length) return $string; } + /** + * @return int + */ public function write($string) { if (!isset($this->stream)) { @@ -261,6 +300,9 @@ public function write($string) return $result; } + /** + * @return array|mixed|null + */ public function getMetadata($key = null) { if (!isset($this->stream)) { diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Uri.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Uri.php index 10ac02a..427722a 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Uri.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Http/Psr7/Uri.php @@ -384,11 +384,17 @@ public static function fromParts(array $parts) return $uri; } + /** + * @return string + */ public function getScheme() { return $this->scheme; } + /** + * @return string + */ public function getAuthority() { $authority = $this->host; @@ -403,36 +409,57 @@ public function getAuthority() return $authority; } + /** + * @return string + */ public function getUserInfo() { return $this->userInfo; } + /** + * @return string + */ public function getHost() { return $this->host; } + /** + * @return int|null + */ public function getPort() { return $this->port; } + /** + * @return string + */ public function getPath() { return $this->path; } + /** + * @return string + */ public function getQuery() { return $this->query; } + /** + * @return string + */ public function getFragment() { return $this->fragment; } + /** + * @return Uri + */ public function withScheme($scheme) { $scheme = $this->filterScheme($scheme); @@ -449,6 +476,9 @@ public function withScheme($scheme) return $new; } + /** + * @return Uri + */ public function withUserInfo($user, $password = null) { $info = $user; @@ -467,6 +497,9 @@ public function withUserInfo($user, $password = null) return $new; } + /** + * @return Uri + */ public function withHost($host) { $host = $this->filterHost($host); @@ -482,6 +515,9 @@ public function withHost($host) return $new; } + /** + * @return Uri + */ public function withPort($port) { $port = $this->filterPort($port); @@ -498,6 +534,9 @@ public function withPort($port) return $new; } + /** + * @return Uri + */ public function withPath($path) { $path = $this->filterPath($path); @@ -513,6 +552,9 @@ public function withPath($path) return $new; } + /** + * @return Uri + */ public function withQuery($query) { $query = $this->filterQueryAndFragment($query); @@ -527,6 +569,9 @@ public function withQuery($query) return $new; } + /** + * @return Uri + */ public function withFragment($fragment) { $fragment = $this->filterQueryAndFragment($fragment); diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/InsightsClient.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/InsightsClient.php index 237f2e9..a7fd16d 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/InsightsClient.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/InsightsClient.php @@ -5,12 +5,13 @@ use Algolia\AlgoliaSearch\Config\InsightsConfig; use Algolia\AlgoliaSearch\Insights\UserInsightsClient; use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapper; +use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapperInterface; use Algolia\AlgoliaSearch\RetryStrategy\ClusterHosts; final class InsightsClient { /** - * @var \Algolia\AlgoliaSearch\RetryStrategy\ApiWrapper + * @var ApiWrapperInterface */ private $api; @@ -19,7 +20,7 @@ final class InsightsClient */ private $config; - public function __construct(ApiWrapper $api, InsightsConfig $config) + public function __construct(ApiWrapperInterface $api, InsightsConfig $config) { $this->api = $api; $this->config = $config; diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/PersonalizationClient.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/PersonalizationClient.php index 7dc7fa5..633a270 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/PersonalizationClient.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/PersonalizationClient.php @@ -5,12 +5,13 @@ use Algolia\AlgoliaSearch\Config\PersonalizationConfig; use Algolia\AlgoliaSearch\RequestOptions\RequestOptions; use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapper; +use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapperInterface; use Algolia\AlgoliaSearch\RetryStrategy\ClusterHosts; final class PersonalizationClient { /** - * @var \Algolia\AlgoliaSearch\RetryStrategy\ApiWrapper + * @var ApiWrapperInterface */ private $api; @@ -22,7 +23,7 @@ final class PersonalizationClient /** * RecommendationClient constructor. */ - public function __construct(ApiWrapper $api, PersonalizationConfig $config) + public function __construct(ApiWrapperInterface $api, PersonalizationConfig $config) { $this->api = $api; $this->config = $config; diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendClient.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendClient.php index 46432b3..f76217f 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendClient.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendClient.php @@ -6,6 +6,7 @@ use Algolia\AlgoliaSearch\Exceptions\AlgoliaException; use Algolia\AlgoliaSearch\RequestOptions\RequestOptions; use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapper; +use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapperInterface; use Algolia\AlgoliaSearch\RetryStrategy\ClusterHosts; final class RecommendClient @@ -14,7 +15,7 @@ final class RecommendClient const BOUGHT_TOGETHER = 'bought-together'; /** - * @var ApiWrapper + * @var ApiWrapperInterface */ private $api; @@ -23,7 +24,7 @@ final class RecommendClient */ private $config; - public function __construct(ApiWrapper $api, RecommendConfig $config) + public function __construct(ApiWrapperInterface $api, RecommendConfig $config) { $this->api = $api; $this->config = $config; diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendationClient.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendationClient.php index e1988a0..a028220 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendationClient.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/RecommendationClient.php @@ -5,6 +5,7 @@ use Algolia\AlgoliaSearch\Config\RecommendationConfig; use Algolia\AlgoliaSearch\RequestOptions\RequestOptions; use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapper; +use Algolia\AlgoliaSearch\RetryStrategy\ApiWrapperInterface; use Algolia\AlgoliaSearch\RetryStrategy\ClusterHosts; /** @@ -13,7 +14,7 @@ final class RecommendationClient { /** - * @var \Algolia\AlgoliaSearch\RetryStrategy\ApiWrapper + * @var ApiWrapperInterface */ private $api; @@ -25,7 +26,7 @@ final class RecommendationClient /** * RecommendationClient constructor. */ - public function __construct(ApiWrapper $api, RecommendationConfig $config) + public function __construct(ApiWrapperInterface $api, RecommendationConfig $config) { $this->api = $api; $this->config = $config; diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/AbstractResponse.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/AbstractResponse.php index f888c94..305f84b 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/AbstractResponse.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/AbstractResponse.php @@ -21,6 +21,8 @@ public function getBody() /** * {@inheritdoc} + * + * @return bool */ #[\ReturnTypeWillChange] public function offsetExists($offset) @@ -30,6 +32,8 @@ public function offsetExists($offset) /** * {@inheritdoc} + * + * @return mixed */ #[\ReturnTypeWillChange] public function offsetGet($offset) @@ -39,6 +43,8 @@ public function offsetGet($offset) /** * {@inheritdoc} + * + * @return void */ #[\ReturnTypeWillChange] public function offsetSet($offset, $value) @@ -48,6 +54,8 @@ public function offsetSet($offset, $value) /** * {@inheritdoc} + * + * @return void */ #[\ReturnTypeWillChange] public function offsetUnset($offset) diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/BatchIndexingResponse.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/BatchIndexingResponse.php index b04c865..cb429e9 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/BatchIndexingResponse.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/BatchIndexingResponse.php @@ -47,30 +47,45 @@ public function count() return count($this->apiResponse); } + /** + * @return mixed + */ #[\ReturnTypeWillChange] public function current() { return $this->apiResponse[$this->key]; } + /** + * @return void + */ #[\ReturnTypeWillChange] public function next() { $this->key++; } + /** + * @return bool|float|int|mixed|string|null + */ #[\ReturnTypeWillChange] public function key() { return $this->key; } + /** + * @return bool + */ #[\ReturnTypeWillChange] public function valid() { return isset($this->apiResponse[$this->key]); } + /** + * @return void + */ #[\ReturnTypeWillChange] public function rewind() { diff --git a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/MultiResponse.php b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/MultiResponse.php index 7e3fbde..a8a4b7e 100644 --- a/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/MultiResponse.php +++ b/wp-search-with-algolia/includes/libraries/algoliasearch-client-php/src/Response/MultiResponse.php @@ -20,36 +20,54 @@ public function wait($requestOptions = []) return $this; } + /** + * @return int + */ #[\ReturnTypeWillChange] public function count() { return count($this->apiResponse); } + /** + * @return mixed + */ #[\ReturnTypeWillChange] public function current() { return $this->apiResponse[$this->key]; } + /** + * @return void + */ #[\ReturnTypeWillChange] public function next() { $this->key++; } + /** + * @return bool|float|int|mixed|string|null + */ #[\ReturnTypeWillChange] public function key() { return $this->key; } + /** + * @return bool + */ #[\ReturnTypeWillChange] public function valid() { return isset($this->apiResponse[$this->key]); } + /** + * @return void + */ #[\ReturnTypeWillChange] public function rewind() { diff --git a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.esm.browser.js b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.esm.browser.js index 4b86df1..3f11e1d 100644 --- a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.esm.browser.js +++ b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.esm.browser.js @@ -181,7 +181,7 @@ function encode(format, ...args) { return format.replace(/%s/g, () => encodeURIComponent(args[i++])); } -const version = '4.13.0'; +const version = '4.14.2'; const AuthMode = { /** diff --git a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.umd.js b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.umd.js index 2a9a6e8..0dad143 100644 --- a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.umd.js +++ b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch-lite.umd.js @@ -1,2 +1,2 @@ -/*! algoliasearch-lite.umd.js | 4.13.0 | © Algolia, inc. | https://github.com/algolia/algoliasearch-client-javascript */ -!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define(t):(e=e||self).algoliasearch=t()}(this,(function(){"use strict";function e(e,t,r){return t in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function t(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),r.push.apply(r,n)}return r}function r(r){for(var n=1;n=0||(o[r]=e[r]);return o}(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,r)&&(o[r]=e[r])}return o}function o(e,t){return function(e){if(Array.isArray(e))return e}(e)||function(e,t){if(!(Symbol.iterator in Object(e)||"[object Arguments]"===Object.prototype.toString.call(e)))return;var r=[],n=!0,o=!1,a=void 0;try{for(var u,i=e[Symbol.iterator]();!(n=(u=i.next()).done)&&(r.push(u.value),!t||r.length!==t);n=!0);}catch(e){o=!0,a=e}finally{try{n||null==i.return||i.return()}finally{if(o)throw a}}return r}(e,t)||function(){throw new TypeError("Invalid attempt to destructure non-iterable instance")}()}function a(e){return function(e){if(Array.isArray(e)){for(var t=0,r=new Array(e.length);t2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return Promise.resolve().then((function(){var r=JSON.stringify(e),n=a()[r];return Promise.all([n||t(),void 0!==n])})).then((function(e){var t=o(e,2),n=t[0],a=t[1];return Promise.all([n,a||r.miss(n)])})).then((function(e){return o(e,1)[0]}))},set:function(e,t){return Promise.resolve().then((function(){var o=a();return o[JSON.stringify(e)]=t,n().setItem(r,JSON.stringify(o)),t}))},delete:function(e){return Promise.resolve().then((function(){var t=a();delete t[JSON.stringify(e)],n().setItem(r,JSON.stringify(t))}))},clear:function(){return Promise.resolve().then((function(){n().removeItem(r)}))}}}function i(e){var t=a(e.caches),r=t.shift();return void 0===r?{get:function(e,t){var r=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},n=t();return n.then((function(e){return Promise.all([e,r.miss(e)])})).then((function(e){return o(e,1)[0]}))},set:function(e,t){return Promise.resolve(t)},delete:function(e){return Promise.resolve()},clear:function(){return Promise.resolve()}}:{get:function(e,n){var o=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return r.get(e,n,o).catch((function(){return i({caches:t}).get(e,n,o)}))},set:function(e,n){return r.set(e,n).catch((function(){return i({caches:t}).set(e,n)}))},delete:function(e){return r.delete(e).catch((function(){return i({caches:t}).delete(e)}))},clear:function(){return r.clear().catch((function(){return i({caches:t}).clear()}))}}}function s(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{serializable:!0},t={};return{get:function(r,n){var o=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},a=JSON.stringify(r);if(a in t)return Promise.resolve(e.serializable?JSON.parse(t[a]):t[a]);var u=n(),i=o&&o.miss||function(){return Promise.resolve()};return u.then((function(e){return i(e)})).then((function(){return u}))},set:function(r,n){return t[JSON.stringify(r)]=e.serializable?JSON.stringify(n):n,Promise.resolve(n)},delete:function(e){return delete t[JSON.stringify(e)],Promise.resolve()},clear:function(){return t={},Promise.resolve()}}}function c(e){for(var t=e.length-1;t>0;t--){var r=Math.floor(Math.random()*(t+1)),n=e[t];e[t]=e[r],e[r]=n}return e}function l(e,t){return t?(Object.keys(t).forEach((function(r){e[r]=t[r](e)})),e):e}function f(e){for(var t=arguments.length,r=new Array(t>1?t-1:0),n=1;n0?n:void 0,timeout:r.timeout||t,headers:r.headers||{},queryParameters:r.queryParameters||{},cacheable:r.cacheable}}var m={Read:1,Write:2,Any:3},p=1,v=2,y=3;function g(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:p;return r(r({},e),{},{status:t,lastUpdate:Date.now()})}function b(e){return"string"==typeof e?{protocol:"https",url:e,accept:m.Any}:{protocol:e.protocol||"https",url:e.url,accept:e.accept||m.Any}}var O="GET",P="POST";function q(e,t){return Promise.all(t.map((function(t){return e.get(t,(function(){return Promise.resolve(g(t))}))}))).then((function(e){var r=e.filter((function(e){return function(e){return e.status===p||Date.now()-e.lastUpdate>12e4}(e)})),n=e.filter((function(e){return function(e){return e.status===y&&Date.now()-e.lastUpdate<=12e4}(e)})),o=[].concat(a(r),a(n));return{getTimeout:function(e,t){return(0===n.length&&0===e?1:n.length+3+e)*t},statelessHosts:o.length>0?o.map((function(e){return b(e)})):t}}))}function w(e,t,n,o){var u=[],i=function(e,t){if(e.method===O||void 0===e.data&&void 0===t.data)return;var n=Array.isArray(e.data)?e.data:r(r({},e.data),t.data);return JSON.stringify(n)}(n,o),s=function(e,t){var n=r(r({},e.headers),t.headers),o={};return Object.keys(n).forEach((function(e){var t=n[e];o[e.toLowerCase()]=t})),o}(e,o),c=n.method,l=n.method!==O?{}:r(r({},n.data),o.data),f=r(r(r({"x-algolia-agent":e.userAgent.value},e.queryParameters),l),o.queryParameters),h=0,d=function t(r,a){var l=r.pop();if(void 0===l)throw{name:"RetryError",message:"Unreachable hosts - your application id may be incorrect. If the error persists, contact support@algolia.com.",transporterStackTrace:A(u)};var d={data:i,headers:s,method:c,url:S(l,n.path,f),connectTimeout:a(h,e.timeouts.connect),responseTimeout:a(h,o.timeout)},m=function(e){var t={request:d,response:e,host:l,triesLeft:r.length};return u.push(t),t},p={onSuccess:function(e){return function(e){try{return JSON.parse(e.content)}catch(t){throw function(e,t){return{name:"DeserializationError",message:e,response:t}}(t.message,e)}}(e)},onRetry:function(n){var o=m(n);return n.isTimedOut&&h++,Promise.all([e.logger.info("Retryable failure",x(o)),e.hostsCache.set(l,g(l,n.isTimedOut?y:v))]).then((function(){return t(r,a)}))},onFail:function(e){throw m(e),function(e,t){var r=e.content,n=e.status,o=r;try{o=JSON.parse(r).message}catch(e){}return function(e,t,r){return{name:"ApiError",message:e,status:t,transporterStackTrace:r}}(o,n,t)}(e,A(u))}};return e.requester.send(d).then((function(e){return function(e,t){return function(e){var t=e.status;return e.isTimedOut||function(e){var t=e.isTimedOut,r=e.status;return!t&&0==~~r}(e)||2!=~~(t/100)&&4!=~~(t/100)}(e)?t.onRetry(e):2==~~(e.status/100)?t.onSuccess(e):t.onFail(e)}(e,p)}))};return q(e.hostsCache,t).then((function(e){return d(a(e.statelessHosts).reverse(),e.getTimeout)}))}function j(e){var t={value:"Algolia for JavaScript (".concat(e,")"),add:function(e){var r="; ".concat(e.segment).concat(void 0!==e.version?" (".concat(e.version,")"):"");return-1===t.value.indexOf(r)&&(t.value="".concat(t.value).concat(r)),t}};return t}function S(e,t,r){var n=T(r),o="".concat(e.protocol,"://").concat(e.url,"/").concat("/"===t.charAt(0)?t.substr(1):t);return n.length&&(o+="?".concat(n)),o}function T(e){return Object.keys(e).map((function(t){return f("%s=%s",t,(r=e[t],"[object Object]"===Object.prototype.toString.call(r)||"[object Array]"===Object.prototype.toString.call(r)?JSON.stringify(e[t]):e[t]));var r})).join("&")}function A(e){return e.map((function(e){return x(e)}))}function x(e){var t=e.request.headers["x-algolia-api-key"]?{"x-algolia-api-key":"*****"}:{};return r(r({},e),{},{request:r(r({},e.request),{},{headers:r(r({},e.request.headers),t)})})}var N=function(e){var t=e.appId,n=function(e,t,r){var n={"x-algolia-api-key":r,"x-algolia-application-id":t};return{headers:function(){return e===h.WithinHeaders?n:{}},queryParameters:function(){return e===h.WithinQueryParameters?n:{}}}}(void 0!==e.authMode?e.authMode:h.WithinHeaders,t,e.apiKey),a=function(e){var t=e.hostsCache,r=e.logger,n=e.requester,a=e.requestsCache,u=e.responsesCache,i=e.timeouts,s=e.userAgent,c=e.hosts,l=e.queryParameters,f={hostsCache:t,logger:r,requester:n,requestsCache:a,responsesCache:u,timeouts:i,userAgent:s,headers:e.headers,queryParameters:l,hosts:c.map((function(e){return b(e)})),read:function(e,t){var r=d(t,f.timeouts.read),n=function(){return w(f,f.hosts.filter((function(e){return 0!=(e.accept&m.Read)})),e,r)};if(!0!==(void 0!==r.cacheable?r.cacheable:e.cacheable))return n();var a={request:e,mappedRequestOptions:r,transporter:{queryParameters:f.queryParameters,headers:f.headers}};return f.responsesCache.get(a,(function(){return f.requestsCache.get(a,(function(){return f.requestsCache.set(a,n()).then((function(e){return Promise.all([f.requestsCache.delete(a),e])}),(function(e){return Promise.all([f.requestsCache.delete(a),Promise.reject(e)])})).then((function(e){var t=o(e,2);t[0];return t[1]}))}))}),{miss:function(e){return f.responsesCache.set(a,e)}})},write:function(e,t){return w(f,f.hosts.filter((function(e){return 0!=(e.accept&m.Write)})),e,d(t,f.timeouts.write))}};return f}(r(r({hosts:[{url:"".concat(t,"-dsn.algolia.net"),accept:m.Read},{url:"".concat(t,".algolia.net"),accept:m.Write}].concat(c([{url:"".concat(t,"-1.algolianet.com")},{url:"".concat(t,"-2.algolianet.com")},{url:"".concat(t,"-3.algolianet.com")}]))},e),{},{headers:r(r(r({},n.headers()),{"content-type":"application/x-www-form-urlencoded"}),e.headers),queryParameters:r(r({},n.queryParameters()),e.queryParameters)}));return l({transporter:a,appId:t,addAlgoliaAgent:function(e,t){a.userAgent.add({segment:e,version:t})},clearCache:function(){return Promise.all([a.requestsCache.clear(),a.responsesCache.clear()]).then((function(){}))}},e.methods)},C=function(e){return function(t,r){return t.method===O?e.transporter.read(t,r):e.transporter.write(t,r)}},k=function(e){return function(t){var r=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},n={transporter:e.transporter,appId:e.appId,indexName:t};return l(n,r.methods)}},J=function(e){return function(t,n){var o=t.map((function(e){return r(r({},e),{},{params:T(e.params||{})})}));return e.transporter.read({method:P,path:"1/indexes/*/queries",data:{requests:o},cacheable:!0},n)}},E=function(e){return function(t,o){return Promise.all(t.map((function(t){var a=t.params,u=a.facetName,i=a.facetQuery,s=n(a,["facetName","facetQuery"]);return k(e)(t.indexName,{methods:{searchForFacetValues:R}}).searchForFacetValues(u,i,r(r({},o),s))})))}},I=function(e){return function(t,r,n){return e.transporter.read({method:P,path:f("1/answers/%s/prediction",e.indexName),data:{query:t,queryLanguages:r},cacheable:!0},n)}},F=function(e){return function(t,r){return e.transporter.read({method:P,path:f("1/indexes/%s/query",e.indexName),data:{query:t},cacheable:!0},r)}},R=function(e){return function(t,r,n){return e.transporter.read({method:P,path:f("1/indexes/%s/facets/%s/query",e.indexName,t),data:{facetQuery:r},cacheable:!0},n)}},D=1,W=2,H=3;function Q(e,t,n){var o,a={appId:e,apiKey:t,timeouts:{connect:1,read:2,write:30},requester:{send:function(e){return new Promise((function(t){var r=new XMLHttpRequest;r.open(e.method,e.url,!0),Object.keys(e.headers).forEach((function(t){return r.setRequestHeader(t,e.headers[t])}));var n,o=function(e,n){return setTimeout((function(){r.abort(),t({status:0,content:n,isTimedOut:!0})}),1e3*e)},a=o(e.connectTimeout,"Connection timeout");r.onreadystatechange=function(){r.readyState>r.OPENED&&void 0===n&&(clearTimeout(a),n=o(e.responseTimeout,"Socket timeout"))},r.onerror=function(){0===r.status&&(clearTimeout(a),clearTimeout(n),t({content:r.responseText||"Network request failed",status:r.status,isTimedOut:!1}))},r.onload=function(){clearTimeout(a),clearTimeout(n),t({content:r.responseText,status:r.status,isTimedOut:!1})},r.send(e.data)}))}},logger:(o=H,{debug:function(e,t){return D>=o&&console.debug(e,t),Promise.resolve()},info:function(e,t){return W>=o&&console.info(e,t),Promise.resolve()},error:function(e,t){return console.error(e,t),Promise.resolve()}}),responsesCache:s(),requestsCache:s({serializable:!1}),hostsCache:i({caches:[u({key:"".concat("4.13.0","-").concat(e)}),s()]}),userAgent:j("4.13.0").add({segment:"Browser",version:"lite"}),authMode:h.WithinQueryParameters};return N(r(r(r({},a),n),{},{methods:{search:J,searchForFacetValues:E,multipleQueries:J,multipleSearchForFacetValues:E,customRequest:C,initIndex:function(e){return function(t){return k(e)(t,{methods:{search:F,searchForFacetValues:R,findAnswers:I}})}}}}))}return Q.version="4.13.0",Q})); +/*! algoliasearch-lite.umd.js | 4.14.2 | © Algolia, inc. | https://github.com/algolia/algoliasearch-client-javascript */ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define(t):(e=e||self).algoliasearch=t()}(this,(function(){"use strict";function e(e,t,r){return t in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function t(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),r.push.apply(r,n)}return r}function r(r){for(var n=1;n=0||(o[r]=e[r]);return o}(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,r)&&(o[r]=e[r])}return o}function o(e,t){return function(e){if(Array.isArray(e))return e}(e)||function(e,t){if(!(Symbol.iterator in Object(e)||"[object Arguments]"===Object.prototype.toString.call(e)))return;var r=[],n=!0,o=!1,a=void 0;try{for(var u,i=e[Symbol.iterator]();!(n=(u=i.next()).done)&&(r.push(u.value),!t||r.length!==t);n=!0);}catch(e){o=!0,a=e}finally{try{n||null==i.return||i.return()}finally{if(o)throw a}}return r}(e,t)||function(){throw new TypeError("Invalid attempt to destructure non-iterable instance")}()}function a(e){return function(e){if(Array.isArray(e)){for(var t=0,r=new Array(e.length);t2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return Promise.resolve().then((function(){var r=JSON.stringify(e),n=a()[r];return Promise.all([n||t(),void 0!==n])})).then((function(e){var t=o(e,2),n=t[0],a=t[1];return Promise.all([n,a||r.miss(n)])})).then((function(e){return o(e,1)[0]}))},set:function(e,t){return Promise.resolve().then((function(){var o=a();return o[JSON.stringify(e)]=t,n().setItem(r,JSON.stringify(o)),t}))},delete:function(e){return Promise.resolve().then((function(){var t=a();delete t[JSON.stringify(e)],n().setItem(r,JSON.stringify(t))}))},clear:function(){return Promise.resolve().then((function(){n().removeItem(r)}))}}}function i(e){var t=a(e.caches),r=t.shift();return void 0===r?{get:function(e,t){var r=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},n=t();return n.then((function(e){return Promise.all([e,r.miss(e)])})).then((function(e){return o(e,1)[0]}))},set:function(e,t){return Promise.resolve(t)},delete:function(e){return Promise.resolve()},clear:function(){return Promise.resolve()}}:{get:function(e,n){var o=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return r.get(e,n,o).catch((function(){return i({caches:t}).get(e,n,o)}))},set:function(e,n){return r.set(e,n).catch((function(){return i({caches:t}).set(e,n)}))},delete:function(e){return r.delete(e).catch((function(){return i({caches:t}).delete(e)}))},clear:function(){return r.clear().catch((function(){return i({caches:t}).clear()}))}}}function s(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{serializable:!0},t={};return{get:function(r,n){var o=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},a=JSON.stringify(r);if(a in t)return Promise.resolve(e.serializable?JSON.parse(t[a]):t[a]);var u=n(),i=o&&o.miss||function(){return Promise.resolve()};return u.then((function(e){return i(e)})).then((function(){return u}))},set:function(r,n){return t[JSON.stringify(r)]=e.serializable?JSON.stringify(n):n,Promise.resolve(n)},delete:function(e){return delete t[JSON.stringify(e)],Promise.resolve()},clear:function(){return t={},Promise.resolve()}}}function c(e){for(var t=e.length-1;t>0;t--){var r=Math.floor(Math.random()*(t+1)),n=e[t];e[t]=e[r],e[r]=n}return e}function l(e,t){return t?(Object.keys(t).forEach((function(r){e[r]=t[r](e)})),e):e}function f(e){for(var t=arguments.length,r=new Array(t>1?t-1:0),n=1;n0?n:void 0,timeout:r.timeout||t,headers:r.headers||{},queryParameters:r.queryParameters||{},cacheable:r.cacheable}}var m={Read:1,Write:2,Any:3},p=1,v=2,y=3;function g(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:p;return r(r({},e),{},{status:t,lastUpdate:Date.now()})}function b(e){return"string"==typeof e?{protocol:"https",url:e,accept:m.Any}:{protocol:e.protocol||"https",url:e.url,accept:e.accept||m.Any}}var O="GET",P="POST";function q(e,t){return Promise.all(t.map((function(t){return e.get(t,(function(){return Promise.resolve(g(t))}))}))).then((function(e){var r=e.filter((function(e){return function(e){return e.status===p||Date.now()-e.lastUpdate>12e4}(e)})),n=e.filter((function(e){return function(e){return e.status===y&&Date.now()-e.lastUpdate<=12e4}(e)})),o=[].concat(a(r),a(n));return{getTimeout:function(e,t){return(0===n.length&&0===e?1:n.length+3+e)*t},statelessHosts:o.length>0?o.map((function(e){return b(e)})):t}}))}function w(e,t,n,o){var u=[],i=function(e,t){if(e.method===O||void 0===e.data&&void 0===t.data)return;var n=Array.isArray(e.data)?e.data:r(r({},e.data),t.data);return JSON.stringify(n)}(n,o),s=function(e,t){var n=r(r({},e.headers),t.headers),o={};return Object.keys(n).forEach((function(e){var t=n[e];o[e.toLowerCase()]=t})),o}(e,o),c=n.method,l=n.method!==O?{}:r(r({},n.data),o.data),f=r(r(r({"x-algolia-agent":e.userAgent.value},e.queryParameters),l),o.queryParameters),h=0,d=function t(r,a){var l=r.pop();if(void 0===l)throw{name:"RetryError",message:"Unreachable hosts - your application id may be incorrect. If the error persists, contact support@algolia.com.",transporterStackTrace:A(u)};var d={data:i,headers:s,method:c,url:S(l,n.path,f),connectTimeout:a(h,e.timeouts.connect),responseTimeout:a(h,o.timeout)},m=function(e){var t={request:d,response:e,host:l,triesLeft:r.length};return u.push(t),t},p={onSuccess:function(e){return function(e){try{return JSON.parse(e.content)}catch(t){throw function(e,t){return{name:"DeserializationError",message:e,response:t}}(t.message,e)}}(e)},onRetry:function(n){var o=m(n);return n.isTimedOut&&h++,Promise.all([e.logger.info("Retryable failure",x(o)),e.hostsCache.set(l,g(l,n.isTimedOut?y:v))]).then((function(){return t(r,a)}))},onFail:function(e){throw m(e),function(e,t){var r=e.content,n=e.status,o=r;try{o=JSON.parse(r).message}catch(e){}return function(e,t,r){return{name:"ApiError",message:e,status:t,transporterStackTrace:r}}(o,n,t)}(e,A(u))}};return e.requester.send(d).then((function(e){return function(e,t){return function(e){var t=e.status;return e.isTimedOut||function(e){var t=e.isTimedOut,r=e.status;return!t&&0==~~r}(e)||2!=~~(t/100)&&4!=~~(t/100)}(e)?t.onRetry(e):2==~~(e.status/100)?t.onSuccess(e):t.onFail(e)}(e,p)}))};return q(e.hostsCache,t).then((function(e){return d(a(e.statelessHosts).reverse(),e.getTimeout)}))}function j(e){var t={value:"Algolia for JavaScript (".concat(e,")"),add:function(e){var r="; ".concat(e.segment).concat(void 0!==e.version?" (".concat(e.version,")"):"");return-1===t.value.indexOf(r)&&(t.value="".concat(t.value).concat(r)),t}};return t}function S(e,t,r){var n=T(r),o="".concat(e.protocol,"://").concat(e.url,"/").concat("/"===t.charAt(0)?t.substr(1):t);return n.length&&(o+="?".concat(n)),o}function T(e){return Object.keys(e).map((function(t){return f("%s=%s",t,(r=e[t],"[object Object]"===Object.prototype.toString.call(r)||"[object Array]"===Object.prototype.toString.call(r)?JSON.stringify(e[t]):e[t]));var r})).join("&")}function A(e){return e.map((function(e){return x(e)}))}function x(e){var t=e.request.headers["x-algolia-api-key"]?{"x-algolia-api-key":"*****"}:{};return r(r({},e),{},{request:r(r({},e.request),{},{headers:r(r({},e.request.headers),t)})})}var N=function(e){var t=e.appId,n=function(e,t,r){var n={"x-algolia-api-key":r,"x-algolia-application-id":t};return{headers:function(){return e===h.WithinHeaders?n:{}},queryParameters:function(){return e===h.WithinQueryParameters?n:{}}}}(void 0!==e.authMode?e.authMode:h.WithinHeaders,t,e.apiKey),a=function(e){var t=e.hostsCache,r=e.logger,n=e.requester,a=e.requestsCache,u=e.responsesCache,i=e.timeouts,s=e.userAgent,c=e.hosts,l=e.queryParameters,f={hostsCache:t,logger:r,requester:n,requestsCache:a,responsesCache:u,timeouts:i,userAgent:s,headers:e.headers,queryParameters:l,hosts:c.map((function(e){return b(e)})),read:function(e,t){var r=d(t,f.timeouts.read),n=function(){return w(f,f.hosts.filter((function(e){return 0!=(e.accept&m.Read)})),e,r)};if(!0!==(void 0!==r.cacheable?r.cacheable:e.cacheable))return n();var a={request:e,mappedRequestOptions:r,transporter:{queryParameters:f.queryParameters,headers:f.headers}};return f.responsesCache.get(a,(function(){return f.requestsCache.get(a,(function(){return f.requestsCache.set(a,n()).then((function(e){return Promise.all([f.requestsCache.delete(a),e])}),(function(e){return Promise.all([f.requestsCache.delete(a),Promise.reject(e)])})).then((function(e){var t=o(e,2);t[0];return t[1]}))}))}),{miss:function(e){return f.responsesCache.set(a,e)}})},write:function(e,t){return w(f,f.hosts.filter((function(e){return 0!=(e.accept&m.Write)})),e,d(t,f.timeouts.write))}};return f}(r(r({hosts:[{url:"".concat(t,"-dsn.algolia.net"),accept:m.Read},{url:"".concat(t,".algolia.net"),accept:m.Write}].concat(c([{url:"".concat(t,"-1.algolianet.com")},{url:"".concat(t,"-2.algolianet.com")},{url:"".concat(t,"-3.algolianet.com")}]))},e),{},{headers:r(r(r({},n.headers()),{"content-type":"application/x-www-form-urlencoded"}),e.headers),queryParameters:r(r({},n.queryParameters()),e.queryParameters)}));return l({transporter:a,appId:t,addAlgoliaAgent:function(e,t){a.userAgent.add({segment:e,version:t})},clearCache:function(){return Promise.all([a.requestsCache.clear(),a.responsesCache.clear()]).then((function(){}))}},e.methods)},C=function(e){return function(t,r){return t.method===O?e.transporter.read(t,r):e.transporter.write(t,r)}},k=function(e){return function(t){var r=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},n={transporter:e.transporter,appId:e.appId,indexName:t};return l(n,r.methods)}},J=function(e){return function(t,n){var o=t.map((function(e){return r(r({},e),{},{params:T(e.params||{})})}));return e.transporter.read({method:P,path:"1/indexes/*/queries",data:{requests:o},cacheable:!0},n)}},E=function(e){return function(t,o){return Promise.all(t.map((function(t){var a=t.params,u=a.facetName,i=a.facetQuery,s=n(a,["facetName","facetQuery"]);return k(e)(t.indexName,{methods:{searchForFacetValues:R}}).searchForFacetValues(u,i,r(r({},o),s))})))}},I=function(e){return function(t,r,n){return e.transporter.read({method:P,path:f("1/answers/%s/prediction",e.indexName),data:{query:t,queryLanguages:r},cacheable:!0},n)}},F=function(e){return function(t,r){return e.transporter.read({method:P,path:f("1/indexes/%s/query",e.indexName),data:{query:t},cacheable:!0},r)}},R=function(e){return function(t,r,n){return e.transporter.read({method:P,path:f("1/indexes/%s/facets/%s/query",e.indexName,t),data:{facetQuery:r},cacheable:!0},n)}},D=1,W=2,H=3;function Q(e,t,n){var o,a={appId:e,apiKey:t,timeouts:{connect:1,read:2,write:30},requester:{send:function(e){return new Promise((function(t){var r=new XMLHttpRequest;r.open(e.method,e.url,!0),Object.keys(e.headers).forEach((function(t){return r.setRequestHeader(t,e.headers[t])}));var n,o=function(e,n){return setTimeout((function(){r.abort(),t({status:0,content:n,isTimedOut:!0})}),1e3*e)},a=o(e.connectTimeout,"Connection timeout");r.onreadystatechange=function(){r.readyState>r.OPENED&&void 0===n&&(clearTimeout(a),n=o(e.responseTimeout,"Socket timeout"))},r.onerror=function(){0===r.status&&(clearTimeout(a),clearTimeout(n),t({content:r.responseText||"Network request failed",status:r.status,isTimedOut:!1}))},r.onload=function(){clearTimeout(a),clearTimeout(n),t({content:r.responseText,status:r.status,isTimedOut:!1})},r.send(e.data)}))}},logger:(o=H,{debug:function(e,t){return D>=o&&console.debug(e,t),Promise.resolve()},info:function(e,t){return W>=o&&console.info(e,t),Promise.resolve()},error:function(e,t){return console.error(e,t),Promise.resolve()}}),responsesCache:s(),requestsCache:s({serializable:!1}),hostsCache:i({caches:[u({key:"".concat("4.14.2","-").concat(e)}),s()]}),userAgent:j("4.14.2").add({segment:"Browser",version:"lite"}),authMode:h.WithinQueryParameters};return N(r(r(r({},a),n),{},{methods:{search:J,searchForFacetValues:E,multipleQueries:J,multipleSearchForFacetValues:E,customRequest:C,initIndex:function(e){return function(t){return k(e)(t,{methods:{search:F,searchForFacetValues:R,findAnswers:I}})}}}}))}return Q.version="4.14.2",Q})); diff --git a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.esm.browser.js b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.esm.browser.js index 552242a..80d950e 100644 --- a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.esm.browser.js +++ b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.esm.browser.js @@ -207,7 +207,7 @@ function encode(format, ...args) { return format.replace(/%s/g, () => encodeURIComponent(args[i++])); } -const version = '4.13.0'; +const version = '4.14.2'; const AuthMode = { /** diff --git a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.umd.js b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.umd.js index a98fcff..3c4bc57 100644 --- a/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.umd.js +++ b/wp-search-with-algolia/js/algoliasearch/dist/algoliasearch.umd.js @@ -1,2 +1,2 @@ -/*! algoliasearch.umd.js | 4.13.0 | © Algolia, inc. | https://github.com/algolia/algoliasearch-client-javascript */ -!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t=t||self).algoliasearch=e()}(this,(function(){"use strict";function t(t,e,r){return e in t?Object.defineProperty(t,e,{value:r,enumerable:!0,configurable:!0,writable:!0}):t[e]=r,t}function e(t,e){var r=Object.keys(t);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(t);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable}))),r.push.apply(r,n)}return r}function r(r){for(var n=1;n=0||(a[r]=t[r]);return a}(t,e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(t);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(t,r)&&(a[r]=t[r])}return a}function a(t,e){return function(t){if(Array.isArray(t))return t}(t)||function(t,e){if(!(Symbol.iterator in Object(t)||"[object Arguments]"===Object.prototype.toString.call(t)))return;var r=[],n=!0,a=!1,o=void 0;try{for(var i,u=t[Symbol.iterator]();!(n=(i=u.next()).done)&&(r.push(i.value),!e||r.length!==e);n=!0);}catch(t){a=!0,o=t}finally{try{n||null==u.return||u.return()}finally{if(a)throw o}}return r}(t,e)||function(){throw new TypeError("Invalid attempt to destructure non-iterable instance")}()}function o(t){return function(t){if(Array.isArray(t)){for(var e=0,r=new Array(t.length);e2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return Promise.resolve().then((function(){var r=JSON.stringify(t),n=o()[r];return Promise.all([n||e(),void 0!==n])})).then((function(t){var e=a(t,2),n=e[0],o=e[1];return Promise.all([n,o||r.miss(n)])})).then((function(t){return a(t,1)[0]}))},set:function(t,e){return Promise.resolve().then((function(){var a=o();return a[JSON.stringify(t)]=e,n().setItem(r,JSON.stringify(a)),e}))},delete:function(t){return Promise.resolve().then((function(){var e=o();delete e[JSON.stringify(t)],n().setItem(r,JSON.stringify(e))}))},clear:function(){return Promise.resolve().then((function(){n().removeItem(r)}))}}}function u(t){var e=o(t.caches),r=e.shift();return void 0===r?{get:function(t,e){var r=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},n=e();return n.then((function(t){return Promise.all([t,r.miss(t)])})).then((function(t){return a(t,1)[0]}))},set:function(t,e){return Promise.resolve(e)},delete:function(t){return Promise.resolve()},clear:function(){return Promise.resolve()}}:{get:function(t,n){var a=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return r.get(t,n,a).catch((function(){return u({caches:e}).get(t,n,a)}))},set:function(t,n){return r.set(t,n).catch((function(){return u({caches:e}).set(t,n)}))},delete:function(t){return r.delete(t).catch((function(){return u({caches:e}).delete(t)}))},clear:function(){return r.clear().catch((function(){return u({caches:e}).clear()}))}}}function s(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{serializable:!0},e={};return{get:function(r,n){var a=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},o=JSON.stringify(r);if(o in e)return Promise.resolve(t.serializable?JSON.parse(e[o]):e[o]);var i=n(),u=a&&a.miss||function(){return Promise.resolve()};return i.then((function(t){return u(t)})).then((function(){return i}))},set:function(r,n){return e[JSON.stringify(r)]=t.serializable?JSON.stringify(n):n,Promise.resolve(n)},delete:function(t){return delete e[JSON.stringify(t)],Promise.resolve()},clear:function(){return e={},Promise.resolve()}}}function c(t,e,r){var n={"x-algolia-api-key":r,"x-algolia-application-id":e};return{headers:function(){return t===m.WithinHeaders?n:{}},queryParameters:function(){return t===m.WithinQueryParameters?n:{}}}}function f(t){var e=0;return t((function r(){return e++,new Promise((function(n){setTimeout((function(){n(t(r))}),Math.min(100*e,1e3))}))}))}function d(t){var e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:function(t,e){return Promise.resolve()};return Object.assign(t,{wait:function(r){return d(t.then((function(t){return Promise.all([e(t,r),t])})).then((function(t){return t[1]})))}})}function l(t){for(var e=t.length-1;e>0;e--){var r=Math.floor(Math.random()*(e+1)),n=t[e];t[e]=t[r],t[r]=n}return t}function p(t,e){return e?(Object.keys(e).forEach((function(r){t[r]=e[r](t)})),t):t}function h(t){for(var e=arguments.length,r=new Array(e>1?e-1:0),n=1;n0?n:void 0,timeout:r.timeout||e,headers:r.headers||{},queryParameters:r.queryParameters||{},cacheable:r.cacheable}}var g={Read:1,Write:2,Any:3},v=1,b=2,P=3;function w(t){var e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:v;return r(r({},t),{},{status:e,lastUpdate:Date.now()})}function O(t){return"string"==typeof t?{protocol:"https",url:t,accept:g.Any}:{protocol:t.protocol||"https",url:t.url,accept:t.accept||g.Any}}var I="DELETE",x="GET",j="POST",D="PUT";function q(t,e){return Promise.all(e.map((function(e){return t.get(e,(function(){return Promise.resolve(w(e))}))}))).then((function(t){var r=t.filter((function(t){return function(t){return t.status===v||Date.now()-t.lastUpdate>12e4}(t)})),n=t.filter((function(t){return function(t){return t.status===P&&Date.now()-t.lastUpdate<=12e4}(t)})),a=[].concat(o(r),o(n));return{getTimeout:function(t,e){return(0===n.length&&0===t?1:n.length+3+t)*e},statelessHosts:a.length>0?a.map((function(t){return O(t)})):e}}))}function S(t,e,n,a){var i=[],u=function(t,e){if(t.method===x||void 0===t.data&&void 0===e.data)return;var n=Array.isArray(t.data)?t.data:r(r({},t.data),e.data);return JSON.stringify(n)}(n,a),s=function(t,e){var n=r(r({},t.headers),e.headers),a={};return Object.keys(n).forEach((function(t){var e=n[t];a[t.toLowerCase()]=e})),a}(t,a),c=n.method,f=n.method!==x?{}:r(r({},n.data),a.data),d=r(r(r({"x-algolia-agent":t.userAgent.value},t.queryParameters),f),a.queryParameters),l=0,p=function e(r,o){var f=r.pop();if(void 0===f)throw{name:"RetryError",message:"Unreachable hosts - your application id may be incorrect. If the error persists, contact support@algolia.com.",transporterStackTrace:R(i)};var p={data:u,headers:s,method:c,url:N(f,n.path,d),connectTimeout:o(l,t.timeouts.connect),responseTimeout:o(l,a.timeout)},h=function(t){var e={request:p,response:t,host:f,triesLeft:r.length};return i.push(e),e},m={onSuccess:function(t){return function(t){try{return JSON.parse(t.content)}catch(e){throw function(t,e){return{name:"DeserializationError",message:t,response:e}}(e.message,t)}}(t)},onRetry:function(n){var a=h(n);return n.isTimedOut&&l++,Promise.all([t.logger.info("Retryable failure",A(a)),t.hostsCache.set(f,w(f,n.isTimedOut?P:b))]).then((function(){return e(r,o)}))},onFail:function(t){throw h(t),function(t,e){var r=t.content,n=t.status,a=r;try{a=JSON.parse(r).message}catch(t){}return function(t,e,r){return{name:"ApiError",message:t,status:e,transporterStackTrace:r}}(a,n,e)}(t,R(i))}};return t.requester.send(p).then((function(t){return function(t,e){return function(t){var e=t.status;return t.isTimedOut||function(t){var e=t.isTimedOut,r=t.status;return!e&&0==~~r}(t)||2!=~~(e/100)&&4!=~~(e/100)}(t)?e.onRetry(t):2==~~(t.status/100)?e.onSuccess(t):e.onFail(t)}(t,m)}))};return q(t.hostsCache,e).then((function(t){return p(o(t.statelessHosts).reverse(),t.getTimeout)}))}function k(t){var e=t.hostsCache,r=t.logger,n=t.requester,o=t.requestsCache,i=t.responsesCache,u=t.timeouts,s=t.userAgent,c=t.hosts,f=t.queryParameters,d={hostsCache:e,logger:r,requester:n,requestsCache:o,responsesCache:i,timeouts:u,userAgent:s,headers:t.headers,queryParameters:f,hosts:c.map((function(t){return O(t)})),read:function(t,e){var r=y(e,d.timeouts.read),n=function(){return S(d,d.hosts.filter((function(t){return 0!=(t.accept&g.Read)})),t,r)};if(!0!==(void 0!==r.cacheable?r.cacheable:t.cacheable))return n();var o={request:t,mappedRequestOptions:r,transporter:{queryParameters:d.queryParameters,headers:d.headers}};return d.responsesCache.get(o,(function(){return d.requestsCache.get(o,(function(){return d.requestsCache.set(o,n()).then((function(t){return Promise.all([d.requestsCache.delete(o),t])}),(function(t){return Promise.all([d.requestsCache.delete(o),Promise.reject(t)])})).then((function(t){var e=a(t,2);e[0];return e[1]}))}))}),{miss:function(t){return d.responsesCache.set(o,t)}})},write:function(t,e){return S(d,d.hosts.filter((function(t){return 0!=(t.accept&g.Write)})),t,y(e,d.timeouts.write))}};return d}function T(t){var e={value:"Algolia for JavaScript (".concat(t,")"),add:function(t){var r="; ".concat(t.segment).concat(void 0!==t.version?" (".concat(t.version,")"):"");return-1===e.value.indexOf(r)&&(e.value="".concat(e.value).concat(r)),e}};return e}function N(t,e,r){var n=E(r),a="".concat(t.protocol,"://").concat(t.url,"/").concat("/"===e.charAt(0)?e.substr(1):e);return n.length&&(a+="?".concat(n)),a}function E(t){return Object.keys(t).map((function(e){return h("%s=%s",e,(r=t[e],"[object Object]"===Object.prototype.toString.call(r)||"[object Array]"===Object.prototype.toString.call(r)?JSON.stringify(t[e]):t[e]));var r})).join("&")}function R(t){return t.map((function(t){return A(t)}))}function A(t){var e=t.request.headers["x-algolia-api-key"]?{"x-algolia-api-key":"*****"}:{};return r(r({},t),{},{request:r(r({},t.request),{},{headers:r(r({},t.request.headers),e)})})}var C=function(t){return function(e,r){return t.transporter.write({method:j,path:"2/abtests",data:e},r)}},U=function(t){return function(e,r){return t.transporter.write({method:I,path:h("2/abtests/%s",e)},r)}},z=function(t){return function(e,r){return t.transporter.read({method:x,path:h("2/abtests/%s",e)},r)}},J=function(t){return function(e){return t.transporter.read({method:x,path:"2/abtests"},e)}},F=function(t){return function(e,r){return t.transporter.write({method:j,path:h("2/abtests/%s/stop",e)},r)}},H=function(t){return function(e){return t.transporter.read({method:x,path:"1/strategies/personalization"},e)}},M=function(t){return function(e,r){return t.transporter.write({method:j,path:"1/strategies/personalization",data:e},r)}};function K(t){return function e(r){return t.request(r).then((function(n){if(void 0!==t.batch&&t.batch(n.hits),!t.shouldStop(n))return n.cursor?e({cursor:n.cursor}):e({page:(r.page||0)+1})}))}({})}var W=function(t){return function(e,a){var o=a||{},i=o.queryParameters,u=n(o,["queryParameters"]),s=r({acl:e},void 0!==i?{queryParameters:i}:{});return d(t.transporter.write({method:j,path:"1/keys",data:s},u),(function(e,r){return f((function(n){return tt(t)(e.key,r).catch((function(t){if(404!==t.status)throw t;return n()}))}))}))}},B=function(t){return function(e,r,n){var a=y(n);return a.queryParameters["X-Algolia-User-ID"]=e,t.transporter.write({method:j,path:"1/clusters/mapping",data:{cluster:r}},a)}},Q=function(t){return function(e,r,n){return t.transporter.write({method:j,path:"1/clusters/mapping/batch",data:{users:e,cluster:r}},n)}},G=function(t){return function(e,r){return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!0,requests:{action:"addEntry",body:[]}}},r),(function(e,r){return jt(t)(e.taskID,r)}))}},L=function(t){return function(e,r,n){return d(t.transporter.write({method:j,path:h("1/indexes/%s/operation",e),data:{operation:"copy",destination:r}},n),(function(r,n){return ut(t)(e,{methods:{waitTask:de}}).waitTask(r.taskID,n)}))}},V=function(t){return function(e,n,a){return L(t)(e,n,r(r({},a),{},{scope:[pe.Rules]}))}},_=function(t){return function(e,n,a){return L(t)(e,n,r(r({},a),{},{scope:[pe.Settings]}))}},X=function(t){return function(e,n,a){return L(t)(e,n,r(r({},a),{},{scope:[pe.Synonyms]}))}},Y=function(t){return function(e,r){return e.method===x?t.transporter.read(e,r):t.transporter.write(e,r)}},Z=function(t){return function(e,r){return d(t.transporter.write({method:I,path:h("1/keys/%s",e)},r),(function(r,n){return f((function(r){return tt(t)(e,n).then(r).catch((function(t){if(404!==t.status)throw t}))}))}))}},$=function(t){return function(e,r,n){var a=r.map((function(t){return{action:"deleteEntry",body:{objectID:t}}}));return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!1,requests:a}},n),(function(e,r){return jt(t)(e.taskID,r)}))}},tt=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/keys/%s",e)},r)}},et=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/task/%s",e.toString())},r)}},rt=function(t){return function(e){return t.transporter.read({method:x,path:"/1/dictionaries/*/settings"},e)}},nt=function(t){return function(e){return t.transporter.read({method:x,path:"1/logs"},e)}},at=function(t){return function(e){return t.transporter.read({method:x,path:"1/clusters/mapping/top"},e)}},ot=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/clusters/mapping/%s",e)},r)}},it=function(t){return function(e){var r=e||{},a=r.retrieveMappings,o=n(r,["retrieveMappings"]);return!0===a&&(o.getClusters=!0),t.transporter.read({method:x,path:"1/clusters/mapping/pending"},o)}},ut=function(t){return function(e){var r=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},n={transporter:t.transporter,appId:t.appId,indexName:e};return p(n,r.methods)}},st=function(t){return function(e){return t.transporter.read({method:x,path:"1/keys"},e)}},ct=function(t){return function(e){return t.transporter.read({method:x,path:"1/clusters"},e)}},ft=function(t){return function(e){return t.transporter.read({method:x,path:"1/indexes"},e)}},dt=function(t){return function(e){return t.transporter.read({method:x,path:"1/clusters/mapping"},e)}},lt=function(t){return function(e,r,n){return d(t.transporter.write({method:j,path:h("1/indexes/%s/operation",e),data:{operation:"move",destination:r}},n),(function(r,n){return ut(t)(e,{methods:{waitTask:de}}).waitTask(r.taskID,n)}))}},pt=function(t){return function(e,r){return d(t.transporter.write({method:j,path:"1/indexes/*/batch",data:{requests:e}},r),(function(e,r){return Promise.all(Object.keys(e.taskID).map((function(n){return ut(t)(n,{methods:{waitTask:de}}).waitTask(e.taskID[n],r)})))}))}},ht=function(t){return function(e,r){return t.transporter.read({method:j,path:"1/indexes/*/objects",data:{requests:e}},r)}},mt=function(t){return function(e,n){var a=e.map((function(t){return r(r({},t),{},{params:E(t.params||{})})}));return t.transporter.read({method:j,path:"1/indexes/*/queries",data:{requests:a},cacheable:!0},n)}},yt=function(t){return function(e,a){return Promise.all(e.map((function(e){var o=e.params,i=o.facetName,u=o.facetQuery,s=n(o,["facetName","facetQuery"]);return ut(t)(e.indexName,{methods:{searchForFacetValues:ue}}).searchForFacetValues(i,u,r(r({},a),s))})))}},gt=function(t){return function(e,r){var n=y(r);return n.queryParameters["X-Algolia-User-ID"]=e,t.transporter.write({method:I,path:"1/clusters/mapping"},n)}},vt=function(t){return function(e,r,n){var a=r.map((function(t){return{action:"addEntry",body:t}}));return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!0,requests:a}},n),(function(e,r){return jt(t)(e.taskID,r)}))}},bt=function(t){return function(e,r){return d(t.transporter.write({method:j,path:h("1/keys/%s/restore",e)},r),(function(r,n){return f((function(r){return tt(t)(e,n).catch((function(t){if(404!==t.status)throw t;return r()}))}))}))}},Pt=function(t){return function(e,r,n){var a=r.map((function(t){return{action:"addEntry",body:t}}));return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!1,requests:a}},n),(function(e,r){return jt(t)(e.taskID,r)}))}},wt=function(t){return function(e,r,n){return t.transporter.read({method:j,path:h("/1/dictionaries/%s/search",e),data:{query:r},cacheable:!0},n)}},Ot=function(t){return function(e,r){return t.transporter.read({method:j,path:"1/clusters/mapping/search",data:{query:e}},r)}},It=function(t){return function(e,r){return d(t.transporter.write({method:D,path:"/1/dictionaries/*/settings",data:e},r),(function(e,r){return jt(t)(e.taskID,r)}))}},xt=function(t){return function(e,r){var a=Object.assign({},r),o=r||{},i=o.queryParameters,u=n(o,["queryParameters"]),s=i?{queryParameters:i}:{},c=["acl","indexes","referers","restrictSources","queryParameters","description","maxQueriesPerIPPerHour","maxHitsPerQuery"];return d(t.transporter.write({method:D,path:h("1/keys/%s",e),data:s},u),(function(r,n){return f((function(r){return tt(t)(e,n).then((function(t){return function(t){return Object.keys(a).filter((function(t){return-1!==c.indexOf(t)})).every((function(e){return t[e]===a[e]}))}(t)?Promise.resolve():r()}))}))}))}},jt=function(t){return function(e,r){return f((function(n){return et(t)(e,r).then((function(t){return"published"!==t.status?n():void 0}))}))}},Dt=function(t){return function(e,r){return d(t.transporter.write({method:j,path:h("1/indexes/%s/batch",t.indexName),data:{requests:e}},r),(function(e,r){return de(t)(e.taskID,r)}))}},qt=function(t){return function(e){return K(r(r({shouldStop:function(t){return void 0===t.cursor}},e),{},{request:function(r){return t.transporter.read({method:j,path:h("1/indexes/%s/browse",t.indexName),data:r},e)}}))}},St=function(t){return function(e){var n=r({hitsPerPage:1e3},e);return K(r(r({shouldStop:function(t){return t.hits.length0&&void 0!==arguments[0]?arguments[0]:0,c=[];for(a=o;a=t.nbPages)throw{name:"ObjectNotFoundError",message:"Object not found."};return n()}))}()}},Wt=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/indexes/%s/%s",t.indexName,e)},r)}},Bt=function(){return function(t,e){for(var r=0,n=Object.entries(t.hits);rr.OPENED&&void 0===n&&(clearTimeout(o),n=a(t.responseTimeout,"Socket timeout"))},r.onerror=function(){0===r.status&&(clearTimeout(o),clearTimeout(n),e({content:r.responseText||"Network request failed",status:r.status,isTimedOut:!1}))},r.onload=function(){clearTimeout(o),clearTimeout(n),e({content:r.responseText,status:r.status,isTimedOut:!1})},r.send(t.data)}))}},logger:(a=ye,{debug:function(t,e){return he>=a&&console.debug(t,e),Promise.resolve()},info:function(t,e){return me>=a&&console.info(t,e),Promise.resolve()},error:function(t,e){return console.error(t,e),Promise.resolve()}}),responsesCache:s(),requestsCache:s({serializable:!1}),hostsCache:u({caches:[i({key:"".concat("4.13.0","-").concat(t)}),s()]}),userAgent:T("4.13.0").add({segment:"Browser"})},f=r(r({},o),n),d=function(){return function(t){return function(t){var e=t.region||"us",n=c(m.WithinHeaders,t.appId,t.apiKey),a=k(r(r({hosts:[{url:"personalization.".concat(e,".algolia.com")}]},t),{},{headers:r(r(r({},n.headers()),{"content-type":"application/json"}),t.headers),queryParameters:r(r({},n.queryParameters()),t.queryParameters)}));return p({appId:t.appId,transporter:a},t.methods)}(r(r(r({},o),t),{},{methods:{getPersonalizationStrategy:H,setPersonalizationStrategy:M}}))}};return function(t){var e=t.appId,n=c(void 0!==t.authMode?t.authMode:m.WithinHeaders,e,t.apiKey),a=k(r(r({hosts:[{url:"".concat(e,"-dsn.algolia.net"),accept:g.Read},{url:"".concat(e,".algolia.net"),accept:g.Write}].concat(l([{url:"".concat(e,"-1.algolianet.com")},{url:"".concat(e,"-2.algolianet.com")},{url:"".concat(e,"-3.algolianet.com")}]))},t),{},{headers:r(r(r({},n.headers()),{"content-type":"application/x-www-form-urlencoded"}),t.headers),queryParameters:r(r({},n.queryParameters()),t.queryParameters)}));return p({transporter:a,appId:e,addAlgoliaAgent:function(t,e){a.userAgent.add({segment:t,version:e})},clearCache:function(){return Promise.all([a.requestsCache.clear(),a.responsesCache.clear()]).then((function(){}))}},t.methods)}(r(r({},f),{},{methods:{search:mt,searchForFacetValues:yt,multipleBatch:pt,multipleGetObjects:ht,multipleQueries:mt,copyIndex:L,copySettings:_,copySynonyms:X,copyRules:V,moveIndex:lt,listIndices:ft,getLogs:nt,listClusters:ct,multipleSearchForFacetValues:yt,getApiKey:tt,addApiKey:W,listApiKeys:st,updateApiKey:xt,deleteApiKey:Z,restoreApiKey:bt,assignUserID:B,assignUserIDs:Q,getUserID:ot,searchUserIDs:Ot,listUserIDs:dt,getTopUserIDs:at,removeUserID:gt,hasPendingMappings:it,clearDictionaryEntries:G,deleteDictionaryEntries:$,getDictionarySettings:rt,getAppTask:et,replaceDictionaryEntries:vt,saveDictionaryEntries:Pt,searchDictionaryEntries:wt,setDictionarySettings:It,waitAppTask:jt,customRequest:Y,initIndex:function(t){return function(e){return ut(t)(e,{methods:{batch:Dt,delete:Ct,findAnswers:Mt,getObject:Wt,getObjects:Qt,saveObject:te,saveObjects:ee,search:ie,searchForFacetValues:ue,waitTask:de,setSettings:fe,getSettings:Lt,partialUpdateObject:_t,partialUpdateObjects:Xt,deleteObject:Ut,deleteObjects:zt,deleteBy:At,clearObjects:Nt,browseObjects:qt,getObjectPosition:Bt,findObject:Kt,exists:Ht,saveSynonym:ae,saveSynonyms:oe,getSynonym:Vt,searchSynonyms:ce,browseSynonyms:kt,deleteSynonym:Ft,clearSynonyms:Rt,replaceAllObjects:Yt,replaceAllSynonyms:$t,searchRules:se,getRule:Gt,deleteRule:Jt,saveRule:re,saveRules:ne,replaceAllRules:Zt,browseRules:St,clearRules:Et}})}},initAnalytics:function(){return function(t){return function(t){var e=t.region||"us",n=c(m.WithinHeaders,t.appId,t.apiKey),a=k(r(r({hosts:[{url:"analytics.".concat(e,".algolia.com")}]},t),{},{headers:r(r(r({},n.headers()),{"content-type":"application/json"}),t.headers),queryParameters:r(r({},n.queryParameters()),t.queryParameters)}));return p({appId:t.appId,transporter:a},t.methods)}(r(r(r({},o),t),{},{methods:{addABTest:C,getABTest:z,getABTests:J,stopABTest:F,deleteABTest:U}}))}},initPersonalization:d,initRecommendation:function(){return function(t){return f.logger.info("The `initRecommendation` method is deprecated. Use `initPersonalization` instead."),d()(t)}}}}))}return ge.version="4.13.0",ge})); +/*! algoliasearch.umd.js | 4.14.2 | © Algolia, inc. | https://github.com/algolia/algoliasearch-client-javascript */ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t=t||self).algoliasearch=e()}(this,(function(){"use strict";function t(t,e,r){return e in t?Object.defineProperty(t,e,{value:r,enumerable:!0,configurable:!0,writable:!0}):t[e]=r,t}function e(t,e){var r=Object.keys(t);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(t);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable}))),r.push.apply(r,n)}return r}function r(r){for(var n=1;n=0||(a[r]=t[r]);return a}(t,e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(t);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(t,r)&&(a[r]=t[r])}return a}function a(t,e){return function(t){if(Array.isArray(t))return t}(t)||function(t,e){if(!(Symbol.iterator in Object(t)||"[object Arguments]"===Object.prototype.toString.call(t)))return;var r=[],n=!0,a=!1,o=void 0;try{for(var i,u=t[Symbol.iterator]();!(n=(i=u.next()).done)&&(r.push(i.value),!e||r.length!==e);n=!0);}catch(t){a=!0,o=t}finally{try{n||null==u.return||u.return()}finally{if(a)throw o}}return r}(t,e)||function(){throw new TypeError("Invalid attempt to destructure non-iterable instance")}()}function o(t){return function(t){if(Array.isArray(t)){for(var e=0,r=new Array(t.length);e2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return Promise.resolve().then((function(){var r=JSON.stringify(t),n=o()[r];return Promise.all([n||e(),void 0!==n])})).then((function(t){var e=a(t,2),n=e[0],o=e[1];return Promise.all([n,o||r.miss(n)])})).then((function(t){return a(t,1)[0]}))},set:function(t,e){return Promise.resolve().then((function(){var a=o();return a[JSON.stringify(t)]=e,n().setItem(r,JSON.stringify(a)),e}))},delete:function(t){return Promise.resolve().then((function(){var e=o();delete e[JSON.stringify(t)],n().setItem(r,JSON.stringify(e))}))},clear:function(){return Promise.resolve().then((function(){n().removeItem(r)}))}}}function u(t){var e=o(t.caches),r=e.shift();return void 0===r?{get:function(t,e){var r=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},n=e();return n.then((function(t){return Promise.all([t,r.miss(t)])})).then((function(t){return a(t,1)[0]}))},set:function(t,e){return Promise.resolve(e)},delete:function(t){return Promise.resolve()},clear:function(){return Promise.resolve()}}:{get:function(t,n){var a=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}};return r.get(t,n,a).catch((function(){return u({caches:e}).get(t,n,a)}))},set:function(t,n){return r.set(t,n).catch((function(){return u({caches:e}).set(t,n)}))},delete:function(t){return r.delete(t).catch((function(){return u({caches:e}).delete(t)}))},clear:function(){return r.clear().catch((function(){return u({caches:e}).clear()}))}}}function s(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{serializable:!0},e={};return{get:function(r,n){var a=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{miss:function(){return Promise.resolve()}},o=JSON.stringify(r);if(o in e)return Promise.resolve(t.serializable?JSON.parse(e[o]):e[o]);var i=n(),u=a&&a.miss||function(){return Promise.resolve()};return i.then((function(t){return u(t)})).then((function(){return i}))},set:function(r,n){return e[JSON.stringify(r)]=t.serializable?JSON.stringify(n):n,Promise.resolve(n)},delete:function(t){return delete e[JSON.stringify(t)],Promise.resolve()},clear:function(){return e={},Promise.resolve()}}}function c(t,e,r){var n={"x-algolia-api-key":r,"x-algolia-application-id":e};return{headers:function(){return t===m.WithinHeaders?n:{}},queryParameters:function(){return t===m.WithinQueryParameters?n:{}}}}function f(t){var e=0;return t((function r(){return e++,new Promise((function(n){setTimeout((function(){n(t(r))}),Math.min(100*e,1e3))}))}))}function d(t){var e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:function(t,e){return Promise.resolve()};return Object.assign(t,{wait:function(r){return d(t.then((function(t){return Promise.all([e(t,r),t])})).then((function(t){return t[1]})))}})}function l(t){for(var e=t.length-1;e>0;e--){var r=Math.floor(Math.random()*(e+1)),n=t[e];t[e]=t[r],t[r]=n}return t}function p(t,e){return e?(Object.keys(e).forEach((function(r){t[r]=e[r](t)})),t):t}function h(t){for(var e=arguments.length,r=new Array(e>1?e-1:0),n=1;n0?n:void 0,timeout:r.timeout||e,headers:r.headers||{},queryParameters:r.queryParameters||{},cacheable:r.cacheable}}var g={Read:1,Write:2,Any:3},v=1,b=2,P=3;function w(t){var e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:v;return r(r({},t),{},{status:e,lastUpdate:Date.now()})}function O(t){return"string"==typeof t?{protocol:"https",url:t,accept:g.Any}:{protocol:t.protocol||"https",url:t.url,accept:t.accept||g.Any}}var I="DELETE",x="GET",j="POST",D="PUT";function q(t,e){return Promise.all(e.map((function(e){return t.get(e,(function(){return Promise.resolve(w(e))}))}))).then((function(t){var r=t.filter((function(t){return function(t){return t.status===v||Date.now()-t.lastUpdate>12e4}(t)})),n=t.filter((function(t){return function(t){return t.status===P&&Date.now()-t.lastUpdate<=12e4}(t)})),a=[].concat(o(r),o(n));return{getTimeout:function(t,e){return(0===n.length&&0===t?1:n.length+3+t)*e},statelessHosts:a.length>0?a.map((function(t){return O(t)})):e}}))}function S(t,e,n,a){var i=[],u=function(t,e){if(t.method===x||void 0===t.data&&void 0===e.data)return;var n=Array.isArray(t.data)?t.data:r(r({},t.data),e.data);return JSON.stringify(n)}(n,a),s=function(t,e){var n=r(r({},t.headers),e.headers),a={};return Object.keys(n).forEach((function(t){var e=n[t];a[t.toLowerCase()]=e})),a}(t,a),c=n.method,f=n.method!==x?{}:r(r({},n.data),a.data),d=r(r(r({"x-algolia-agent":t.userAgent.value},t.queryParameters),f),a.queryParameters),l=0,p=function e(r,o){var f=r.pop();if(void 0===f)throw{name:"RetryError",message:"Unreachable hosts - your application id may be incorrect. If the error persists, contact support@algolia.com.",transporterStackTrace:R(i)};var p={data:u,headers:s,method:c,url:N(f,n.path,d),connectTimeout:o(l,t.timeouts.connect),responseTimeout:o(l,a.timeout)},h=function(t){var e={request:p,response:t,host:f,triesLeft:r.length};return i.push(e),e},m={onSuccess:function(t){return function(t){try{return JSON.parse(t.content)}catch(e){throw function(t,e){return{name:"DeserializationError",message:t,response:e}}(e.message,t)}}(t)},onRetry:function(n){var a=h(n);return n.isTimedOut&&l++,Promise.all([t.logger.info("Retryable failure",A(a)),t.hostsCache.set(f,w(f,n.isTimedOut?P:b))]).then((function(){return e(r,o)}))},onFail:function(t){throw h(t),function(t,e){var r=t.content,n=t.status,a=r;try{a=JSON.parse(r).message}catch(t){}return function(t,e,r){return{name:"ApiError",message:t,status:e,transporterStackTrace:r}}(a,n,e)}(t,R(i))}};return t.requester.send(p).then((function(t){return function(t,e){return function(t){var e=t.status;return t.isTimedOut||function(t){var e=t.isTimedOut,r=t.status;return!e&&0==~~r}(t)||2!=~~(e/100)&&4!=~~(e/100)}(t)?e.onRetry(t):2==~~(t.status/100)?e.onSuccess(t):e.onFail(t)}(t,m)}))};return q(t.hostsCache,e).then((function(t){return p(o(t.statelessHosts).reverse(),t.getTimeout)}))}function k(t){var e=t.hostsCache,r=t.logger,n=t.requester,o=t.requestsCache,i=t.responsesCache,u=t.timeouts,s=t.userAgent,c=t.hosts,f=t.queryParameters,d={hostsCache:e,logger:r,requester:n,requestsCache:o,responsesCache:i,timeouts:u,userAgent:s,headers:t.headers,queryParameters:f,hosts:c.map((function(t){return O(t)})),read:function(t,e){var r=y(e,d.timeouts.read),n=function(){return S(d,d.hosts.filter((function(t){return 0!=(t.accept&g.Read)})),t,r)};if(!0!==(void 0!==r.cacheable?r.cacheable:t.cacheable))return n();var o={request:t,mappedRequestOptions:r,transporter:{queryParameters:d.queryParameters,headers:d.headers}};return d.responsesCache.get(o,(function(){return d.requestsCache.get(o,(function(){return d.requestsCache.set(o,n()).then((function(t){return Promise.all([d.requestsCache.delete(o),t])}),(function(t){return Promise.all([d.requestsCache.delete(o),Promise.reject(t)])})).then((function(t){var e=a(t,2);e[0];return e[1]}))}))}),{miss:function(t){return d.responsesCache.set(o,t)}})},write:function(t,e){return S(d,d.hosts.filter((function(t){return 0!=(t.accept&g.Write)})),t,y(e,d.timeouts.write))}};return d}function T(t){var e={value:"Algolia for JavaScript (".concat(t,")"),add:function(t){var r="; ".concat(t.segment).concat(void 0!==t.version?" (".concat(t.version,")"):"");return-1===e.value.indexOf(r)&&(e.value="".concat(e.value).concat(r)),e}};return e}function N(t,e,r){var n=E(r),a="".concat(t.protocol,"://").concat(t.url,"/").concat("/"===e.charAt(0)?e.substr(1):e);return n.length&&(a+="?".concat(n)),a}function E(t){return Object.keys(t).map((function(e){return h("%s=%s",e,(r=t[e],"[object Object]"===Object.prototype.toString.call(r)||"[object Array]"===Object.prototype.toString.call(r)?JSON.stringify(t[e]):t[e]));var r})).join("&")}function R(t){return t.map((function(t){return A(t)}))}function A(t){var e=t.request.headers["x-algolia-api-key"]?{"x-algolia-api-key":"*****"}:{};return r(r({},t),{},{request:r(r({},t.request),{},{headers:r(r({},t.request.headers),e)})})}var C=function(t){return function(e,r){return t.transporter.write({method:j,path:"2/abtests",data:e},r)}},U=function(t){return function(e,r){return t.transporter.write({method:I,path:h("2/abtests/%s",e)},r)}},z=function(t){return function(e,r){return t.transporter.read({method:x,path:h("2/abtests/%s",e)},r)}},J=function(t){return function(e){return t.transporter.read({method:x,path:"2/abtests"},e)}},F=function(t){return function(e,r){return t.transporter.write({method:j,path:h("2/abtests/%s/stop",e)},r)}},H=function(t){return function(e){return t.transporter.read({method:x,path:"1/strategies/personalization"},e)}},M=function(t){return function(e,r){return t.transporter.write({method:j,path:"1/strategies/personalization",data:e},r)}};function K(t){return function e(r){return t.request(r).then((function(n){if(void 0!==t.batch&&t.batch(n.hits),!t.shouldStop(n))return n.cursor?e({cursor:n.cursor}):e({page:(r.page||0)+1})}))}({})}var W=function(t){return function(e,a){var o=a||{},i=o.queryParameters,u=n(o,["queryParameters"]),s=r({acl:e},void 0!==i?{queryParameters:i}:{});return d(t.transporter.write({method:j,path:"1/keys",data:s},u),(function(e,r){return f((function(n){return tt(t)(e.key,r).catch((function(t){if(404!==t.status)throw t;return n()}))}))}))}},B=function(t){return function(e,r,n){var a=y(n);return a.queryParameters["X-Algolia-User-ID"]=e,t.transporter.write({method:j,path:"1/clusters/mapping",data:{cluster:r}},a)}},Q=function(t){return function(e,r,n){return t.transporter.write({method:j,path:"1/clusters/mapping/batch",data:{users:e,cluster:r}},n)}},G=function(t){return function(e,r){return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!0,requests:{action:"addEntry",body:[]}}},r),(function(e,r){return jt(t)(e.taskID,r)}))}},L=function(t){return function(e,r,n){return d(t.transporter.write({method:j,path:h("1/indexes/%s/operation",e),data:{operation:"copy",destination:r}},n),(function(r,n){return ut(t)(e,{methods:{waitTask:de}}).waitTask(r.taskID,n)}))}},V=function(t){return function(e,n,a){return L(t)(e,n,r(r({},a),{},{scope:[pe.Rules]}))}},_=function(t){return function(e,n,a){return L(t)(e,n,r(r({},a),{},{scope:[pe.Settings]}))}},X=function(t){return function(e,n,a){return L(t)(e,n,r(r({},a),{},{scope:[pe.Synonyms]}))}},Y=function(t){return function(e,r){return e.method===x?t.transporter.read(e,r):t.transporter.write(e,r)}},Z=function(t){return function(e,r){return d(t.transporter.write({method:I,path:h("1/keys/%s",e)},r),(function(r,n){return f((function(r){return tt(t)(e,n).then(r).catch((function(t){if(404!==t.status)throw t}))}))}))}},$=function(t){return function(e,r,n){var a=r.map((function(t){return{action:"deleteEntry",body:{objectID:t}}}));return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!1,requests:a}},n),(function(e,r){return jt(t)(e.taskID,r)}))}},tt=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/keys/%s",e)},r)}},et=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/task/%s",e.toString())},r)}},rt=function(t){return function(e){return t.transporter.read({method:x,path:"/1/dictionaries/*/settings"},e)}},nt=function(t){return function(e){return t.transporter.read({method:x,path:"1/logs"},e)}},at=function(t){return function(e){return t.transporter.read({method:x,path:"1/clusters/mapping/top"},e)}},ot=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/clusters/mapping/%s",e)},r)}},it=function(t){return function(e){var r=e||{},a=r.retrieveMappings,o=n(r,["retrieveMappings"]);return!0===a&&(o.getClusters=!0),t.transporter.read({method:x,path:"1/clusters/mapping/pending"},o)}},ut=function(t){return function(e){var r=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},n={transporter:t.transporter,appId:t.appId,indexName:e};return p(n,r.methods)}},st=function(t){return function(e){return t.transporter.read({method:x,path:"1/keys"},e)}},ct=function(t){return function(e){return t.transporter.read({method:x,path:"1/clusters"},e)}},ft=function(t){return function(e){return t.transporter.read({method:x,path:"1/indexes"},e)}},dt=function(t){return function(e){return t.transporter.read({method:x,path:"1/clusters/mapping"},e)}},lt=function(t){return function(e,r,n){return d(t.transporter.write({method:j,path:h("1/indexes/%s/operation",e),data:{operation:"move",destination:r}},n),(function(r,n){return ut(t)(e,{methods:{waitTask:de}}).waitTask(r.taskID,n)}))}},pt=function(t){return function(e,r){return d(t.transporter.write({method:j,path:"1/indexes/*/batch",data:{requests:e}},r),(function(e,r){return Promise.all(Object.keys(e.taskID).map((function(n){return ut(t)(n,{methods:{waitTask:de}}).waitTask(e.taskID[n],r)})))}))}},ht=function(t){return function(e,r){return t.transporter.read({method:j,path:"1/indexes/*/objects",data:{requests:e}},r)}},mt=function(t){return function(e,n){var a=e.map((function(t){return r(r({},t),{},{params:E(t.params||{})})}));return t.transporter.read({method:j,path:"1/indexes/*/queries",data:{requests:a},cacheable:!0},n)}},yt=function(t){return function(e,a){return Promise.all(e.map((function(e){var o=e.params,i=o.facetName,u=o.facetQuery,s=n(o,["facetName","facetQuery"]);return ut(t)(e.indexName,{methods:{searchForFacetValues:ue}}).searchForFacetValues(i,u,r(r({},a),s))})))}},gt=function(t){return function(e,r){var n=y(r);return n.queryParameters["X-Algolia-User-ID"]=e,t.transporter.write({method:I,path:"1/clusters/mapping"},n)}},vt=function(t){return function(e,r,n){var a=r.map((function(t){return{action:"addEntry",body:t}}));return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!0,requests:a}},n),(function(e,r){return jt(t)(e.taskID,r)}))}},bt=function(t){return function(e,r){return d(t.transporter.write({method:j,path:h("1/keys/%s/restore",e)},r),(function(r,n){return f((function(r){return tt(t)(e,n).catch((function(t){if(404!==t.status)throw t;return r()}))}))}))}},Pt=function(t){return function(e,r,n){var a=r.map((function(t){return{action:"addEntry",body:t}}));return d(t.transporter.write({method:j,path:h("/1/dictionaries/%s/batch",e),data:{clearExistingDictionaryEntries:!1,requests:a}},n),(function(e,r){return jt(t)(e.taskID,r)}))}},wt=function(t){return function(e,r,n){return t.transporter.read({method:j,path:h("/1/dictionaries/%s/search",e),data:{query:r},cacheable:!0},n)}},Ot=function(t){return function(e,r){return t.transporter.read({method:j,path:"1/clusters/mapping/search",data:{query:e}},r)}},It=function(t){return function(e,r){return d(t.transporter.write({method:D,path:"/1/dictionaries/*/settings",data:e},r),(function(e,r){return jt(t)(e.taskID,r)}))}},xt=function(t){return function(e,r){var a=Object.assign({},r),o=r||{},i=o.queryParameters,u=n(o,["queryParameters"]),s=i?{queryParameters:i}:{},c=["acl","indexes","referers","restrictSources","queryParameters","description","maxQueriesPerIPPerHour","maxHitsPerQuery"];return d(t.transporter.write({method:D,path:h("1/keys/%s",e),data:s},u),(function(r,n){return f((function(r){return tt(t)(e,n).then((function(t){return function(t){return Object.keys(a).filter((function(t){return-1!==c.indexOf(t)})).every((function(e){return t[e]===a[e]}))}(t)?Promise.resolve():r()}))}))}))}},jt=function(t){return function(e,r){return f((function(n){return et(t)(e,r).then((function(t){return"published"!==t.status?n():void 0}))}))}},Dt=function(t){return function(e,r){return d(t.transporter.write({method:j,path:h("1/indexes/%s/batch",t.indexName),data:{requests:e}},r),(function(e,r){return de(t)(e.taskID,r)}))}},qt=function(t){return function(e){return K(r(r({shouldStop:function(t){return void 0===t.cursor}},e),{},{request:function(r){return t.transporter.read({method:j,path:h("1/indexes/%s/browse",t.indexName),data:r},e)}}))}},St=function(t){return function(e){var n=r({hitsPerPage:1e3},e);return K(r(r({shouldStop:function(t){return t.hits.length0&&void 0!==arguments[0]?arguments[0]:0,c=[];for(a=o;a=t.nbPages)throw{name:"ObjectNotFoundError",message:"Object not found."};return n()}))}()}},Wt=function(t){return function(e,r){return t.transporter.read({method:x,path:h("1/indexes/%s/%s",t.indexName,e)},r)}},Bt=function(){return function(t,e){for(var r=0,n=Object.entries(t.hits);rr.OPENED&&void 0===n&&(clearTimeout(o),n=a(t.responseTimeout,"Socket timeout"))},r.onerror=function(){0===r.status&&(clearTimeout(o),clearTimeout(n),e({content:r.responseText||"Network request failed",status:r.status,isTimedOut:!1}))},r.onload=function(){clearTimeout(o),clearTimeout(n),e({content:r.responseText,status:r.status,isTimedOut:!1})},r.send(t.data)}))}},logger:(a=ye,{debug:function(t,e){return he>=a&&console.debug(t,e),Promise.resolve()},info:function(t,e){return me>=a&&console.info(t,e),Promise.resolve()},error:function(t,e){return console.error(t,e),Promise.resolve()}}),responsesCache:s(),requestsCache:s({serializable:!1}),hostsCache:u({caches:[i({key:"".concat("4.14.2","-").concat(t)}),s()]}),userAgent:T("4.14.2").add({segment:"Browser"})},f=r(r({},o),n),d=function(){return function(t){return function(t){var e=t.region||"us",n=c(m.WithinHeaders,t.appId,t.apiKey),a=k(r(r({hosts:[{url:"personalization.".concat(e,".algolia.com")}]},t),{},{headers:r(r(r({},n.headers()),{"content-type":"application/json"}),t.headers),queryParameters:r(r({},n.queryParameters()),t.queryParameters)}));return p({appId:t.appId,transporter:a},t.methods)}(r(r(r({},o),t),{},{methods:{getPersonalizationStrategy:H,setPersonalizationStrategy:M}}))}};return function(t){var e=t.appId,n=c(void 0!==t.authMode?t.authMode:m.WithinHeaders,e,t.apiKey),a=k(r(r({hosts:[{url:"".concat(e,"-dsn.algolia.net"),accept:g.Read},{url:"".concat(e,".algolia.net"),accept:g.Write}].concat(l([{url:"".concat(e,"-1.algolianet.com")},{url:"".concat(e,"-2.algolianet.com")},{url:"".concat(e,"-3.algolianet.com")}]))},t),{},{headers:r(r(r({},n.headers()),{"content-type":"application/x-www-form-urlencoded"}),t.headers),queryParameters:r(r({},n.queryParameters()),t.queryParameters)}));return p({transporter:a,appId:e,addAlgoliaAgent:function(t,e){a.userAgent.add({segment:t,version:e})},clearCache:function(){return Promise.all([a.requestsCache.clear(),a.responsesCache.clear()]).then((function(){}))}},t.methods)}(r(r({},f),{},{methods:{search:mt,searchForFacetValues:yt,multipleBatch:pt,multipleGetObjects:ht,multipleQueries:mt,copyIndex:L,copySettings:_,copySynonyms:X,copyRules:V,moveIndex:lt,listIndices:ft,getLogs:nt,listClusters:ct,multipleSearchForFacetValues:yt,getApiKey:tt,addApiKey:W,listApiKeys:st,updateApiKey:xt,deleteApiKey:Z,restoreApiKey:bt,assignUserID:B,assignUserIDs:Q,getUserID:ot,searchUserIDs:Ot,listUserIDs:dt,getTopUserIDs:at,removeUserID:gt,hasPendingMappings:it,clearDictionaryEntries:G,deleteDictionaryEntries:$,getDictionarySettings:rt,getAppTask:et,replaceDictionaryEntries:vt,saveDictionaryEntries:Pt,searchDictionaryEntries:wt,setDictionarySettings:It,waitAppTask:jt,customRequest:Y,initIndex:function(t){return function(e){return ut(t)(e,{methods:{batch:Dt,delete:Ct,findAnswers:Mt,getObject:Wt,getObjects:Qt,saveObject:te,saveObjects:ee,search:ie,searchForFacetValues:ue,waitTask:de,setSettings:fe,getSettings:Lt,partialUpdateObject:_t,partialUpdateObjects:Xt,deleteObject:Ut,deleteObjects:zt,deleteBy:At,clearObjects:Nt,browseObjects:qt,getObjectPosition:Bt,findObject:Kt,exists:Ht,saveSynonym:ae,saveSynonyms:oe,getSynonym:Vt,searchSynonyms:ce,browseSynonyms:kt,deleteSynonym:Ft,clearSynonyms:Rt,replaceAllObjects:Yt,replaceAllSynonyms:$t,searchRules:se,getRule:Gt,deleteRule:Jt,saveRule:re,saveRules:ne,replaceAllRules:Zt,browseRules:St,clearRules:Et}})}},initAnalytics:function(){return function(t){return function(t){var e=t.region||"us",n=c(m.WithinHeaders,t.appId,t.apiKey),a=k(r(r({hosts:[{url:"analytics.".concat(e,".algolia.com")}]},t),{},{headers:r(r(r({},n.headers()),{"content-type":"application/json"}),t.headers),queryParameters:r(r({},n.queryParameters()),t.queryParameters)}));return p({appId:t.appId,transporter:a},t.methods)}(r(r(r({},o),t),{},{methods:{addABTest:C,getABTest:z,getABTests:J,stopABTest:F,deleteABTest:U}}))}},initPersonalization:d,initRecommendation:function(){return function(t){return f.logger.info("The `initRecommendation` method is deprecated. Use `initPersonalization` instead."),d()(t)}}}}))}return ge.version="4.14.2",ge})); diff --git a/wp-search-with-algolia/js/instantsearch.js/CHANGELOG.md b/wp-search-with-algolia/js/instantsearch.js/CHANGELOG.md index 80d3610..d8f485b 100644 --- a/wp-search-with-algolia/js/instantsearch.js/CHANGELOG.md +++ b/wp-search-with-algolia/js/instantsearch.js/CHANGELOG.md @@ -1,3 +1,199 @@ +# [4.49.0](https://github.com/algolia/instantsearch.js/compare/v4.48.1...v4.49.0) (2022-10-25) + + +### Features + +* **poweredBy:** update component logo ([#5145](https://github.com/algolia/instantsearch.js/issues/5145)) ([7df7816](https://github.com/algolia/instantsearch.js/commit/7df7816eac1bb3d2eafee5da7b6f4f59611468b2)) + + + +## [4.48.1](https://github.com/algolia/instantsearch.js/compare/v4.48.0...v4.48.1) (2022-10-18) + + +### Bug Fixes + +* **bundlesize:** consolidate usage of "classnames" helper ([#5138](https://github.com/algolia/instantsearch.js/issues/5138)) ([f1ec288](https://github.com/algolia/instantsearch.js/commit/f1ec28889be5c2f906dd398f37d072587e29cf3a)) +* **currentRefinements:** reset page number on refine ([#5136](https://github.com/algolia/instantsearch.js/issues/5136)) ([407b576](https://github.com/algolia/instantsearch.js/commit/407b5767b51c26d5f471071a92f2e32762898f24)) +* **events:** prevent warning on low number of listeners ([#5143](https://github.com/algolia/instantsearch.js/issues/5143)) ([432aa70](https://github.com/algolia/instantsearch.js/commit/432aa7006e7d8eefd1c8c382f59ea2d2974a19da)) + + + +# [4.48.0](https://github.com/algolia/instantsearch.js/compare/v4.47.0...v4.48.0) (2022-10-10) + + +### Bug Fixes + +* **insightsMiddleware:** infer type of insightsClient for onEvent ([#5130](https://github.com/algolia/instantsearch.js/issues/5130)) ([dd5fca4](https://github.com/algolia/instantsearch.js/commit/dd5fca4c185c66f1e31ebe9c0568bcad48e062f3)), closes [#5129](https://github.com/algolia/instantsearch.js/issues/5129) + + +### Features + +* **routing:** include repeated indexId in URL correctly ([#5134](https://github.com/algolia/instantsearch.js/issues/5134)) ([679f5da](https://github.com/algolia/instantsearch.js/commit/679f5dad839536def6ae9c3a18416296d40ed49a)) + + + +# [4.47.0](https://github.com/algolia/instantsearch.js/compare/v4.46.3...v4.47.0) (2022-10-03) + + +### Bug Fixes + +* **hierarchicalMenu:** pass correct attribute name to Insights ([#5124](https://github.com/algolia/instantsearch.js/issues/5124)) ([fe18a16](https://github.com/algolia/instantsearch.js/commit/fe18a168b1b195d067298770b55fd29a7fdb6edb)) + + +### Features + +* **status:** introduce status in InstantSearch class ([#5115](https://github.com/algolia/instantsearch.js/issues/5115)) ([21f3147](https://github.com/algolia/instantsearch.js/commit/21f31476e75e162b38b002d5439f231f3990e785)) +* **hierarchicalMenu**: introduce `ais-HierarchicalMenu-item--selected` class ([#5125](https://github.com/algolia/instantsearch.js/issues/5125)) ([4ebb828](https://github.com/algolia/instantsearch.js/commit/4ebb828c93afabfd8083246dfe7edfd33932d5fd)) + + +## [4.46.3](https://github.com/algolia/instantsearch.js/compare/v4.46.2...v4.46.3) (2022-09-27) + + +### Bug Fixes + +* **currentRefinements:** implement noRefinementRoot modifier class ([#5114](https://github.com/algolia/instantsearch.js/issues/5114)) ([cb66830](https://github.com/algolia/instantsearch.js/commit/cb668305af26bf919841c25bd4cc8493fcdf8cf9)) + + + +## [4.46.2](https://github.com/algolia/instantsearch.js/compare/v4.46.1...v4.46.2) (2022-09-22) + + +### Bug Fixes + +* **build:** remove jsx pragma comments from build output ([#5112](https://github.com/algolia/instantsearch.js/issues/5112)) ([6582083](https://github.com/algolia/instantsearch.js/commit/65820831b7d7e14867f13a2947795491730b8442)) +* **imports:** split out templating from ./utils ([#5111](https://github.com/algolia/instantsearch.js/issues/5111)) ([fc765f3](https://github.com/algolia/instantsearch.js/commit/fc765f35ddd85068237edc81c66932b098e3b55a)), closes [#5109](https://github.com/algolia/instantsearch.js/issues/5109) + + + +## [4.46.1](https://github.com/algolia/instantsearch.js/compare/v4.46.0...v4.46.1) (2022-09-15) + + +### Bug Fixes + +* **hierarchicalMenu:** use existing facet filters in multi queries for parent facet values ([#5105](https://github.com/algolia/instantsearch.js/issues/5105)) ([10a83f1](https://github.com/algolia/instantsearch.js/commit/10a83f146f714d9f97bb8edca2499f16df4ca22d)) +* **insights:** make sure change in userToken can't reset the search parameters ([#5101](https://github.com/algolia/instantsearch.js/issues/5101)) ([b20c8dc](https://github.com/algolia/instantsearch.js/commit/b20c8dc70e34c1f234dc10eb7fc69296f30986a4)) +* **setUiState**: call onStateChange handler ([#5104](https://github.com/algolia/instantsearch.js/issues/5104)) ([231853d](https://github.com/algolia/instantsearch.js/commit/231853dab731189a33ee480cdb196789c7336fda))) + + + +## [4.46.0](https://github.com/algolia/instantsearch.js/compare/v4.45.1...v4.46.0) (2022-09-12) + + +### Features + +* **html:** deprecate Hogan.js and string-based templates ([#5095](https://github.com/algolia/instantsearch.js/issues/5095)) ([a06ddf5](https://github.com/algolia/instantsearch.js/commit/a06ddf55f1ffd1a93cddab2fcf95d2be3220a423)) +* **html:** introduce `html` templating ([#5081](https://github.com/algolia/instantsearch.js/issues/5081)) ([e55e224](https://github.com/algolia/instantsearch.js/commit/e55e2245256193d27f2c85f24b8aab7c9048c554)) + + + +## [4.45.1](https://github.com/algolia/instantsearch.js/compare/v4.45.0...v4.45.1) (2022-09-06) + + +### Bug Fixes + +* **ratingMenu:** fix `undefined` facet values error when `disjunctiveFacets` is empty ([#5096](https://github.com/algolia/instantsearch.js/issues/5096)) ([dd870d5](https://github.com/algolia/instantsearch.js/commit/dd870d5a658ce42b068eadf34f9b69772291aa20)) + + + +# [4.45.0](https://github.com/algolia/instantsearch.js/compare/v4.44.1...v4.45.0) (2022-08-29) + + +### Features + +* **connectors:** deprecate `hasNoResults` in favor of `canRefine` ([#5091](https://github.com/algolia/instantsearch.js/issues/5091)) ([1749a4e](https://github.com/algolia/instantsearch.js/commit/1749a4eb9a2f28fa4a8d442163e3b10acbde7c22)) + + + +## [4.44.1](https://github.com/algolia/instantsearch.js/compare/v4.44.0...v4.44.1) (2022-08-25) + + +### Bug Fixes + +* **connectNumericMenu + connectRange:** stop sending invalid clickedFilters event ([#5085](https://github.com/algolia/instantsearch.js/issues/5085)) ([20996c7](https://github.com/algolia/instantsearch.js/commit/20996c7a159988c58e00ff24d2d2dc98af8b980f)) + + + +# [4.44.0](https://github.com/algolia/instantsearch.js/compare/v4.43.1...v4.44.0) (2022-08-08) + + +### Features + +* **geo-search:** make `GeoHit` type generic ([#5083](https://github.com/algolia/instantsearch.js/issues/5083)) ([3d3c7b2](https://github.com/algolia/instantsearch.js/commit/3d3c7b298b74effe9bb722a04fbb47dc39a4bd95)) + + + +## [4.43.1](https://github.com/algolia/instantsearch.js/compare/v4.43.0...v4.43.1) (2022-07-11) + + +### Bug Fixes + +* **errors:** rethrow error as error if it's an object ([#5075](https://github.com/algolia/instantsearch.js/issues/5075)) ([34132bb](https://github.com/algolia/instantsearch.js/commit/34132bba38c05fa2f5e4e54c6889e9335e62e4f4)) +* **ratingMenu:** don't warn if results are artificial ([#5073](https://github.com/algolia/instantsearch.js/issues/5073)) ([d747d23](https://github.com/algolia/instantsearch.js/commit/d747d23b28c380fe82a40eeab06c57359af8004a)) +* **types:** use correct case for _geoloc property ([#5074](https://github.com/algolia/instantsearch.js/issues/5074)) ([6fed7d8](https://github.com/algolia/instantsearch.js/commit/6fed7d870c3607980776d33a3697f8e2789aa08b)) + + + +# [4.43.0](https://github.com/algolia/instantsearch.js/compare/v4.42.0...v4.43.0) (2022-06-28) + + +### Features + +* **types:** support algoliasearch v5 ([#5066](https://github.com/algolia/instantsearch.js/issues/5066)) ([3eb4dc7](https://github.com/algolia/instantsearch.js/commit/3eb4dc75a5935f2ee4fead8787f39af0150b24c4)) + + + +# [4.42.0](https://github.com/algolia/instantsearch.js/compare/v4.41.2...v4.42.0) (2022-06-21) + + +### Bug Fixes + +* **es:** update import path for `infiniteHitsCache` in depreciation message ([#5068](https://github.com/algolia/instantsearch.js/issues/5068)) ([545cbaf](https://github.com/algolia/instantsearch.js/commit/545cbafd748bb8be32bff66ac60b5f3f9133a5b4)) + + +### Features + +* **core:** sort parameters & support client.search for sffv ([#5069](https://github.com/algolia/instantsearch.js/issues/5069)) ([34e2b00](https://github.com/algolia/instantsearch.js/commit/34e2b00cbc93f1bc86ee0abaec6b6e132bd18354)) + + + +## [4.41.2](https://github.com/algolia/instantsearch.js/compare/v4.41.1...v4.41.2) (2022-06-15) + + +### Bug Fixes + +* **hierarchicalMenu:** show full hierarchical parent values ([#5063](https://github.com/algolia/instantsearch.js/issues/5063)) ([cd1db34](https://github.com/algolia/instantsearch.js/commit/cd1db34815f92acb3d2d0cec6c1ae7865d14fb13)) + + + +## [4.41.1](https://github.com/algolia/instantsearch.js/compare/v4.41.0...v4.41.1) (2022-06-14) + + +### Bug Fixes + +* **insights:** don't send view event if search is stalled ([#5058](https://github.com/algolia/instantsearch.js/issues/5058)) ([1686dfb](https://github.com/algolia/instantsearch.js/commit/1686dfb096cfce062e268feda7956e3b160bf2da)), closes [/github.com/algolia/instantsearch.js/blob/99f6fe1dc51e4815e5b9efcfb30e3e2f3127e763/src/lib/utils/createSendEventForHits.ts#L168](https://github.com//github.com/algolia/instantsearch.js/blob/99f6fe1dc51e4815e5b9efcfb30e3e2f3127e763/src/lib/utils/createSendEventForHits.ts/issues/L168) [/github.com/algolia/instantsearch.js/blob/55313e4ea4105b777f3f102e9f48a7e440496d25/src/middlewares/createInsightsMiddleware.ts#L144](https://github.com//github.com/algolia/instantsearch.js/blob/55313e4ea4105b777f3f102e9f48a7e440496d25/src/middlewares/createInsightsMiddleware.ts/issues/L144) +* **types:** avoid inferring UiState type from initialUiState ([#5061](https://github.com/algolia/instantsearch.js/issues/5061)) ([80ca07e](https://github.com/algolia/instantsearch.js/commit/80ca07e29064357343ee997be94ef10beadba637)), closes [/github.com/Microsoft/TypeScript/issues/14829#issuecomment-504042546](https://github.com//github.com/Microsoft/TypeScript/issues/14829/issues/issuecomment-504042546) [#5060](https://github.com/algolia/instantsearch.js/issues/5060) +* **types:** make all usages of UiState in InstantSearch generic ([#5060](https://github.com/algolia/instantsearch.js/issues/5060)) ([2b9e76b](https://github.com/algolia/instantsearch.js/commit/2b9e76b568fb4d4cc5bd49c384ee583d84d6f39a)) + + + +# [4.41.0](https://github.com/algolia/instantsearch.js/compare/v4.40.6...v4.41.0) (2022-06-01) + + +### Features + +* **core:** don't schedule search without widgets ([#5056](https://github.com/algolia/instantsearch.js/issues/5056)) ([ea3d6d9](https://github.com/algolia/instantsearch.js/commit/ea3d6d9c6ae1fe2f90bf5643d4bdcbb89507e9bc)) + + + +## [4.40.6](https://github.com/algolia/instantsearch.js/compare/v4.40.5...v4.40.6) (2022-05-24) + + +### Bug Fixes + +* **types:** only allow `null` for parent in `getWidgetRenderState` if widget is an index ([#5052](https://github.com/algolia/instantsearch.js/issues/5052)) ([fe0fce0](https://github.com/algolia/instantsearch.js/commit/fe0fce0641ffff9af1d1303b7ee71d77ba08f8bd)) + + + ## [4.40.5](https://github.com/algolia/instantsearch.js/compare/v4.40.4...v4.40.5) (2022-04-26) diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Answers/Answers.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Answers/Answers.js index 6479b56..a0676fe 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Answers/Answers.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Answers/Answers.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -17,17 +17,17 @@ function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (O function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } -function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } - function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } +function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } + var Answers = function Answers(_ref) { var hits = _ref.hits, isLoading = _ref.isLoading, cssClasses = _ref.cssClasses, templateProps = _ref.templateProps; return (0, _preact.h)("div", { - className: (0, _classnames.default)(cssClasses.root, _defineProperty({}, cssClasses.emptyRoot, hits.length === 0)) + className: (0, _uiComponentsShared.cx)(cssClasses.root, hits.length === 0 && cssClasses.emptyRoot) }, (0, _preact.h)(_Template.default, _extends({}, templateProps, { templateKey: "header", rootProps: { diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Breadcrumb/Breadcrumb.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Breadcrumb/Breadcrumb.js index f65495b..7630051 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Breadcrumb/Breadcrumb.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Breadcrumb/Breadcrumb.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -15,8 +15,6 @@ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { de function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - var Breadcrumb = function Breadcrumb(_ref) { var items = _ref.items, cssClasses = _ref.cssClasses, @@ -24,11 +22,11 @@ var Breadcrumb = function Breadcrumb(_ref) { createURL = _ref.createURL, refine = _ref.refine; return (0, _preact.h)("div", { - className: (0, _classnames.default)(cssClasses.root, _defineProperty({}, cssClasses.noRefinementRoot, items.length === 0)) + className: (0, _uiComponentsShared.cx)(cssClasses.root, items.length === 0 && cssClasses.noRefinementRoot) }, (0, _preact.h)("ul", { className: cssClasses.list }, (0, _preact.h)("li", { - className: (0, _classnames.default)(cssClasses.item, _defineProperty({}, cssClasses.selectedItem, items.length === 0)) + className: (0, _uiComponentsShared.cx)(cssClasses.item, items.length === 0 && cssClasses.selectedItem) }, (0, _preact.h)(_Template.default, _extends({}, templateProps, { templateKey: "home", rootTagName: "a", @@ -44,7 +42,7 @@ var Breadcrumb = function Breadcrumb(_ref) { var isLast = idx === items.length - 1; return (0, _preact.h)("li", { key: item.label + idx, - className: (0, _classnames.default)(cssClasses.item, _defineProperty({}, cssClasses.selectedItem, isLast)) + className: (0, _uiComponentsShared.cx)(cssClasses.item, isLast && cssClasses.selectedItem) }, (0, _preact.h)(_Template.default, _extends({}, templateProps, { templateKey: "separator", rootTagName: "span", diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/ClearRefinements/ClearRefinements.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/ClearRefinements/ClearRefinements.js index 8e566a2..96ae284 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/ClearRefinements/ClearRefinements.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/ClearRefinements/ClearRefinements.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -15,8 +15,6 @@ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { de function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - var ClearRefinements = function ClearRefinements(_ref) { var hasRefinements = _ref.hasRefinements, refine = _ref.refine, @@ -28,7 +26,7 @@ var ClearRefinements = function ClearRefinements(_ref) { templateKey: "resetLabel", rootTagName: "button", rootProps: { - className: (0, _classnames.default)(cssClasses.button, _defineProperty({}, cssClasses.disabledButton, !hasRefinements)), + className: (0, _uiComponentsShared.cx)(cssClasses.button, !hasRefinements && cssClasses.disabledButton), onClick: refine, disabled: !hasRefinements }, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/CurrentRefinements/CurrentRefinements.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/CurrentRefinements/CurrentRefinements.js index 729d7f2..0a0b7b3 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/CurrentRefinements/CurrentRefinements.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/CurrentRefinements/CurrentRefinements.js @@ -7,9 +7,10 @@ exports.default = void 0; var _preact = require("preact"); +var _uiComponentsShared = require("@algolia/ui-components-shared"); + var _index = require("../../lib/utils/index.js"); -/** @jsx h */ var createItemKey = function createItemKey(_ref) { var attribute = _ref.attribute, value = _ref.value, @@ -33,9 +34,10 @@ var handleClick = function handleClick(callback) { var CurrentRefinements = function CurrentRefinements(_ref2) { var items = _ref2.items, - cssClasses = _ref2.cssClasses; + cssClasses = _ref2.cssClasses, + canRefine = _ref2.canRefine; return (0, _preact.h)("div", { - className: cssClasses.root + className: (0, _uiComponentsShared.cx)(cssClasses.root, !canRefine && cssClasses.noRefinementRoot) }, (0, _preact.h)("ul", { className: cssClasses.list }, items.map(function (item, index) { diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchButton.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchButton.js index c85a2ca..4578ce7 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchButton.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchButton.js @@ -7,7 +7,6 @@ exports.default = void 0; var _preact = require("preact"); -/** @jsx h */ var GeoSearchButton = function GeoSearchButton(_ref) { var className = _ref.className, _ref$disabled = _ref.disabled, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchControls.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchControls.js index 79735fb..6e91531 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchControls.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchControls.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -19,8 +19,6 @@ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { de function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - var GeoSearchControls = function GeoSearchControls(_ref) { var cssClasses = _ref.cssClasses, enableRefine = _ref.enableRefine, @@ -36,7 +34,7 @@ var GeoSearchControls = function GeoSearchControls(_ref) { return (0, _preact.h)(_preact.Fragment, null, enableRefine && (0, _preact.h)("div", null, enableRefineControl && (0, _preact.h)("div", { className: cssClasses.control }, isRefineOnMapMove || !hasMapMoveSinceLastRefine ? (0, _preact.h)(_GeoSearchToggle.default, { - classNameLabel: (0, _classnames.default)(cssClasses.label, _defineProperty({}, cssClasses.selectedLabel, isRefineOnMapMove)), + classNameLabel: (0, _uiComponentsShared.cx)(cssClasses.label, isRefineOnMapMove && cssClasses.selectedLabel), classNameInput: cssClasses.input, checked: isRefineOnMapMove, onToggle: onRefineToggle @@ -53,7 +51,7 @@ var GeoSearchControls = function GeoSearchControls(_ref) { })))), !enableRefineControl && !isRefineOnMapMove && (0, _preact.h)("div", { className: cssClasses.control }, (0, _preact.h)(_GeoSearchButton.default, { - className: (0, _classnames.default)(cssClasses.redo, _defineProperty({}, cssClasses.disabledRedo, !hasMapMoveSinceLastRefine)), + className: (0, _uiComponentsShared.cx)(cssClasses.redo, !hasMapMoveSinceLastRefine && cssClasses.disabledRedo), disabled: !hasMapMoveSinceLastRefine, onClick: onRefineClick }, (0, _preact.h)(_Template.default, _extends({}, templateProps, { diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchToggle.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchToggle.js index 6aff622..34e6f94 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchToggle.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/GeoSearchControls/GeoSearchToggle.js @@ -7,7 +7,6 @@ exports.default = void 0; var _preact = require("preact"); -/** @jsx h */ var GeoSearchToggle = function GeoSearchToggle(_ref) { var classNameLabel = _ref.classNameLabel, classNameInput = _ref.classNameInput, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Hits/Hits.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Hits/Hits.js index 9e81459..ea0ac0c 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Hits/Hits.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Hits/Hits.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -25,6 +25,7 @@ var Hits = function Hits(_ref) { var results = _ref.results, hits = _ref.hits, bindEvent = _ref.bindEvent, + sendEvent = _ref.sendEvent, cssClasses = _ref.cssClasses, templateProps = _ref.templateProps; @@ -32,7 +33,7 @@ var Hits = function Hits(_ref) { return (0, _preact.h)(_Template.default, _extends({}, templateProps, { templateKey: "empty", rootProps: { - className: (0, _classnames.default)(cssClasses.root, cssClasses.emptyRoot) + className: (0, _uiComponentsShared.cx)(cssClasses.root, cssClasses.emptyRoot) }, data: results })); @@ -53,7 +54,8 @@ var Hits = function Hits(_ref) { data: _objectSpread(_objectSpread({}, hit), {}, { __hitIndex: index }), - bindEvent: bindEvent + bindEvent: bindEvent, + sendEvent: sendEvent })); }))); }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/InfiniteHits/InfiniteHits.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/InfiniteHits/InfiniteHits.js index 49f989f..9d0f425 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/InfiniteHits/InfiniteHits.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/InfiniteHits/InfiniteHits.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -25,6 +25,7 @@ var InfiniteHits = function InfiniteHits(_ref) { var results = _ref.results, hits = _ref.hits, bindEvent = _ref.bindEvent, + sendEvent = _ref.sendEvent, hasShowPrevious = _ref.hasShowPrevious, showPrevious = _ref.showPrevious, showMore = _ref.showMore, @@ -37,7 +38,7 @@ var InfiniteHits = function InfiniteHits(_ref) { return (0, _preact.h)(_Template.default, _extends({}, templateProps, { templateKey: "empty", rootProps: { - className: (0, _classnames.default)(cssClasses.root, cssClasses.emptyRoot) + className: (0, _uiComponentsShared.cx)(cssClasses.root, cssClasses.emptyRoot) }, data: results })); @@ -49,7 +50,7 @@ var InfiniteHits = function InfiniteHits(_ref) { templateKey: "showPreviousText", rootTagName: "button", rootProps: { - className: (0, _classnames.default)(cssClasses.loadPrevious, _defineProperty({}, cssClasses.disabledLoadPrevious, isFirstPage)), + className: (0, _uiComponentsShared.cx)(cssClasses.loadPrevious, isFirstPage && cssClasses.disabledLoadPrevious), disabled: isFirstPage, onClick: showPrevious } @@ -66,13 +67,14 @@ var InfiniteHits = function InfiniteHits(_ref) { data: _objectSpread(_objectSpread({}, hit), {}, { __hitIndex: position }), - bindEvent: bindEvent + bindEvent: bindEvent, + sendEvent: sendEvent })); })), (0, _preact.h)(_Template.default, _extends({}, templateProps, { templateKey: "showMoreText", rootTagName: "button", rootProps: { - className: (0, _classnames.default)(cssClasses.loadMore, _defineProperty({}, cssClasses.disabledLoadMore, isLastPage)), + className: (0, _uiComponentsShared.cx)(cssClasses.loadMore, isLastPage && cssClasses.disabledLoadMore), disabled: isLastPage, onClick: showMore } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/MenuSelect/MenuSelect.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/MenuSelect/MenuSelect.js index 3b2ce16..1db8864 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/MenuSelect/MenuSelect.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/MenuSelect/MenuSelect.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -17,8 +17,6 @@ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { de function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - function MenuSelect(_ref) { var cssClasses = _ref.cssClasses, templateProps = _ref.templateProps, @@ -33,7 +31,7 @@ function MenuSelect(_ref) { selectedValue = _ref2.value; return (0, _preact.h)("div", { - className: (0, _classnames.default)(cssClasses.root, _defineProperty({}, cssClasses.noRefinementRoot, items.length === 0)) + className: (0, _uiComponentsShared.cx)(cssClasses.root, items.length === 0 && cssClasses.noRefinementRoot) }, (0, _preact.h)("select", { className: cssClasses.select, value: selectedValue, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Pagination/Pagination.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Pagination/Pagination.js index e2c878b..5ab61de 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Pagination/Pagination.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Pagination/Pagination.js @@ -7,14 +7,10 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } - -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - function Pagination(props) { function createClickHandler(pageNumber) { return function (event) { @@ -30,7 +26,7 @@ function Pagination(props) { } return (0, _preact.h)("div", { - className: (0, _classnames.default)(props.cssClasses.root, _defineProperty({}, props.cssClasses.noRefinementRoot, props.nbPages <= 1)) + className: (0, _uiComponentsShared.cx)(props.cssClasses.root, props.nbPages <= 1 && props.cssClasses.noRefinementRoot) }, (0, _preact.h)("ul", { className: props.cssClasses.list }, props.showFirst && (0, _preact.h)(PaginationLink, { @@ -97,7 +93,7 @@ function PaginationLink(_ref) { createURL = _ref.createURL, createClickHandler = _ref.createClickHandler; return (0, _preact.h)("li", { - className: (0, _classnames.default)(cssClasses.item, className, isDisabled && cssClasses.disabledItem, isSelected && cssClasses.selectedItem) + className: (0, _uiComponentsShared.cx)(cssClasses.item, className, isDisabled && cssClasses.disabledItem, isSelected && cssClasses.selectedItem) }, isDisabled ? (0, _preact.h)("span", { className: cssClasses.link, dangerouslySetInnerHTML: { diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Panel/Panel.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Panel/Panel.js index 6c9ea90..f9badf0 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Panel/Panel.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Panel/Panel.js @@ -9,14 +9,12 @@ var _preact = require("preact"); var _hooks = require("preact/hooks"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); } function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } @@ -30,8 +28,6 @@ function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !( function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; } function Panel(props) { - var _cx; - var _useState = (0, _hooks.useState)(props.isCollapsed), _useState2 = _slicedToArray(_useState, 2), isCollapsed = _useState2[0], @@ -61,7 +57,7 @@ function Panel(props) { } return (0, _preact.h)("div", { - className: (0, _classnames.default)(props.cssClasses.root, (_cx = {}, _defineProperty(_cx, props.cssClasses.noRefinementRoot, props.hidden), _defineProperty(_cx, props.cssClasses.collapsibleRoot, props.collapsible), _defineProperty(_cx, props.cssClasses.collapsedRoot, isCollapsed), _cx)), + className: (0, _uiComponentsShared.cx)(props.cssClasses.root, props.hidden && props.cssClasses.noRefinementRoot, props.collapsible && props.cssClasses.collapsibleRoot, isCollapsed && props.cssClasses.collapsedRoot), hidden: props.hidden }, props.templates.header && (0, _preact.h)("div", { className: props.cssClasses.header diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/PoweredBy/PoweredBy.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/PoweredBy/PoweredBy.js index c426c2f..b319395 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/PoweredBy/PoweredBy.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/PoweredBy/PoweredBy.js @@ -7,17 +7,6 @@ exports.default = void 0; var _preact = require("preact"); -/** @jsx h */ -var _ref2 = (0, _preact.h)("path", { - fill: "#5468FF", - d: "M78.99.94h16.6a2.97 2.97 0 012.96 2.96v16.6a2.97 2.97 0 01-2.97 2.96h-16.6a2.97 2.97 0 01-2.96-2.96V3.9A2.96 2.96 0 0179 .94" -}); - -var _ref3 = (0, _preact.h)("path", { - fill: "#FFF", - d: "M89.63 5.97v-.78a.98.98 0 00-.98-.97h-2.28a.98.98 0 00-.97.97V6c0 .09.08.15.17.13a7.13 7.13 0 013.9-.02c.08.02.16-.04.16-.13m-6.25 1L83 6.6a.98.98 0 00-1.38 0l-.46.46a.97.97 0 000 1.38l.38.39c.06.06.15.04.2-.02a7.49 7.49 0 011.63-1.62c.07-.04.08-.14.02-.2m4.16 2.45v3.34c0 .1.1.17.2.12l2.97-1.54c.06-.03.08-.12.05-.18a3.7 3.7 0 00-3.08-1.87c-.07 0-.14.06-.14.13m0 8.05a4.49 4.49 0 110-8.98 4.49 4.49 0 010 8.98m0-10.85a6.37 6.37 0 100 12.74 6.37 6.37 0 000-12.74" -}); - var PoweredBy = function PoweredBy(_ref) { var url = _ref.url, theme = _ref.theme, @@ -33,18 +22,18 @@ var PoweredBy = function PoweredBy(_ref) { }, (0, _preact.h)("svg", { height: "1.2em", className: cssClasses.logo, - viewBox: "0 0 168 24" // This style is necessary as long as it's not included in InstantSearch.css. + viewBox: "0 0 572 64" // This style is necessary as long as it's not included in InstantSearch.css. // For now, InstantSearch.css sets a maximum width of 70px. , style: { width: 'auto' } }, (0, _preact.h)("path", { - fill: theme === 'dark' ? '#FFF' : '#5D6494', - d: "M6.97 6.68V8.3a4.47 4.47 0 00-2.42-.67 2.2 2.2 0 00-1.38.4c-.34.26-.5.6-.5 1.02 0 .43.16.77.49 1.03.33.25.83.53 1.51.83a7.04 7.04 0 011.9 1.08c.34.24.58.54.73.89.15.34.23.74.23 1.18 0 .95-.33 1.7-1 2.24a4 4 0 01-2.6.81 5.71 5.71 0 01-2.94-.68v-1.71c.84.63 1.81.94 2.92.94.58 0 1.05-.14 1.39-.4.34-.28.5-.65.5-1.13 0-.29-.1-.55-.3-.8a2.2 2.2 0 00-.65-.53 23.03 23.03 0 00-1.64-.78 13.67 13.67 0 01-1.11-.64c-.12-.1-.28-.22-.46-.4a1.72 1.72 0 01-.39-.5 4.46 4.46 0 01-.22-.6c-.07-.23-.1-.48-.1-.75 0-.91.33-1.63 1-2.17a4 4 0 012.57-.8c.97 0 1.8.18 2.47.52zm7.47 5.7v-.3a2.26 2.26 0 00-.5-1.44c-.3-.35-.74-.53-1.32-.53-.53 0-.99.2-1.37.58a2.9 2.9 0 00-.72 1.68h3.91zm1 2.79v1.4c-.6.34-1.38.51-2.36.51a4.02 4.02 0 01-3-1.13 4.04 4.04 0 01-1.11-2.97c0-1.3.34-2.32 1.02-3.06a3.38 3.38 0 012.6-1.1c1.03 0 1.85.32 2.46.96.6.64.9 1.57.9 2.78 0 .33-.03.68-.09 1.04h-5.31c.1.7.4 1.24.89 1.61.49.38 1.1.56 1.85.56.86 0 1.58-.2 2.15-.6zm6.61-1.78h-1.21c-.6 0-1.05.12-1.35.36-.3.23-.46.53-.46.89 0 .37.12.66.36.88.23.2.57.32 1.02.32.5 0 .9-.15 1.2-.43.3-.28.44-.65.44-1.1v-.92zm-4.07-2.55V9.33a4.96 4.96 0 012.5-.55c2.1 0 3.17 1.03 3.17 3.08V17H22.1v-.96c-.42.68-1.15 1.02-2.19 1.02-.76 0-1.38-.22-1.84-.66-.46-.44-.7-1-.7-1.68 0-.78.3-1.38.88-1.81.59-.43 1.4-.65 2.46-.65h1.34v-.46c0-.55-.13-.97-.4-1.25-.26-.29-.7-.43-1.32-.43-.86 0-1.65.24-2.35.72zm9.34-1.93v1.42c.39-1 1.1-1.5 2.12-1.5.15 0 .31.02.5.05v1.53c-.23-.1-.48-.14-.76-.14-.54 0-.99.24-1.34.71a2.8 2.8 0 00-.52 1.71V17h-1.57V8.91h1.57zm5 4.09a3 3 0 00.76 2.01c.47.53 1.14.8 2 .8.64 0 1.24-.18 1.8-.53v1.4c-.53.32-1.2.48-2 .48a3.98 3.98 0 01-4.17-4.18c0-1.16.38-2.15 1.14-2.98a4 4 0 013.1-1.23c.7 0 1.34.15 1.92.44v1.44a3.24 3.24 0 00-1.77-.5A2.65 2.65 0 0032.33 13zm7.92-7.28v4.58c.46-1 1.3-1.5 2.5-1.5.8 0 1.42.24 1.9.73.48.5.72 1.17.72 2.05V17H43.8v-5.1c0-.56-.14-.99-.43-1.29-.28-.3-.65-.45-1.1-.45-.54 0-1 .2-1.42.6-.4.4-.61 1.02-.61 1.85V17h-1.56V5.72h1.56zM55.2 15.74c.6 0 1.1-.25 1.5-.76.4-.5.6-1.16.6-1.95 0-.92-.2-1.62-.6-2.12-.4-.5-.92-.74-1.55-.74-.56 0-1.05.22-1.5.67-.44.45-.66 1.13-.66 2.06 0 .96.22 1.67.64 2.14.43.47.95.7 1.57.7zM53 5.72v4.42a2.74 2.74 0 012.43-1.34c1.03 0 1.86.38 2.51 1.15.65.76.97 1.78.97 3.05 0 1.13-.3 2.1-.92 2.9-.62.81-1.47 1.21-2.54 1.21s-1.9-.45-2.46-1.34V17h-1.58V5.72H53zm9.9 11.1l-3.22-7.9h1.74l1 2.62 1.26 3.42c.1-.32.48-1.46 1.15-3.42l.91-2.63h1.66l-2.92 7.87c-.78 2.07-1.96 3.1-3.56 3.1-.28 0-.53-.02-.73-.07v-1.34c.17.04.35.06.54.06 1.03 0 1.76-.57 2.17-1.7z" - }), _ref2, _ref3, (0, _preact.h)("path", { - fill: theme === 'dark' ? '#FFF' : '#5468FF', - d: "M120.92 18.8c-4.38.02-4.38-3.54-4.38-4.1V1.36l2.67-.42v13.25c0 .32 0 2.36 1.71 2.37v2.24zm-10.84-2.18c.82 0 1.43-.04 1.85-.12v-2.72a5.48 5.48 0 00-1.57-.2c-.3 0-.6.02-.9.07-.3.04-.57.12-.81.24-.24.11-.44.28-.58.49a.93.93 0 00-.22.65c0 .63.22 1 .61 1.23.4.24.94.36 1.62.36zm-.23-9.7c.88 0 1.62.11 2.23.33.6.22 1.09.53 1.44.92.36.4.61.92.76 1.48.16.56.23 1.17.23 1.85v6.87a21.69 21.69 0 01-4.68.5c-.69 0-1.32-.07-1.9-.2a4 4 0 01-1.46-.63 3.3 3.3 0 01-.96-1.13 4.3 4.3 0 01-.34-1.8 3.13 3.13 0 011.43-2.63c.45-.3.95-.5 1.54-.62a8.8 8.8 0 013.79.05v-.44c0-.3-.04-.6-.11-.87a1.78 1.78 0 00-1.1-1.22 3.2 3.2 0 00-1.15-.2 9.75 9.75 0 00-2.95.46l-.33-2.19a11.43 11.43 0 013.56-.53zm52.84 9.63c.82 0 1.43-.05 1.85-.13V13.7a5.42 5.42 0 00-1.57-.2c-.3 0-.6.02-.9.07-.3.04-.57.12-.81.24-.24.12-.44.28-.58.5a.93.93 0 00-.22.65c0 .63.22.99.61 1.23.4.24.94.36 1.62.36zm-.23-9.7c.88 0 1.63.11 2.23.33.6.22 1.1.53 1.45.92.35.39.6.92.76 1.48.15.56.23 1.18.23 1.85v6.88c-.41.08-1.03.19-1.87.31-.83.12-1.77.18-2.81.18-.7 0-1.33-.06-1.9-.2a4 4 0 01-1.47-.63c-.4-.3-.72-.67-.95-1.13a4.3 4.3 0 01-.34-1.8c0-.66.13-1.08.38-1.53.26-.45.61-.82 1.05-1.1.44-.3.95-.5 1.53-.62a8.8 8.8 0 013.8.05v-.43c0-.31-.04-.6-.12-.88-.07-.28-.2-.52-.38-.73a1.78 1.78 0 00-.73-.5c-.3-.1-.68-.2-1.14-.2a9.85 9.85 0 00-2.95.47l-.32-2.19a11.63 11.63 0 013.55-.53zm-8.03-1.27a1.62 1.62 0 000-3.24 1.62 1.62 0 100 3.24zm1.35 13.22h-2.7V7.27l2.7-.42V18.8zm-4.72 0c-4.38.02-4.38-3.54-4.38-4.1l-.01-13.34 2.67-.42v13.25c0 .32 0 2.36 1.72 2.37v2.24zm-8.7-5.9a4.7 4.7 0 00-.74-2.79 2.4 2.4 0 00-2.07-1 2.4 2.4 0 00-2.06 1 4.7 4.7 0 00-.74 2.8c0 1.16.25 1.94.74 2.62a2.4 2.4 0 002.07 1.02c.88 0 1.57-.34 2.07-1.02a4.2 4.2 0 00.73-2.63zm2.74 0a6.46 6.46 0 01-1.52 4.23c-.49.53-1.07.94-1.76 1.22-.68.29-1.73.45-2.26.45a6.6 6.6 0 01-2.25-.45 5.1 5.1 0 01-2.88-3.13 7.3 7.3 0 01-.01-4.84 5.13 5.13 0 012.9-3.1 5.67 5.67 0 012.22-.42c.81 0 1.56.14 2.24.42.69.29 1.28.69 1.75 1.22.49.52.87 1.15 1.14 1.89a7 7 0 01.43 2.5zm-20.14 0c0 1.11.25 2.36.74 2.88.5.52 1.13.78 1.91.78a4.07 4.07 0 002.12-.6V9.33c-.19-.04-.99-.2-1.76-.23a2.67 2.67 0 00-2.23 1 4.73 4.73 0 00-.78 2.8zm7.44 5.27c0 1.82-.46 3.16-1.4 4-.94.85-2.37 1.27-4.3 1.27-.7 0-2.17-.13-3.34-.4l.43-2.11c.98.2 2.27.26 2.95.26 1.08 0 1.84-.22 2.3-.66.46-.43.68-1.08.68-1.94v-.44a5.2 5.2 0 01-2.54.6 5.6 5.6 0 01-2.01-.36 4.2 4.2 0 01-2.58-2.71 9.88 9.88 0 01.02-5.35 4.92 4.92 0 012.93-2.96 6.6 6.6 0 012.43-.46 19.64 19.64 0 014.43.66v10.6z" + fill: theme === 'dark' ? '#FFF' : '#36395A', + d: "M16 48.3c-3.4 0-6.3-.6-8.7-1.7A12.4 12.4 0 0 1 1.9 42C.6 40 0 38 0 35.4h6.5a6.7 6.7 0 0 0 3.9 6c1.4.7 3.3 1.1 5.6 1.1 2.2 0 4-.3 5.4-1a7 7 0 0 0 3-2.4 6 6 0 0 0 1-3.4c0-1.5-.6-2.8-1.9-3.7-1.3-1-3.3-1.6-5.9-1.8l-4-.4c-3.7-.3-6.6-1.4-8.8-3.4a10 10 0 0 1-3.3-7.9c0-2.4.6-4.6 1.8-6.4a12 12 0 0 1 5-4.3c2.2-1 4.7-1.6 7.5-1.6s5.5.5 7.6 1.6a12 12 0 0 1 5 4.4c1.2 1.8 1.8 4 1.8 6.7h-6.5a6.4 6.4 0 0 0-3.5-5.9c-1-.6-2.6-1-4.4-1s-3.2.3-4.4 1c-1.1.6-2 1.4-2.6 2.4-.5 1-.8 2-.8 3.1a5 5 0 0 0 1.5 3.6c1 1 2.6 1.7 4.7 1.9l4 .3c2.8.2 5.2.8 7.2 1.8 2.1 1 3.7 2.2 4.9 3.8a9.7 9.7 0 0 1 1.7 5.8c0 2.5-.7 4.7-2 6.6a13 13 0 0 1-5.6 4.4c-2.4 1-5.2 1.6-8.4 1.6Zm35.6 0c-2.6 0-4.8-.4-6.7-1.3a13 13 0 0 1-4.7-3.5 17.1 17.1 0 0 1-3.6-10.4v-1c0-2 .3-3.8 1-5.6a13 13 0 0 1 7.3-8.3 15 15 0 0 1 6.3-1.4A13.2 13.2 0 0 1 64 24.3c1 2.2 1.6 4.6 1.6 7.2V34H39.4v-4.3h21.8l-1.8 2.2c0-2-.3-3.7-.9-5.1a7.3 7.3 0 0 0-2.7-3.4c-1.2-.7-2.7-1.1-4.6-1.1s-3.4.4-4.7 1.3a8 8 0 0 0-2.9 3.6c-.6 1.5-.9 3.3-.9 5.4 0 2 .3 3.7 1 5.3a7.9 7.9 0 0 0 2.8 3.7c1.3.8 3 1.3 5 1.3s3.8-.5 5.1-1.3c1.3-1 2.1-2 2.4-3.2h6a11.8 11.8 0 0 1-7 8.7 16 16 0 0 1-6.4 1.2ZM80 48c-2.2 0-4-.3-5.7-1a8.4 8.4 0 0 1-3.7-3.3 9.7 9.7 0 0 1-1.3-5.2c0-2 .5-3.8 1.5-5.2a9 9 0 0 1 4.3-3.1c1.8-.7 4-1 6.7-1H89v4.1h-7.5c-2 0-3.4.5-4.4 1.4-1 1-1.6 2.1-1.6 3.6s.5 2.7 1.6 3.6c1 1 2.5 1.4 4.4 1.4 1.1 0 2.2-.2 3.2-.7 1-.4 1.9-1 2.6-2 .6-1 1-2.4 1-4.2l1.7 2.1c-.2 2-.7 3.8-1.5 5.2a9 9 0 0 1-3.4 3.3 12 12 0 0 1-5.3 1Zm9.5-.7v-8.8h-1v-10c0-1.8-.5-3.2-1.4-4.1-1-1-2.4-1.4-4.2-1.4a142.9 142.9 0 0 0-10.2.4v-5.6a74.8 74.8 0 0 1 8.6-.4c3 0 5.5.4 7.5 1.2s3.4 2 4.4 3.6c1 1.7 1.4 4 1.4 6.7v18.4h-5Zm12.9 0V17.8h5v12.3h-.2c0-4.2 1-7.4 2.8-9.5a11 11 0 0 1 8.3-3.1h1v5.6h-2a9 9 0 0 0-6.3 2.2c-1.5 1.5-2.2 3.6-2.2 6.4v15.6h-6.4Zm34.4 1a15 15 0 0 1-6.6-1.3c-1.9-.9-3.4-2-4.7-3.5a15.5 15.5 0 0 1-2.7-5c-.6-1.7-1-3.6-1-5.4v-1c0-2 .4-3.8 1-5.6a15 15 0 0 1 2.8-4.9c1.3-1.5 2.8-2.6 4.6-3.5a16.4 16.4 0 0 1 13.3.2c2 1 3.5 2.3 4.8 4a12 12 0 0 1 2 6H144c-.2-1.6-1-3-2.2-4.1a7.5 7.5 0 0 0-5.2-1.7 8 8 0 0 0-4.7 1.3 8 8 0 0 0-2.8 3.6 13.8 13.8 0 0 0 0 10.3c.6 1.5 1.5 2.7 2.8 3.6s2.8 1.3 4.8 1.3c1.5 0 2.7-.2 3.8-.8a7 7 0 0 0 2.6-2c.7-1 1-2 1.2-3.2h6.2a11 11 0 0 1-2 6.2 15.1 15.1 0 0 1-11.8 5.5Zm19.7-1v-40h6.4V31h-1.3c0-3 .4-5.5 1.1-7.6a9.7 9.7 0 0 1 3.5-4.8A9.9 9.9 0 0 1 172 17h.3c3.5 0 6 1.1 7.9 3.5 1.7 2.3 2.6 5.7 2.6 10v16.8h-6.4V29.6c0-2.1-.6-3.8-1.8-5a6.4 6.4 0 0 0-4.8-1.8c-2 0-3.7.7-5 2a7.8 7.8 0 0 0-1.9 5.5v17h-6.4Zm63.8 1a12.2 12.2 0 0 1-10.9-6.2 19 19 0 0 1-1.8-7.3h1.4v12.5h-5.1v-40h6.4v19.8l-2 3.5c.2-3.1.8-5.7 1.9-7.7a11 11 0 0 1 4.4-4.5c1.8-1 3.9-1.5 6.1-1.5a13.4 13.4 0 0 1 12.8 9.1c.7 1.9 1 3.8 1 6v1c0 2.2-.3 4.1-1 6a13.6 13.6 0 0 1-13.2 9.4Zm-1.2-5.5a8.4 8.4 0 0 0 7.9-5c.7-1.5 1.1-3.3 1.1-5.3s-.4-3.8-1.1-5.3a8.7 8.7 0 0 0-3.2-3.6 9.6 9.6 0 0 0-9.2-.2 8.5 8.5 0 0 0-3.3 3.2c-.8 1.4-1.3 3-1.3 5v2.3a9 9 0 0 0 1.3 4.8 9 9 0 0 0 3.4 3c1.4.7 2.8 1 4.4 1Zm27.3 3.9-10-28.9h6.5l9.5 28.9h-6Zm-7.5 12.2v-5.7h4.9c1 0 2-.1 2.9-.4a4 4 0 0 0 2-1.4c.4-.7.9-1.6 1.2-2.7l8.6-30.9h6.2l-9.3 32.4a14 14 0 0 1-2.5 5 8.9 8.9 0 0 1-4 2.8c-1.5.6-3.4.9-5.6.9h-4.4Zm9-12.2v-5.2h6.4v5.2H248Z" + }), (0, _preact.h)("path", { + fill: theme === 'dark' ? '#FFF' : '#003DFF', + d: "M534.4 9.1H528a.8.8 0 0 1-.7-.7V1.8c0-.4.2-.7.6-.8l6.5-1c.4 0 .8.2.9.6v7.8c0 .4-.4.7-.8.7zM428 35.2V.8c0-.5-.3-.8-.7-.8h-.2l-6.4 1c-.4 0-.7.4-.7.8v35c0 1.6 0 11.8 12.3 12.2.5 0 .8-.4.8-.8V43c0-.4-.3-.7-.6-.8-4.5-.5-4.5-6-4.5-7zm106.5-21.8H528c-.4 0-.7.4-.7.8v34c0 .4.3.8.7.8h6.5c.4 0 .8-.4.8-.8v-34c0-.5-.4-.8-.8-.8zm-17.7 21.8V.8c0-.5-.3-.8-.8-.8l-6.5 1c-.4 0-.7.4-.7.8v35c0 1.6 0 11.8 12.3 12.2.4 0 .8-.4.8-.8V43c0-.4-.3-.7-.7-.8-4.4-.5-4.4-6-4.4-7zm-22.2-20.6a16.5 16.5 0 0 1 8.6 9.3c.8 2.2 1.3 4.8 1.3 7.5a19.4 19.4 0 0 1-4.6 12.6 14.8 14.8 0 0 1-5.2 3.6c-2 .9-5.2 1.4-6.8 1.4a21 21 0 0 1-6.7-1.4 15.4 15.4 0 0 1-8.6-9.3 21.3 21.3 0 0 1 0-14.4 15.2 15.2 0 0 1 8.6-9.3c2-.8 4.3-1.2 6.7-1.2s4.6.4 6.7 1.2zm-6.7 27.6c2.7 0 4.7-1 6.2-3s2.2-4.3 2.2-7.8-.7-6.3-2.2-8.3-3.5-3-6.2-3-4.7 1-6.1 3c-1.5 2-2.2 4.8-2.2 8.3s.7 5.8 2.2 7.8 3.5 3 6.2 3zm-88.8-28.8c-6.2 0-11.7 3.3-14.8 8.2a18.6 18.6 0 0 0 4.8 25.2c1.8 1.2 4 1.8 6.2 1.7s.1 0 .1 0h.9c4.2-.7 8-4 9.1-8.1v7.4c0 .4.3.7.8.7h6.4a.7.7 0 0 0 .7-.7V14.2c0-.5-.3-.8-.7-.8h-13.5zm6.3 26.5a9.8 9.8 0 0 1-5.7 2h-.5a10 10 0 0 1-9.2-14c1.4-3.7 5-6.3 9-6.3h6.4v18.3zm152.3-26.5h13.5c.5 0 .8.3.8.7v33.7c0 .4-.3.7-.8.7h-6.4a.7.7 0 0 1-.8-.7v-7.4c-1.2 4-4.8 7.4-9 8h-.1a4.2 4.2 0 0 1-.5.1h-.9a10.3 10.3 0 0 1-7-2.6c-4-3.3-6.5-8.4-6.5-14.2 0-3.7 1-7.2 3-10 3-5 8.5-8.3 14.7-8.3zm.6 28.4c2.2-.1 4.2-.6 5.7-2V21.7h-6.3a9.8 9.8 0 0 0-9 6.4 10.2 10.2 0 0 0 9.1 13.9h.5zM452.8 13.4c-6.2 0-11.7 3.3-14.8 8.2a18.5 18.5 0 0 0 3.6 24.3 10.4 10.4 0 0 0 13 .6c2.2-1.5 3.8-3.7 4.5-6.1v7.8c0 2.8-.8 5-2.2 6.3-1.5 1.5-4 2.2-7.5 2.2l-6-.3c-.3 0-.7.2-.8.5l-1.6 5.5c-.1.4.1.8.5 1h.1c2.8.4 5.5.6 7 .6 6.3 0 11-1.4 14-4.1 2.7-2.5 4.2-6.3 4.5-11.4V14.2c0-.5-.4-.8-.8-.8h-13.5zm6.3 8.2v18.3a9.6 9.6 0 0 1-5.6 2h-1a10.3 10.3 0 0 1-8.8-14c1.4-3.7 5-6.3 9-6.3h6.4zM291 31.5A32 32 0 0 1 322.8 0h30.8c.6 0 1.2.5 1.2 1.2v61.5c0 1.1-1.3 1.7-2.2 1l-19.2-17a18 18 0 0 1-11 3.4 18.1 18.1 0 1 1 18.2-14.8c-.1.4-.5.7-.9.6-.1 0-.3 0-.4-.2l-3.8-3.4c-.4-.3-.6-.8-.7-1.4a12 12 0 1 0-2.4 8.3c.4-.4 1-.5 1.6-.2l14.7 13.1v-46H323a26 26 0 1 0 10 49.7c.8-.4 1.6-.2 2.3.3l3 2.7c.3.2.3.7 0 1l-.2.2a32 32 0 0 1-47.2-28.6z" })))); }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/QueryRuleCustomData/QueryRuleCustomData.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/QueryRuleCustomData/QueryRuleCustomData.js index 24b05b5..ec6086f 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/QueryRuleCustomData/QueryRuleCustomData.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/QueryRuleCustomData/QueryRuleCustomData.js @@ -11,7 +11,6 @@ var _Template = _interopRequireDefault(require("../Template/Template.js")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } -/** @jsx h */ var QueryRuleCustomData = function QueryRuleCustomData(_ref) { var cssClasses = _ref.cssClasses, templates = _ref.templates, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/RangeInput/RangeInput.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/RangeInput/RangeInput.js index 19547a6..7fc578d 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/RangeInput/RangeInput.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/RangeInput/RangeInput.js @@ -9,7 +9,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -100,7 +100,7 @@ var RangeInput = /*#__PURE__*/function (_Component) { templateProps = _this$props.templateProps; var isDisabled = min && max ? min >= max : false; var hasRefinements = Boolean(minValue || maxValue); - var rootClassNames = (0, _classnames.default)(cssClasses.root, _defineProperty({}, cssClasses.noRefinement, !hasRefinements)); + var rootClassNames = (0, _uiComponentsShared.cx)(cssClasses.root, !hasRefinements && cssClasses.noRefinement); return (0, _preact.h)("div", { className: rootClassNames }, (0, _preact.h)("form", { @@ -109,7 +109,7 @@ var RangeInput = /*#__PURE__*/function (_Component) { }, (0, _preact.h)("label", { className: cssClasses.label }, (0, _preact.h)("input", { - className: (0, _classnames.default)(cssClasses.input, cssClasses.inputMin), + className: (0, _uiComponentsShared.cx)(cssClasses.input, cssClasses.inputMin), type: "number", min: min, max: max, @@ -127,7 +127,7 @@ var RangeInput = /*#__PURE__*/function (_Component) { })), (0, _preact.h)("label", { className: cssClasses.label }, (0, _preact.h)("input", { - className: (0, _classnames.default)(cssClasses.input, cssClasses.inputMax), + className: (0, _uiComponentsShared.cx)(cssClasses.input, cssClasses.inputMax), type: "number", min: min, max: max, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/RefinementList/RefinementList.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/RefinementList/RefinementList.js index 0e9454e..4c2bef6 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/RefinementList/RefinementList.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/RefinementList/RefinementList.js @@ -9,7 +9,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -94,8 +94,6 @@ var RefinementList = /*#__PURE__*/function (_Component) { }, { key: "_generateFacetItem", value: function _generateFacetItem(facetValue) { - var _cx; - var subItems; if (isHierarchicalMenuItem(facetValue) && Array.isArray(facetValue.data) && facetValue.data.length > 0) { @@ -133,7 +131,7 @@ var RefinementList = /*#__PURE__*/function (_Component) { key += "/".concat(facetValue.count); } - var refinementListItemClassName = (0, _classnames.default)(this.props.cssClasses.item, (_cx = {}, _defineProperty(_cx, this.props.cssClasses.selectedItem, facetValue.isRefined), _defineProperty(_cx, this.props.cssClasses.disabledItem, !facetValue.count), _defineProperty(_cx, this.props.cssClasses.parentItem, isHierarchicalMenuItem(facetValue) && Array.isArray(facetValue.data) && facetValue.data.length > 0), _cx)); + var refinementListItemClassName = (0, _uiComponentsShared.cx)(this.props.cssClasses.item, facetValue.isRefined && this.props.cssClasses.selectedItem, !facetValue.count && this.props.cssClasses.disabledItem, Boolean(isHierarchicalMenuItem(facetValue) && Array.isArray(facetValue.data) && facetValue.data.length > 0) && this.props.cssClasses.parentItem); return (0, _preact.h)(_RefinementListItem.default, { templateKey: "item", key: key, @@ -227,7 +225,7 @@ var RefinementList = /*#__PURE__*/function (_Component) { value: function render() { var _this2 = this; - var showMoreButtonClassName = (0, _classnames.default)(this.props.cssClasses.showMore, _defineProperty({}, this.props.cssClasses.disabledShowMore, !(this.props.showMore === true && this.props.canToggleShowMore))); + var showMoreButtonClassName = (0, _uiComponentsShared.cx)(this.props.cssClasses.showMore, !(this.props.showMore === true && this.props.canToggleShowMore) && this.props.cssClasses.disabledShowMore); var showMoreButton = this.props.showMore === true && (0, _preact.h)(_Template.default, _extends({}, this.props.templateProps, { templateKey: "showMoreText", rootTagName: "button", @@ -271,7 +269,7 @@ var RefinementList = /*#__PURE__*/function (_Component) { className: this.props.cssClasses.noResults } })); - var rootClassName = (0, _classnames.default)(this.props.cssClasses.root, _defineProperty({}, this.props.cssClasses.noRefinementRoot, !this.props.facetValues || this.props.facetValues.length === 0), this.props.className); + var rootClassName = (0, _uiComponentsShared.cx)(this.props.cssClasses.root, (!this.props.facetValues || this.props.facetValues.length === 0) && this.props.cssClasses.noRefinementRoot, this.props.className); return (0, _preact.h)("div", { className: rootClassName }, this.props.children, searchBox, facetValues, noResults, showMoreButton); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/RelevantSort/RelevantSort.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/RelevantSort/RelevantSort.js index 2c7e790..217b229 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/RelevantSort/RelevantSort.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/RelevantSort/RelevantSort.js @@ -11,7 +11,6 @@ var _Template = _interopRequireDefault(require("../Template/Template.js")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } -/** @jsx h */ var RelevantSort = function RelevantSort(_ref) { var cssClasses = _ref.cssClasses, templates = _ref.templates, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Selector/Selector.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Selector/Selector.js index b89220d..e60b277 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Selector/Selector.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Selector/Selector.js @@ -7,25 +7,22 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } - -/** @jsx h */ function Selector(_ref) { var currentValue = _ref.currentValue, options = _ref.options, cssClasses = _ref.cssClasses, setValue = _ref.setValue; return (0, _preact.h)("select", { - className: (0, _classnames.default)(cssClasses.select), + className: (0, _uiComponentsShared.cx)(cssClasses.select), onChange: function onChange(event) { return setValue(event.target.value); }, value: "".concat(currentValue) }, options.map(function (option) { return (0, _preact.h)("option", { - className: (0, _classnames.default)(cssClasses.option), + className: (0, _uiComponentsShared.cx)(cssClasses.option), key: option.label + option.value, value: "".concat(option.value) }, option.label); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Pit.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Pit.js index cccf081..4cd3fc5 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Pit.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Pit.js @@ -7,9 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _uiComponentsShared = require("@algolia/ui-components-shared"); function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } @@ -29,9 +27,7 @@ var Pit = function Pit(_ref) { style: _objectSpread(_objectSpread({}, style), {}, { marginLeft: positionValue === 100 ? '-2px' : 0 }), - className: (0, _classnames.default)('rheostat-marker', 'rheostat-marker-horizontal', { - 'rheostat-marker-large': shouldDisplayValue - }) + className: (0, _uiComponentsShared.cx)('rheostat-marker', 'rheostat-marker-horizontal', shouldDisplayValue && 'rheostat-marker-large') }, shouldDisplayValue && (0, _preact.h)("div", { className: 'rheostat-value' }, pitValue)); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Slider.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Slider.js index c44d540..8dde613 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Slider.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Slider/Slider.js @@ -9,7 +9,7 @@ var _preact = require("preact"); var _Rheostat = _interopRequireDefault(require("./Rheostat.js")); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -86,10 +86,7 @@ var Slider = /*#__PURE__*/function (_Component) { var roundedValue = Math.round( // have to cast as a string, as the value given to the prop is a number, but becomes a string when read parseFloat(props['aria-valuenow']) * 100) / 100; var value = _typeof(tooltips) === 'object' && tooltips.format ? tooltips.format(roundedValue) : roundedValue; - var className = (0, _classnames.default)(props.className, { - 'rheostat-handle-lower': props['data-handle-key'] === 0, - 'rheostat-handle-upper': props['data-handle-key'] === 1 - }); + var className = (0, _uiComponentsShared.cx)(props.className, props['data-handle-key'] === 0 && 'rheostat-handle-lower', props['data-handle-key'] === 1 && 'rheostat-handle-upper'); return (0, _preact.h)("div", _extends({}, props, { className: className }), tooltips && (0, _preact.h)("div", { @@ -163,7 +160,7 @@ var Slider = /*#__PURE__*/function (_Component) { max: max }); return (0, _preact.h)("div", { - className: (0, _classnames.default)(cssClasses.root, _defineProperty({}, cssClasses.disabledRoot, this.isDisabled)) + className: (0, _uiComponentsShared.cx)(cssClasses.root, this.isDisabled && cssClasses.disabledRoot) }, (0, _preact.h)(_Rheostat.default, { handle: this.createHandleComponent(tooltips), onChange: this.handleChange, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Stats/Stats.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Stats/Stats.js index 069e8eb..adedb98 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Stats/Stats.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Stats/Stats.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Template = _interopRequireDefault(require("../Template/Template.js")); @@ -33,7 +33,7 @@ var Stats = function Stats(_ref) { rest = _objectWithoutProperties(_ref, ["nbHits", "nbSortedHits", "cssClasses", "templateProps"]); return (0, _preact.h)("div", { - className: (0, _classnames.default)(cssClasses.root) + className: (0, _uiComponentsShared.cx)(cssClasses.root) }, (0, _preact.h)(_Template.default, _extends({}, templateProps, { templateKey: "text", rootTagName: "span", diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Template/Template.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Template/Template.js index 1c526f3..cd064be 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/Template/Template.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/Template/Template.js @@ -1,7 +1,5 @@ "use strict"; -function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); } - Object.defineProperty(exports, "__esModule", { value: true }); @@ -11,8 +9,12 @@ var _preact = require("preact"); var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } +function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); } + function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } @@ -63,16 +65,22 @@ var Template = /*#__PURE__*/function (_Component) { }, { key: "render", value: function render() { + var _this = this; + + process.env.NODE_ENV === 'development' ? (0, _index.warning)(Object.keys(this.props.templates).every(function (key) { + return typeof _this.props.templates[key] === 'function'; + }), "Hogan.js and string-based templates are deprecated and will not be supported in InstantSearch.js 5.x.\n\nYou can replace them with function-form templates and use either the provided `html` function or JSX templates.\n\nSee: https://www.algolia.com/doc/guides/building-search-ui/upgrade-guides/js/#upgrade-templates") : void 0; var RootTagName = this.props.rootTagName; var useCustomCompileOptions = this.props.useCustomCompileOptions[this.props.templateKey]; var compileOptions = useCustomCompileOptions ? this.props.templatesConfig.compileOptions : {}; - var content = (0, _index.renderTemplate)({ + var content = (0, _index2.renderTemplate)({ templates: this.props.templates, templateKey: this.props.templateKey, compileOptions: compileOptions, helpers: this.props.templatesConfig.helpers, data: this.props.data, - bindEvent: this.props.bindEvent + bindEvent: this.props.bindEvent, + sendEvent: this.props.sendEvent }); if (content === null) { @@ -81,6 +89,10 @@ var Template = /*#__PURE__*/function (_Component) { return null; } + if (_typeof(content) === 'object') { + return (0, _preact.h)(RootTagName, this.props.rootProps, content); + } + return (0, _preact.h)(RootTagName, _extends({}, this.props.rootProps, { dangerouslySetInnerHTML: { __html: content diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/components/VoiceSearch/VoiceSearch.js b/wp-search-with-algolia/js/instantsearch.js/cjs/components/VoiceSearch/VoiceSearch.js index 85ec7b3..8882f62 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/components/VoiceSearch/VoiceSearch.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/components/VoiceSearch/VoiceSearch.js @@ -11,7 +11,6 @@ var _Template = _interopRequireDefault(require("../Template/Template.js")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } -/** @jsx h */ var VoiceSearch = function VoiceSearch(_ref) { var cssClasses = _ref.cssClasses, isBrowserSupported = _ref.isBrowserSupported, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/current-refinements/connectCurrentRefinements.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/current-refinements/connectCurrentRefinements.js index 7ba0b3b..5f87cc4 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/current-refinements/connectCurrentRefinements.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/current-refinements/connectCurrentRefinements.js @@ -149,6 +149,8 @@ function getRefinementsItems(_ref3) { } function clearRefinementFromState(state, refinement) { + state = state.resetPage(); + switch (refinement.type) { case 'facet': return state.removeFacetRefinement(refinement.attribute, String(refinement.value)); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hierarchical-menu/connectHierarchicalMenu.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hierarchical-menu/connectHierarchicalMenu.js index 5f8f951..a7c7904 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hierarchical-menu/connectHierarchicalMenu.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hierarchical-menu/connectHierarchicalMenu.js @@ -175,7 +175,10 @@ var connectHierarchicalMenu = function connectHierarchicalMenu(renderFn) { sendEvent = (0, _index.createSendEventForFacet)({ instantSearchInstance: instantSearchInstance, helper: helper, - attribute: hierarchicalFacetName, + attribute: function attribute(facetValue) { + var index = facetValue.split(separator).length - 1; + return attributes[index]; + }, widgetType: this.$$type }); } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hits-per-page/connectHitsPerPage.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hits-per-page/connectHitsPerPage.js index 95101fa..015b477 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hits-per-page/connectHitsPerPage.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/hits-per-page/connectHitsPerPage.js @@ -127,6 +127,7 @@ var connectHitsPerPage = function connectHitsPerPage(renderFn) { results = _ref5.results, createURL = _ref5.createURL, helper = _ref5.helper; + var canRefine = results ? results.nbHits > 0 : false; return { items: transformItems(normalizeItems(state), { results: results @@ -136,7 +137,8 @@ var connectHitsPerPage = function connectHitsPerPage(renderFn) { state: state, createURL: createURL }), - hasNoResults: results ? results.nbHits === 0 : true, + hasNoResults: !canRefine, + canRefine: canRefine, widgetParams: widgetParams }; }, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/numeric-menu/connectNumericMenu.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/numeric-menu/connectNumericMenu.js index 21d0a1f..902c738 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/numeric-menu/connectNumericMenu.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/numeric-menu/connectNumericMenu.js @@ -7,6 +7,8 @@ exports.default = void 0; var _index = require("../../lib/utils/index.js"); +function _createForOfIteratorHelper(o, allowArrayLike) { var it; if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e2) { throw _e2; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = o[Symbol.iterator](); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e3) { didErr = true; err = _e3; }, f: function f() { try { if (!normalCompletion && it.return != null) it.return(); } finally { if (didErr) throw err; } } }; } + function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); } function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } @@ -32,46 +34,11 @@ var withUsage = (0, _index.createDocumentationMessageGenerator)({ var $$type = 'ais.numericMenu'; var createSendEvent = function createSendEvent(_ref) { - var instantSearchInstance = _ref.instantSearchInstance, - helper = _ref.helper, - attribute = _ref.attribute; + var instantSearchInstance = _ref.instantSearchInstance; return function () { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } - - if (args.length === 1) { - instantSearchInstance.sendEventToInsights(args[0]); - return; - } - - var eventType = args[0], - facetValue = args[1], - _args$ = args[2], - eventName = _args$ === void 0 ? 'Filter Applied' : _args$; - - if (eventType !== 'click') { + if (arguments.length === 1) { + instantSearchInstance.sendEventToInsights(arguments.length <= 0 ? undefined : arguments[0]); return; - } // facetValue === "%7B%22start%22:5,%22end%22:10%7D" - - - var filters = (0, _index.convertNumericRefinementsToFilters)(getRefinedState(helper.state, attribute, facetValue), attribute); - - if (filters && filters.length > 0) { - /* - filters === ["price<=10", "price>=5"] - */ - instantSearchInstance.sendEventToInsights({ - insightsMethod: 'clickedFilters', - widgetType: $$type, - eventType: eventType, - payload: { - eventName: eventName, - index: helper.getIndex(), - filters: filters - }, - attribute: attribute - }); } }; }; @@ -216,18 +183,39 @@ var connectNumericMenu = function connectNumericMenu(renderFn) { if (!connectorState.sendEvent) { connectorState.sendEvent = createSendEvent({ - instantSearchInstance: instantSearchInstance, - helper: helper, - attribute: attribute + instantSearchInstance: instantSearchInstance }); } + var hasNoResults = results ? results.nbHits === 0 : true; + var preparedItems = prepareItems(state); + var allIsSelected = true; + + var _iterator = _createForOfIteratorHelper(preparedItems), + _step; + + try { + for (_iterator.s(); !(_step = _iterator.n()).done;) { + var item = _step.value; + + if (item.isRefined && decodeURI(item.value) !== '{}') { + allIsSelected = false; + break; + } + } + } catch (err) { + _iterator.e(err); + } finally { + _iterator.f(); + } + return { createURL: connectorState.createURL(state), - items: transformItems(prepareItems(state), { + items: transformItems(preparedItems, { results: results }), - hasNoResults: results ? results.nbHits === 0 : true, + hasNoResults: hasNoResults, + canRefine: !(hasNoResults && allIsSelected), refine: connectorState.refine, sendEvent: connectorState.sendEvent, widgetParams: widgetParams diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/range/connectRange.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/range/connectRange.js index ad7ce0a..771e4dd 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/range/connectRange.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/range/connectRange.js @@ -156,50 +156,12 @@ var connectRange = function connectRange(renderFn) { return null; }; - var sendEventWithRefinedState = function sendEventWithRefinedState(refinedState, instantSearchInstance, helper) { - var eventName = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'Filter Applied'; - var filters = (0, _index.convertNumericRefinementsToFilters)(refinedState, attribute); - - if (filters && filters.length > 0) { - instantSearchInstance.sendEventToInsights({ - insightsMethod: 'clickedFilters', - widgetType: $$type, - eventType: 'click', - payload: { - eventName: eventName, - index: helper.getIndex(), - filters: filters - }, - attribute: attribute - }); - } - }; - - var createSendEvent = function createSendEvent(instantSearchInstance, helper, currentRange) { + var createSendEvent = function createSendEvent(instantSearchInstance) { return function () { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } - - if (args.length === 1) { - instantSearchInstance.sendEventToInsights(args[0]); + if (arguments.length === 1) { + instantSearchInstance.sendEventToInsights(arguments.length <= 0 ? undefined : arguments[0]); return; } - - var eventType = args[0], - facetValue = args[1], - eventName = args[2]; - - if (eventType !== 'click') { - return; - } - - var _facetValue = _slicedToArray(facetValue, 2), - nextMin = _facetValue[0], - nextMax = _facetValue[1]; - - var refinedState = getRefinedState(helper, currentRange, nextMin, nextMax); - sendEventWithRefinedState(refinedState, instantSearchInstance, helper, eventName); }; }; @@ -245,7 +207,7 @@ var connectRange = function connectRange(renderFn) { return [min, max]; } - function _refine(instantSearchInstance, helper, currentRange) { + function _refine(helper, currentRange) { return function () { var _ref11 = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [undefined, undefined], _ref12 = _slicedToArray(_ref11, 2), @@ -255,7 +217,6 @@ var connectRange = function connectRange(renderFn) { var refinedState = getRefinedState(helper, currentRange, nextMin, nextMax); if (refinedState) { - sendEventWithRefinedState(refinedState, instantSearchInstance, helper); helper.setState(refinedState).search(); } }; @@ -301,12 +262,12 @@ var connectRange = function connectRange(renderFn) { // On first render pass an empty range // to be able to bypass the validation // related to it - refine = _refine(instantSearchInstance, helper, { + refine = _refine(helper, { min: undefined, max: undefined }); } else { - refine = _refine(instantSearchInstance, helper, currentRange); + refine = _refine(helper, currentRange); } return { @@ -314,7 +275,7 @@ var connectRange = function connectRange(renderFn) { canRefine: currentRange.min !== currentRange.max, format: rangeFormatter, range: currentRange, - sendEvent: createSendEvent(instantSearchInstance, helper, currentRange), + sendEvent: createSendEvent(instantSearchInstance), widgetParams: _objectSpread(_objectSpread({}, widgetParams), {}, { precision: precision }), diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/rating-menu/connectRatingMenu.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/rating-menu/connectRatingMenu.js index 1ac3e0d..b2eac99 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/rating-menu/connectRatingMenu.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/rating-menu/connectRatingMenu.js @@ -218,12 +218,15 @@ var connectRatingMenu = function connectRatingMenu(renderFn) { }); } - if (results) { - var facetResults = results.getFacetValues(attribute, {}); + var refinementIsApplied = false; + var totalCount = 0; + var facetResults = results === null || results === void 0 ? void 0 : results.getFacetValues(attribute, {}); + + if (results && facetResults) { var maxValuesPerFacet = facetResults.length; var maxDecimalPlaces = getFacetsMaxDecimalPlaces(facetResults); var maxFacets = Math.pow(10, maxDecimalPlaces) * max; - process.env.NODE_ENV === 'development' ? (0, _index.warning)(maxFacets <= maxValuesPerFacet, getFacetValuesWarningMessage({ + process.env.NODE_ENV === 'development' ? (0, _index.warning)(maxFacets <= maxValuesPerFacet || Boolean(results.__isArtificial), getFacetValuesWarningMessage({ maxDecimalPlaces: maxDecimalPlaces, maxFacets: maxFacets, maxValuesPerFacet: maxValuesPerFacet @@ -233,6 +236,7 @@ var connectRatingMenu = function connectRatingMenu(renderFn) { var _loop = function _loop(star) { var isRefined = refinedStar === star; + refinementIsApplied = refinementIsApplied || isRefined; var count = facetResults.filter(function (f) { return Number(f.name) >= star && Number(f.name) <= max; }).map(function (f) { @@ -240,6 +244,7 @@ var connectRatingMenu = function connectRatingMenu(renderFn) { }).reduce(function (sum, current) { return sum + current; }, 0); + totalCount += count; if (refinedStar && !isRefined && count === 0) { // skip count==0 when at least 1 refinement is enabled @@ -269,10 +274,11 @@ var connectRatingMenu = function connectRatingMenu(renderFn) { } facetValues = facetValues.reverse(); + var hasNoResults = results ? results.nbHits === 0 : true; return { items: facetValues, - hasNoResults: results ? results.nbHits === 0 : true, - canRefine: facetValues.length > 0, + hasNoResults: hasNoResults, + canRefine: (!hasNoResults || refinementIsApplied) && totalCount > 0, refine: connectorState.toggleRefinementFactory(helper), sendEvent: sendEvent, createURL: connectorState.createURLFactory({ diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/sort-by/connectSortBy.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/sort-by/connectSortBy.js index aeb86e2..7675fbd 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/sort-by/connectSortBy.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/sort-by/connectSortBy.js @@ -85,13 +85,15 @@ var connectSortBy = function connectSortBy(renderFn) { }; } + var hasNoResults = results ? results.nbHits === 0 : true; return { currentRefinement: state.index, options: transformItems(items, { results: results }), refine: connectorState.setIndex, - hasNoResults: results ? results.nbHits === 0 : true, + hasNoResults: hasNoResults, + canRefine: !hasNoResults && items.length > 0, widgetParams: widgetParams }; }, diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/toggle-refinement/connectToggleRefinement.js b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/toggle-refinement/connectToggleRefinement.js index b5c0c9b..8c3a978 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/toggle-refinement/connectToggleRefinement.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/connectors/toggle-refinement/connectToggleRefinement.js @@ -88,7 +88,9 @@ var connectToggleRefinement = function connectToggleRefinement(renderFn) { throw new Error(withUsage('The `attribute` option is required.')); } - var hasAnOffValue = userOff !== undefined; + var hasAnOffValue = userOff !== undefined; // even though facet values can be numbers and boolean, + // the helper methods only accept string in the type + var on = (0, _index.toArray)(userOn).map(_index.escapeFacetValue); var off = hasAnOffValue ? (0, _index.toArray)(userOff).map(_index.escapeFacetValue) : undefined; var sendEvent; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/InstantSearch.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/InstantSearch.js index e0f8d4f..e9b6294 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/InstantSearch.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/InstantSearch.js @@ -11,7 +11,7 @@ var _algoliasearchHelper = _interopRequireDefault(require("algoliasearch-helper" var _events = _interopRequireDefault(require("@algolia/events")); -var _index = _interopRequireWildcard(require("../widgets/index/index.js")); +var _index = _interopRequireDefault(require("../widgets/index/index.js")); var _version = _interopRequireDefault(require("./version.js")); @@ -23,10 +23,6 @@ var _createRouterMiddleware = require("../middlewares/createRouterMiddleware.js" var _createMetadataMiddleware = require("../middlewares/createMetadataMiddleware.js"); -function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; } - -function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; } - function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } @@ -61,10 +57,9 @@ var withUsage = (0, _index2.createDocumentationMessageGenerator)({ function defaultCreateURL() { return '#'; -} -/** - * Global options for an InstantSearch instance. - */ +} // this purposely breaks typescript's type inference to ensure it's not used +// as it's used for a default parameter for example +// source: https://github.com/Microsoft/TypeScript/issues/14829#issuecomment-504042546 /** @@ -82,7 +77,7 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { _classCallCheck(this, InstantSearch); - _this = _super.call(this); + _this = _super.call(this); // prevent `render` event listening from causing a warning _defineProperty(_assertThisInitialized(_this), "client", void 0); @@ -108,8 +103,6 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { _defineProperty(_assertThisInitialized(_this), "_searchStalledTimer", void 0); - _defineProperty(_assertThisInitialized(_this), "_isSearchStalled", void 0); - _defineProperty(_assertThisInitialized(_this), "_initialUiState", void 0); _defineProperty(_assertThisInitialized(_this), "_initialResults", void 0); @@ -124,6 +117,10 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { _defineProperty(_assertThisInitialized(_this), "sendEventToInsights", void 0); + _defineProperty(_assertThisInitialized(_this), "status", 'idle'); + + _defineProperty(_assertThisInitialized(_this), "error", undefined); + _defineProperty(_assertThisInitialized(_this), "scheduleSearch", (0, _index2.defer)(function () { if (_this.started) { _this.mainHelper.search(); @@ -131,10 +128,16 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { })); _defineProperty(_assertThisInitialized(_this), "scheduleRender", (0, _index2.defer)(function () { + var shouldResetStatus = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : true; + if (!_this.mainHelper.hasPendingRequests()) { clearTimeout(_this._searchStalledTimer); _this._searchStalledTimer = null; - _this._isSearchStalled = false; + + if (shouldResetStatus) { + _this.status = 'idle'; + _this.error = undefined; + } } _this.mainIndex.render({ @@ -155,6 +158,8 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { }); })); + _this.setMaxListeners(100); + var _options$indexName = options.indexName, indexName = _options$indexName === void 0 ? null : _options$indexName, numberLocale = options.numberLocale, @@ -215,7 +220,6 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { }; _this._stalledSearchDelay = stalledSearchDelay; _this._searchStalledTimer = null; - _this._isSearchStalled = false; _this._createURL = defaultCreateURL; _this._initialUiState = initialUiState; _this._initialResults = null; @@ -244,6 +248,25 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { _createClass(InstantSearch, [{ + key: "_isSearchStalled", + get: + /** + * The status of the search. Can be "idle", "loading", "stalled", or "error". + */ + + /** + * The last returned error from the Search API. + * The error gets cleared when the next valid search response is rendered. + */ + + /** + * @deprecated use `status === 'stalled'` instead + */ + function get() { + process.env.NODE_ENV === 'development' ? (0, _index2.warning)(false, "`InstantSearch._isSearchStalled` is deprecated and will be removed in InstantSearch.js 5.0.\n\nUse `InstantSearch.status === \"stalled\"` instead.") : void 0; + return this.status === 'stalled'; + } + }, { key: "use", value: function use() { var _this2 = this; @@ -255,6 +278,7 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { var newMiddlewareList = middleware.map(function (fn) { var newMiddleware = _objectSpread({ subscribe: _index2.noop, + started: _index2.noop, unsubscribe: _index2.noop, onStateChange: _index2.noop }, fn({ @@ -273,6 +297,7 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { if (this.started) { newMiddlewareList.forEach(function (m) { m.subscribe(); + m.started(); }); } @@ -403,10 +428,18 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { var mainHelper = this.mainHelper || (0, _algoliasearchHelper.default)(this.client, this.indexName); mainHelper.search = function () { - // This solution allows us to keep the exact same API for the users but + _this3.status = 'loading'; // @MAJOR: use scheduleRender here + // For now, widgets don't expect to be rendered at the start of `loading`, + // so it would be a breaking change to add an extra render. We don't have + // these guarantees about the render event, thus emitting it once more + // isn't a breaking change. + + _this3.emit('render'); // This solution allows us to keep the exact same API for the users but // under the hood, we have a different implementation. It should be // completely transparent for the rest of the codebase. Only this module // is impacted. + + return mainHelper.searchOnlyWithDerivedHelpers(); }; @@ -446,13 +479,28 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { mainHelper.on('error', function (_ref4) { var error = _ref4.error; - // If an error is emitted, it is re-thrown by events. In previous versions + + if (!(error instanceof Error)) { + // typescript lies here, error is in some cases { name: string, message: string } + var err = error; + error = Object.keys(err).reduce(function (acc, key) { + acc[key] = err[key]; + return acc; + }, new Error(err.message)); + } // If an error is emitted, it is re-thrown by events. In previous versions // we emitted {error}, which is thrown as: // "Uncaught, unspecified \"error\" event. ([object Object])" // To avoid breaking changes, we make the error available in both // `error` and `error.error` // @MAJOR emit only error + + error.error = error; + _this3.error = error; + _this3.status = 'error'; + + _this3.scheduleRender(false); // This needs to execute last because it throws the error. + _this3.emit('error', error); }); @@ -481,9 +529,17 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { (0, _index2.defer)(function () { _this3.scheduleSearch = originalScheduleSearch; })(); - } else { - this.scheduleSearch(); - } // Keep the previous reference for legacy purpose, some pattern use + } // We only schedule a search when widgets have been added before `start()` + // because there are listeners that can use these results. + // This is especially useful in framework-based flavors that wait for + // dynamically-added widgets to trigger a network request. It avoids + // having to batch this initial network request with the one coming from + // `addWidgets()`. + // Later, we could also skip `index()` widgets and widgets that don't read + // the results, but this is an optimization that has a very low impact for now. + else if (this.mainIndex.getWidgets().length > 0) { + this.scheduleSearch(); + } // Keep the previous reference for legacy purpose, some pattern use // the direct Helper access `search.helper` (e.g multi-index). @@ -491,6 +547,10 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { // to init them directly after add this.started = true; + this.middleware.forEach(function (_ref6) { + var instance = _ref6.instance; + instance.started(); + }); } /** * Removes all widgets without triggering a search afterwards. This is an **EXPERIMENTAL** feature, @@ -517,8 +577,8 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { this.mainHelper.removeAllListeners(); this.mainHelper = null; this.helper = null; - this.middleware.forEach(function (_ref6) { - var instance = _ref6.instance; + this.middleware.forEach(function (_ref7) { + var instance = _ref7.instance; instance.unsubscribe(); }); } @@ -529,15 +589,25 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { if (!this._searchStalledTimer) { this._searchStalledTimer = setTimeout(function () { - _this4._isSearchStalled = true; + _this4.status = 'stalled'; _this4.scheduleRender(); }, this._stalledSearchDelay); } } + /** + * Set the UI state and trigger a search. + * @param uiState The next UI state or a function computing it from the current state + * @param callOnStateChange private parameter used to know if the method is called from a state change + */ + }, { key: "setUiState", value: function setUiState(uiState) { + var _this5 = this; + + var callOnStateChange = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; + if (!this.mainHelper) { throw new Error(withUsage('The `start` method needs to be called before `setUiState`.')); } // We refresh the index UI state to update the local UI state that the @@ -547,25 +617,22 @@ var InstantSearch = /*#__PURE__*/function (_EventEmitter) { this.mainIndex.refreshUiState(); var nextUiState = typeof uiState === 'function' ? uiState(this.mainIndex.getWidgetUiState({})) : uiState; - var setIndexHelperState = function setIndexHelperState(indexWidget) { - var nextIndexUiState = nextUiState[indexWidget.getIndexId()] || {}; - - if (process.env.NODE_ENV === 'development') { - (0, _index2.checkIndexUiState)({ - index: indexWidget, - indexUiState: nextIndexUiState - }); - } + if (this.onStateChange && callOnStateChange) { + this.onStateChange({ + uiState: nextUiState, + setUiState: function setUiState(finalUiState) { + (0, _index2.setIndexHelperState)(typeof finalUiState === 'function' ? finalUiState(nextUiState) : finalUiState, _this5.mainIndex); - indexWidget.getHelper().setState(indexWidget.getWidgetSearchParameters(indexWidget.getHelper().state, { - uiState: nextIndexUiState - })); - indexWidget.getWidgets().filter(_index.isIndexWidget).forEach(setIndexHelperState); - }; + _this5.scheduleSearch(); - setIndexHelperState(this.mainIndex); - this.scheduleSearch(); - this.onInternalStateChange(); + _this5.onInternalStateChange(); + } + }); + } else { + (0, _index2.setIndexHelperState)(nextUiState, this.mainIndex); + this.scheduleSearch(); + this.onInternalStateChange(); + } } }, { key: "getUiState", diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/createHelpers.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/createHelpers.js index e008ce8..6661689 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/createHelpers.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/createHelpers.js @@ -7,6 +7,8 @@ exports.default = hoganHelpers; var _index = require("../helpers/index.js"); +var _formatNumber2 = require("./formatNumber.js"); + function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } @@ -17,7 +19,7 @@ function hoganHelpers(_ref) { var numberLocale = _ref.numberLocale; return { formatNumber: function formatNumber(value, render) { - return Number(render(value)).toLocaleString(numberLocale); + return (0, _formatNumber2.formatNumber)(Number(render(value)), numberLocale); }, highlight: function highlight(options, render) { try { diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/insights/listener.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/insights/listener.js index 7e466a8..85afd4b 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/insights/listener.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/insights/listener.js @@ -11,7 +11,6 @@ var _index = require("../utils/index.js"); var _insights = require("../../helpers/insights.js"); -/** @jsx h */ var findInsightsTarget = function findInsightsTarget(startElement, endElement, validator) { var element = startElement; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/capitalize.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/capitalize.js index b0ca1b5..cef7d0d 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/capitalize.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/capitalize.js @@ -3,11 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.capitalize = capitalize; function capitalize(text) { return text.toString().charAt(0).toUpperCase() + text.toString().slice(1); -} - -var _default = capitalize; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkIndexUiState.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkIndexUiState.js index df399f8..dcf0118 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkIndexUiState.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkIndexUiState.js @@ -5,14 +5,12 @@ Object.defineProperty(exports, "__esModule", { }); exports.checkIndexUiState = checkIndexUiState; -var _capitalize = _interopRequireDefault(require("./capitalize.js")); +var _capitalize = require("./capitalize.js"); var _logger = require("./logger.js"); var _typedObject = require("./typedObject.js"); -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } - function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); } function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } @@ -160,7 +158,7 @@ function checkIndexUiState(_ref) { connectors = _ref8$.connectors, widgets = _ref8$.widgets; - var capitalizedWidget = (0, _capitalize.default)(widgets[0]); + var capitalizedWidget = (0, _capitalize.capitalize)(widgets[0]); var connectorName = connectors[0]; return "const virtual".concat(capitalizedWidget, " = ").concat(connectorName, "(() => null);"); }).join('\n'), "\n\nsearch.addWidgets([\n ").concat(missingWidgets.filter(function (_ref9) { @@ -174,7 +172,7 @@ function checkIndexUiState(_ref) { _stateParameter = _ref12[0], widgets = _ref12[1].widgets; - var capitalizedWidget = (0, _capitalize.default)(widgets[0]); + var capitalizedWidget = (0, _capitalize.capitalize)(widgets[0]); return "virtual".concat(capitalizedWidget, "({ /* ... */ })"); }).join(',\n '), "\n]);\n```\n\nIf you're using custom widgets that do set these query parameters, we recommend using connectors instead.\n\nSee https://www.algolia.com/doc/guides/building-search-ui/widgets/customize-an-existing-widget/js/#customize-the-complete-ui-of-the-widgets")) : void 0; } \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkRendering.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkRendering.js index c1e1a00..3eccfc2 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkRendering.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/checkRendering.js @@ -3,17 +3,12 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.checkRendering = checkRendering; -var _getObjectType = _interopRequireDefault(require("./getObjectType.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _getObjectType = require("./getObjectType.js"); function checkRendering(rendering, usage) { if (rendering === undefined || typeof rendering !== 'function') { - throw new Error("The render function is not valid (received type ".concat((0, _getObjectType.default)(rendering), ").\n\n").concat(usage)); + throw new Error("The render function is not valid (received type ".concat((0, _getObjectType.getObjectType)(rendering), ").\n\n").concat(usage)); } -} - -var _default = checkRendering; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/clearRefinements.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/clearRefinements.js index a75c77d..ed6a794 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/clearRefinements.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/clearRefinements.js @@ -3,16 +3,13 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.clearRefinements = clearRefinements; /** * Clears the refinements of a SearchParameters object based on rules provided. * The included attributes list is applied before the excluded attributes list. If the list * is not provided, this list of all the currently refined attributes is used as included attributes. - * @param {object} $0 parameters - * @param {Helper} $0.helper instance of the Helper - * @param {string[]} [$0.attributesToClear = []] list of parameters to clear - * @returns {SearchParameters} search parameters with refinements cleared + * @returns search parameters with refinements cleared */ function clearRefinements(_ref) { var helper = _ref.helper, @@ -44,7 +41,4 @@ function clearRefinements(_ref) { } return finalState; -} - -var _default = clearRefinements; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/concatHighlightedParts.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/concatHighlightedParts.js index a8f2081..c29a74d 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/concatHighlightedParts.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/concatHighlightedParts.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = concatHighlightedParts; +exports.concatHighlightedParts = concatHighlightedParts; var _escapeHighlight = require("./escape-highlight.js"); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForFacet.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForFacet.js index a128541..cc49f6c 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForFacet.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForFacet.js @@ -5,16 +5,14 @@ Object.defineProperty(exports, "__esModule", { }); exports.createSendEventForFacet = createSendEventForFacet; -var _isFacetRefined = _interopRequireDefault(require("./isFacetRefined.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _isFacetRefined = require("./isFacetRefined.js"); function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); } function createSendEventForFacet(_ref) { var instantSearchInstance = _ref.instantSearchInstance, helper = _ref.helper, - attribute = _ref.attribute, + attr = _ref.attribute, widgetType = _ref.widgetType; var sendEventForFacet = function sendEventForFacet() { @@ -26,11 +24,12 @@ function createSendEventForFacet(_ref) { facetValue = args[1], _args$ = args[2], eventName = _args$ === void 0 ? 'Filter Applied' : _args$; + var attribute = typeof attr === 'string' ? attr : attr(facetValue); if (args.length === 1 && _typeof(args[0]) === 'object') { instantSearchInstance.sendEventToInsights(args[0]); } else if (eventType === 'click' && (args.length === 2 || args.length === 3)) { - if (!(0, _isFacetRefined.default)(helper, attribute, facetValue)) { + if (!(0, _isFacetRefined.isFacetRefined)(helper, attribute, facetValue)) { // send event only when the facet is being checked "ON" instantSearchInstance.sendEventToInsights({ insightsMethod: 'clickedFilters', diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForHits.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForHits.js index 2950ac6..60f90d2 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForHits.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/createSendEventForHits.js @@ -25,7 +25,8 @@ var buildPayloads = function buildPayloads(_ref) { var index = _ref.index, widgetType = _ref.widgetType, methodName = _ref.methodName, - args = _ref.args; + args = _ref.args, + isSearchStalled = _ref.isSearchStalled; // when there's only one argument, that means it's custom if (args.length === 1 && _typeof(args[0]) === 'object') { @@ -72,6 +73,10 @@ var buildPayloads = function buildPayloads(_ref) { }); if (eventType === 'view') { + if (isSearchStalled) { + return []; + } + return hitsChunks.map(function (batch, i) { return { insightsMethod: 'viewedObjectIDs', @@ -142,7 +147,8 @@ function createSendEventForHits(_ref2) { widgetType: widgetType, index: index, methodName: 'sendEvent', - args: args + args: args, + isSearchStalled: instantSearchInstance.status === 'stalled' }); payloads.forEach(function (payload) { return instantSearchInstance.sendEventToInsights(payload); @@ -165,7 +171,8 @@ function createBindEventForHits(_ref3) { widgetType: widgetType, index: index, methodName: 'bindEvent', - args: args + args: args, + isSearchStalled: false }); return payloads.length ? "data-insights-event=".concat((0, _serializer.serializePayload)(payloads)) : ''; }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/defer.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/defer.js index 0108f9d..439a3be 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/defer.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/defer.js @@ -3,10 +3,10 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.defer = defer; var nextMicroTask = Promise.resolve(); -var defer = function defer(callback) { +function defer(callback) { var progress = null; var cancelled = false; @@ -48,7 +48,4 @@ var defer = function defer(callback) { }; return fn; -}; - -var _default = defer; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/detect-insights-client.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/detect-insights-client.js index e6b1d30..7a63aa8 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/detect-insights-client.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/detect-insights-client.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = hasDetectedInsightsClient; +exports.hasDetectedInsightsClient = hasDetectedInsightsClient; var _safelyRunOnBrowser = require("./safelyRunOnBrowser.js"); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/documentation.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/documentation.js index 3b0b701..24fb6d9 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/documentation.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/documentation.js @@ -3,18 +3,17 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.createDocumentationMessageGenerator = exports.createDocumentationLink = void 0; +exports.createDocumentationLink = createDocumentationLink; +exports.createDocumentationMessageGenerator = createDocumentationMessageGenerator; -var createDocumentationLink = function createDocumentationLink(_ref) { +function createDocumentationLink(_ref) { var name = _ref.name, _ref$connector = _ref.connector, connector = _ref$connector === void 0 ? false : _ref$connector; return ['https://www.algolia.com/doc/api-reference/widgets/', name, '/js/', connector ? '#connector' : ''].join(''); -}; - -exports.createDocumentationLink = createDocumentationLink; +} -var createDocumentationMessageGenerator = function createDocumentationMessageGenerator() { +function createDocumentationMessageGenerator() { for (var _len = arguments.length, widgets = new Array(_len), _key = 0; _key < _len; _key++) { widgets[_key] = arguments[_key]; } @@ -25,6 +24,4 @@ var createDocumentationMessageGenerator = function createDocumentationMessageGen return function (message) { return [message, "See documentation: ".concat(links)].filter(Boolean).join('\n\n'); }; -}; - -exports.createDocumentationMessageGenerator = createDocumentationMessageGenerator; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/escape-highlight.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/escape-highlight.js index 4b0af88..ea08215 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/escape-highlight.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/escape-highlight.js @@ -7,11 +7,9 @@ exports.escapeHits = escapeHits; exports.escapeFacets = escapeFacets; exports.TAG_REPLACEMENT = exports.TAG_PLACEHOLDER = void 0; -var _escape = _interopRequireDefault(require("./escape.js")); +var _escapeHtml = require("./escape-html.js"); -var _isPlainObject = _interopRequireDefault(require("./isPlainObject.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _isPlainObject = require("./isPlainObject.js"); function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } @@ -33,11 +31,11 @@ var TAG_REPLACEMENT = { exports.TAG_REPLACEMENT = TAG_REPLACEMENT; function replaceTagsAndEscape(value) { - return (0, _escape.default)(value).replace(new RegExp(TAG_PLACEHOLDER.highlightPreTag, 'g'), TAG_REPLACEMENT.highlightPreTag).replace(new RegExp(TAG_PLACEHOLDER.highlightPostTag, 'g'), TAG_REPLACEMENT.highlightPostTag); + return (0, _escapeHtml.escape)(value).replace(new RegExp(TAG_PLACEHOLDER.highlightPreTag, 'g'), TAG_REPLACEMENT.highlightPreTag).replace(new RegExp(TAG_PLACEHOLDER.highlightPostTag, 'g'), TAG_REPLACEMENT.highlightPostTag); } function recursiveEscape(input) { - if ((0, _isPlainObject.default)(input) && typeof input.value !== 'string') { + if ((0, _isPlainObject.isPlainObject)(input) && typeof input.value !== 'string') { return Object.keys(input).reduce(function (acc, key) { return _objectSpread(_objectSpread({}, acc), {}, _defineProperty({}, key, recursiveEscape(input[key]))); }, {}); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/find.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/find.js index c33e35a..434b685 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/find.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/find.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.find = find; // We aren't using the native `Array.prototype.find` because the refactor away from Lodash is not // published as a major version. @@ -23,7 +23,4 @@ function find(items, predicate) { } return undefined; -} - -var _default = find; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/findIndex.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/findIndex.js index 7f27f0e..ee2c073 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/findIndex.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/findIndex.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.findIndex = findIndex; // We aren't using the native `Array.prototype.findIndex` because the refactor away from Lodash is not // published as a major version. @@ -23,7 +23,4 @@ function findIndex(array, comparator) { } return -1; -} - -var _default = findIndex; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getContainerNode.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getContainerNode.js index ed18bfe..f295c92 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getContainerNode.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getContainerNode.js @@ -3,11 +3,9 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.getContainerNode = getContainerNode; -var _isDomElement = _interopRequireDefault(require("./isDomElement.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _isDomElement = require("./isDomElement.js"); /** * Return the container. If it's a string, it is considered a @@ -22,7 +20,7 @@ function getContainerNode(selectorOrHTMLElement) { var isSelectorString = typeof selectorOrHTMLElement === 'string'; var domElement = isSelectorString ? document.querySelector(selectorOrHTMLElement) : selectorOrHTMLElement; - if (!(0, _isDomElement.default)(domElement)) { + if (!(0, _isDomElement.isDomElement)(domElement)) { var errorMessage = 'Container must be `string` or `HTMLElement`.'; if (isSelectorString) { @@ -33,7 +31,4 @@ function getContainerNode(selectorOrHTMLElement) { } return domElement; -} - -var _default = getContainerNode; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightFromSiblings.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightFromSiblings.js index 85f032c..f25771d 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightFromSiblings.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightFromSiblings.js @@ -3,11 +3,9 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = getHighlightFromSiblings; +exports.getHighlightFromSiblings = getHighlightFromSiblings; -var _unescape = _interopRequireDefault(require("./unescape.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _escapeHtml = require("./escape-html.js"); var hasAlphanumeric = new RegExp(/\w/i); @@ -18,7 +16,7 @@ function getHighlightFromSiblings(parts, i) { var isNextHighlighted = ((_parts = parts[i + 1]) === null || _parts === void 0 ? void 0 : _parts.isHighlighted) || true; var isPreviousHighlighted = ((_parts2 = parts[i - 1]) === null || _parts2 === void 0 ? void 0 : _parts2.isHighlighted) || true; - if (!hasAlphanumeric.test((0, _unescape.default)(current.value)) && isPreviousHighlighted === isNextHighlighted) { + if (!hasAlphanumeric.test((0, _escapeHtml.unescape)(current.value)) && isPreviousHighlighted === isNextHighlighted) { return isPreviousHighlighted; } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightedParts.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightedParts.js index 3423cc4..006eea1 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightedParts.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getHighlightedParts.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = getHighlightedParts; +exports.getHighlightedParts = getHighlightedParts; var _escapeHighlight = require("./escape-highlight.js"); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getObjectType.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getObjectType.js index 1137883..1058664 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getObjectType.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getObjectType.js @@ -3,11 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.getObjectType = getObjectType; function getObjectType(object) { return Object.prototype.toString.call(object).slice(8, -1); -} - -var _default = getObjectType; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getPropertyByPath.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getPropertyByPath.js index 2380c8e..d41c3c9 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getPropertyByPath.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getPropertyByPath.js @@ -3,14 +3,11 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.getPropertyByPath = getPropertyByPath; function getPropertyByPath(object, path) { var parts = Array.isArray(path) ? path : path.split('.'); return parts.reduce(function (current, key) { return current && current[key]; }, object); -} - -var _default = getPropertyByPath; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getRefinements.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getRefinements.js index ffabc00..e369e27 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getRefinements.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/getRefinements.js @@ -3,14 +3,12 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = getRefinements; +exports.getRefinements = getRefinements; -var _find = _interopRequireDefault(require("./find.js")); +var _find = require("./find.js"); var _escapeFacetValue = require("./escapeFacetValue.js"); -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } - function getRefinement(state, type, attribute, name) { var resultsFacets = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : []; var res = { @@ -19,7 +17,7 @@ function getRefinement(state, type, attribute, name) { name: name, escapedValue: (0, _escapeFacetValue.escapeFacetValue)(name) }; - var facet = (0, _find.default)(resultsFacets, function (resultsFacet) { + var facet = (0, _find.find)(resultsFacets, function (resultsFacet) { return resultsFacet.name === attribute; }); var count; @@ -36,7 +34,7 @@ function getRefinement(state, type, attribute, name) { }; var _loop = function _loop(i) { - facet = facet && facet.data && (0, _find.default)(Object.keys(facet.data).map(getFacetRefinement(facet.data)), function (refinement) { + facet = facet && facet.data && (0, _find.find)(Object.keys(facet.data).map(getFacetRefinement(facet.data)), function (refinement) { return refinement.name === nameParts[i]; }); }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/index.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/index.js index b220f3c..fc05f19 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/index.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/index.js @@ -3,496 +3,601 @@ Object.defineProperty(exports, "__esModule", { value: true }); -var _exportNames = { - capitalize: true, - defer: true, - isDomElement: true, - getContainerNode: true, - isSpecialClick: true, - prepareTemplateProps: true, - renderTemplate: true, - getRefinements: true, - clearRefinements: true, - escapeFacetValue: true, - unescapeFacetValue: true, - checkRendering: true, - checkIndexUiState: true, - getPropertyByPath: true, - getObjectType: true, - noop: true, - isFiniteNumber: true, - isPlainObject: true, - uniq: true, - range: true, - isEqual: true, - escape: true, - unescape: true, - concatHighlightedParts: true, - getHighlightedParts: true, - getHighlightFromSiblings: true, - reverseHighlightedParts: true, - find: true, - findIndex: true, - mergeSearchParameters: true, - resolveSearchParameters: true, - toArray: true, - warning: true, - deprecate: true, - escapeHits: true, - TAG_PLACEHOLDER: true, - TAG_REPLACEMENT: true, - escapeFacets: true, - createDocumentationLink: true, - createDocumentationMessageGenerator: true, - aroundLatLngToPosition: true, - insideBoundingBoxToBoundingBox: true, - addAbsolutePosition: true, - addQueryID: true, - isFacetRefined: true, - getAppIdAndApiKey: true, - convertNumericRefinementsToFilters: true, - createConcurrentSafePromise: true, - debounce: true, - serializePayload: true, - deserializePayload: true, - getWidgetAttribute: true, - safelyRunOnBrowser: true -}; -Object.defineProperty(exports, "capitalize", { - enumerable: true, - get: function get() { - return _capitalize.default; - } -}); -Object.defineProperty(exports, "defer", { - enumerable: true, - get: function get() { - return _defer.default; - } -}); -Object.defineProperty(exports, "isDomElement", { - enumerable: true, - get: function get() { - return _isDomElement.default; - } -}); -Object.defineProperty(exports, "getContainerNode", { - enumerable: true, - get: function get() { - return _getContainerNode.default; - } -}); -Object.defineProperty(exports, "isSpecialClick", { - enumerable: true, - get: function get() { - return _isSpecialClick.default; - } -}); -Object.defineProperty(exports, "prepareTemplateProps", { - enumerable: true, - get: function get() { - return _prepareTemplateProps.default; - } -}); -Object.defineProperty(exports, "renderTemplate", { - enumerable: true, - get: function get() { - return _renderTemplate.default; - } -}); -Object.defineProperty(exports, "getRefinements", { - enumerable: true, - get: function get() { - return _getRefinements.default; - } -}); -Object.defineProperty(exports, "clearRefinements", { - enumerable: true, - get: function get() { - return _clearRefinements.default; - } -}); -Object.defineProperty(exports, "escapeFacetValue", { - enumerable: true, - get: function get() { - return _escapeFacetValue.escapeFacetValue; - } -}); -Object.defineProperty(exports, "unescapeFacetValue", { - enumerable: true, - get: function get() { - return _escapeFacetValue.unescapeFacetValue; - } -}); -Object.defineProperty(exports, "checkRendering", { - enumerable: true, - get: function get() { - return _checkRendering.default; - } -}); -Object.defineProperty(exports, "checkIndexUiState", { - enumerable: true, - get: function get() { - return _checkIndexUiState.checkIndexUiState; - } -}); -Object.defineProperty(exports, "getPropertyByPath", { - enumerable: true, - get: function get() { - return _getPropertyByPath.default; - } -}); -Object.defineProperty(exports, "getObjectType", { - enumerable: true, - get: function get() { - return _getObjectType.default; - } -}); -Object.defineProperty(exports, "noop", { - enumerable: true, - get: function get() { - return _noop.default; - } -}); -Object.defineProperty(exports, "isFiniteNumber", { - enumerable: true, - get: function get() { - return _isFiniteNumber.default; - } -}); -Object.defineProperty(exports, "isPlainObject", { - enumerable: true, - get: function get() { - return _isPlainObject.default; - } -}); -Object.defineProperty(exports, "uniq", { - enumerable: true, - get: function get() { - return _uniq.default; - } -}); -Object.defineProperty(exports, "range", { - enumerable: true, - get: function get() { - return _range.default; - } -}); -Object.defineProperty(exports, "isEqual", { - enumerable: true, - get: function get() { - return _isEqual.default; - } -}); -Object.defineProperty(exports, "escape", { - enumerable: true, - get: function get() { - return _escape.default; - } -}); -Object.defineProperty(exports, "unescape", { - enumerable: true, - get: function get() { - return _unescape.default; - } -}); -Object.defineProperty(exports, "concatHighlightedParts", { - enumerable: true, - get: function get() { - return _concatHighlightedParts.default; - } -}); -Object.defineProperty(exports, "getHighlightedParts", { - enumerable: true, - get: function get() { - return _getHighlightedParts.default; - } -}); -Object.defineProperty(exports, "getHighlightFromSiblings", { - enumerable: true, - get: function get() { - return _getHighlightFromSiblings.default; - } -}); -Object.defineProperty(exports, "reverseHighlightedParts", { - enumerable: true, - get: function get() { - return _reverseHighlightedParts.default; - } -}); -Object.defineProperty(exports, "find", { - enumerable: true, - get: function get() { - return _find.default; - } -}); -Object.defineProperty(exports, "findIndex", { - enumerable: true, - get: function get() { - return _findIndex.default; - } -}); -Object.defineProperty(exports, "mergeSearchParameters", { - enumerable: true, - get: function get() { - return _mergeSearchParameters.default; - } -}); -Object.defineProperty(exports, "resolveSearchParameters", { - enumerable: true, - get: function get() { - return _resolveSearchParameters.default; - } -}); -Object.defineProperty(exports, "toArray", { - enumerable: true, - get: function get() { - return _toArray.default; - } -}); -Object.defineProperty(exports, "warning", { - enumerable: true, - get: function get() { - return _logger.warning; - } -}); -Object.defineProperty(exports, "deprecate", { - enumerable: true, - get: function get() { - return _logger.deprecate; - } -}); -Object.defineProperty(exports, "escapeHits", { - enumerable: true, - get: function get() { - return _escapeHighlight.escapeHits; - } -}); -Object.defineProperty(exports, "TAG_PLACEHOLDER", { - enumerable: true, - get: function get() { - return _escapeHighlight.TAG_PLACEHOLDER; - } -}); -Object.defineProperty(exports, "TAG_REPLACEMENT", { - enumerable: true, - get: function get() { - return _escapeHighlight.TAG_REPLACEMENT; - } -}); -Object.defineProperty(exports, "escapeFacets", { - enumerable: true, - get: function get() { - return _escapeHighlight.escapeFacets; - } -}); -Object.defineProperty(exports, "createDocumentationLink", { - enumerable: true, - get: function get() { - return _documentation.createDocumentationLink; - } -}); -Object.defineProperty(exports, "createDocumentationMessageGenerator", { - enumerable: true, - get: function get() { - return _documentation.createDocumentationMessageGenerator; - } -}); -Object.defineProperty(exports, "aroundLatLngToPosition", { - enumerable: true, - get: function get() { - return _geoSearch.aroundLatLngToPosition; - } -}); -Object.defineProperty(exports, "insideBoundingBoxToBoundingBox", { - enumerable: true, - get: function get() { - return _geoSearch.insideBoundingBoxToBoundingBox; - } -}); -Object.defineProperty(exports, "addAbsolutePosition", { - enumerable: true, - get: function get() { - return _hitsAbsolutePosition.addAbsolutePosition; - } -}); -Object.defineProperty(exports, "addQueryID", { - enumerable: true, - get: function get() { - return _hitsQueryId.addQueryID; - } -}); -Object.defineProperty(exports, "isFacetRefined", { - enumerable: true, - get: function get() { - return _isFacetRefined.default; - } -}); -Object.defineProperty(exports, "getAppIdAndApiKey", { - enumerable: true, - get: function get() { - return _getAppIdAndApiKey.getAppIdAndApiKey; - } -}); -Object.defineProperty(exports, "convertNumericRefinementsToFilters", { - enumerable: true, - get: function get() { - return _convertNumericRefinementsToFilters.convertNumericRefinementsToFilters; - } -}); -Object.defineProperty(exports, "createConcurrentSafePromise", { - enumerable: true, - get: function get() { - return _createConcurrentSafePromise.createConcurrentSafePromise; - } -}); -Object.defineProperty(exports, "debounce", { - enumerable: true, - get: function get() { - return _debounce.debounce; - } -}); -Object.defineProperty(exports, "serializePayload", { - enumerable: true, - get: function get() { - return _serializer.serializePayload; - } -}); -Object.defineProperty(exports, "deserializePayload", { - enumerable: true, - get: function get() { - return _serializer.deserializePayload; - } -}); -Object.defineProperty(exports, "getWidgetAttribute", { - enumerable: true, - get: function get() { - return _getWidgetAttribute.getWidgetAttribute; - } -}); -Object.defineProperty(exports, "safelyRunOnBrowser", { - enumerable: true, - get: function get() { - return _safelyRunOnBrowser.safelyRunOnBrowser; - } -}); - -var _capitalize = _interopRequireDefault(require("./capitalize.js")); - -var _defer = _interopRequireDefault(require("./defer.js")); - -var _isDomElement = _interopRequireDefault(require("./isDomElement.js")); - -var _getContainerNode = _interopRequireDefault(require("./getContainerNode.js")); - -var _isSpecialClick = _interopRequireDefault(require("./isSpecialClick.js")); - -var _prepareTemplateProps = _interopRequireDefault(require("./prepareTemplateProps.js")); - -var _renderTemplate = _interopRequireDefault(require("./renderTemplate.js")); - -var _getRefinements = _interopRequireDefault(require("./getRefinements.js")); - -var _clearRefinements = _interopRequireDefault(require("./clearRefinements.js")); -var _escapeFacetValue = require("./escapeFacetValue.js"); +var _capitalize = require("./capitalize.js"); -var _checkRendering = _interopRequireDefault(require("./checkRendering.js")); +Object.keys(_capitalize).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _capitalize[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _capitalize[key]; + } + }); +}); var _checkIndexUiState = require("./checkIndexUiState.js"); -var _getPropertyByPath = _interopRequireDefault(require("./getPropertyByPath.js")); +Object.keys(_checkIndexUiState).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _checkIndexUiState[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _checkIndexUiState[key]; + } + }); +}); -var _getObjectType = _interopRequireDefault(require("./getObjectType.js")); +var _checkRendering = require("./checkRendering.js"); -var _noop = _interopRequireDefault(require("./noop.js")); +Object.keys(_checkRendering).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _checkRendering[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _checkRendering[key]; + } + }); +}); -var _isFiniteNumber = _interopRequireDefault(require("./isFiniteNumber.js")); +var _clearRefinements = require("./clearRefinements.js"); -var _isPlainObject = _interopRequireDefault(require("./isPlainObject.js")); +Object.keys(_clearRefinements).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _clearRefinements[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _clearRefinements[key]; + } + }); +}); -var _uniq = _interopRequireDefault(require("./uniq.js")); +var _concatHighlightedParts = require("./concatHighlightedParts.js"); -var _range = _interopRequireDefault(require("./range.js")); +Object.keys(_concatHighlightedParts).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _concatHighlightedParts[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _concatHighlightedParts[key]; + } + }); +}); -var _isEqual = _interopRequireDefault(require("./isEqual.js")); +var _createConcurrentSafePromise = require("./createConcurrentSafePromise.js"); -var _escape = _interopRequireDefault(require("./escape.js")); +Object.keys(_createConcurrentSafePromise).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _createConcurrentSafePromise[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _createConcurrentSafePromise[key]; + } + }); +}); -var _unescape = _interopRequireDefault(require("./unescape.js")); +var _createSendEventForFacet = require("./createSendEventForFacet.js"); -var _concatHighlightedParts = _interopRequireDefault(require("./concatHighlightedParts.js")); +Object.keys(_createSendEventForFacet).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _createSendEventForFacet[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _createSendEventForFacet[key]; + } + }); +}); -var _getHighlightedParts = _interopRequireDefault(require("./getHighlightedParts.js")); +var _createSendEventForHits = require("./createSendEventForHits.js"); -var _getHighlightFromSiblings = _interopRequireDefault(require("./getHighlightFromSiblings.js")); +Object.keys(_createSendEventForHits).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _createSendEventForHits[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _createSendEventForHits[key]; + } + }); +}); -var _reverseHighlightedParts = _interopRequireDefault(require("./reverseHighlightedParts.js")); +var _setIndexHelperState = require("./setIndexHelperState.js"); -var _find = _interopRequireDefault(require("./find.js")); +Object.keys(_setIndexHelperState).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _setIndexHelperState[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _setIndexHelperState[key]; + } + }); +}); -var _findIndex = _interopRequireDefault(require("./findIndex.js")); +var _isIndexWidget = require("./isIndexWidget.js"); -var _mergeSearchParameters = _interopRequireDefault(require("./mergeSearchParameters.js")); +Object.keys(_isIndexWidget).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _isIndexWidget[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _isIndexWidget[key]; + } + }); +}); -var _resolveSearchParameters = _interopRequireDefault(require("./resolveSearchParameters.js")); +var _debounce = require("./debounce.js"); -var _toArray = _interopRequireDefault(require("./toArray.js")); +Object.keys(_debounce).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _debounce[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _debounce[key]; + } + }); +}); -var _logger = require("./logger.js"); +var _defer = require("./defer.js"); -var _escapeHighlight = require("./escape-highlight.js"); +Object.keys(_defer).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _defer[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _defer[key]; + } + }); +}); var _documentation = require("./documentation.js"); +Object.keys(_documentation).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _documentation[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _documentation[key]; + } + }); +}); + +var _escapeHighlight = require("./escape-highlight.js"); + +Object.keys(_escapeHighlight).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _escapeHighlight[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _escapeHighlight[key]; + } + }); +}); + +var _escapeHtml = require("./escape-html.js"); + +Object.keys(_escapeHtml).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _escapeHtml[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _escapeHtml[key]; + } + }); +}); + +var _escapeFacetValue = require("./escapeFacetValue.js"); + +Object.keys(_escapeFacetValue).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _escapeFacetValue[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _escapeFacetValue[key]; + } + }); +}); + +var _find = require("./find.js"); + +Object.keys(_find).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _find[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _find[key]; + } + }); +}); + +var _findIndex = require("./findIndex.js"); + +Object.keys(_findIndex).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _findIndex[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _findIndex[key]; + } + }); +}); + var _geoSearch = require("./geo-search.js"); +Object.keys(_geoSearch).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _geoSearch[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _geoSearch[key]; + } + }); +}); + +var _getAppIdAndApiKey = require("./getAppIdAndApiKey.js"); + +Object.keys(_getAppIdAndApiKey).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getAppIdAndApiKey[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getAppIdAndApiKey[key]; + } + }); +}); + +var _getContainerNode = require("./getContainerNode.js"); + +Object.keys(_getContainerNode).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getContainerNode[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getContainerNode[key]; + } + }); +}); + +var _getHighlightedParts = require("./getHighlightedParts.js"); + +Object.keys(_getHighlightedParts).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getHighlightedParts[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getHighlightedParts[key]; + } + }); +}); + +var _getHighlightFromSiblings = require("./getHighlightFromSiblings.js"); + +Object.keys(_getHighlightFromSiblings).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getHighlightFromSiblings[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getHighlightFromSiblings[key]; + } + }); +}); + +var _getObjectType = require("./getObjectType.js"); + +Object.keys(_getObjectType).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getObjectType[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getObjectType[key]; + } + }); +}); + +var _getPropertyByPath = require("./getPropertyByPath.js"); + +Object.keys(_getPropertyByPath).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getPropertyByPath[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getPropertyByPath[key]; + } + }); +}); + +var _getRefinements = require("./getRefinements.js"); + +Object.keys(_getRefinements).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getRefinements[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getRefinements[key]; + } + }); +}); + +var _getWidgetAttribute = require("./getWidgetAttribute.js"); + +Object.keys(_getWidgetAttribute).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _getWidgetAttribute[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _getWidgetAttribute[key]; + } + }); +}); + var _hitsAbsolutePosition = require("./hits-absolute-position.js"); +Object.keys(_hitsAbsolutePosition).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _hitsAbsolutePosition[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _hitsAbsolutePosition[key]; + } + }); +}); + var _hitsQueryId = require("./hits-query-id.js"); -var _isFacetRefined = _interopRequireDefault(require("./isFacetRefined.js")); +Object.keys(_hitsQueryId).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _hitsQueryId[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _hitsQueryId[key]; + } + }); +}); + +var _isDomElement = require("./isDomElement.js"); -var _createSendEventForFacet = require("./createSendEventForFacet.js"); +Object.keys(_isDomElement).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _isDomElement[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _isDomElement[key]; + } + }); +}); -Object.keys(_createSendEventForFacet).forEach(function (key) { +var _isEqual = require("./isEqual.js"); + +Object.keys(_isEqual).forEach(function (key) { if (key === "default" || key === "__esModule") return; - if (Object.prototype.hasOwnProperty.call(_exportNames, key)) return; - if (key in exports && exports[key] === _createSendEventForFacet[key]) return; + if (key in exports && exports[key] === _isEqual[key]) return; Object.defineProperty(exports, key, { enumerable: true, get: function get() { - return _createSendEventForFacet[key]; + return _isEqual[key]; } }); }); -var _createSendEventForHits = require("./createSendEventForHits.js"); +var _isFacetRefined = require("./isFacetRefined.js"); -Object.keys(_createSendEventForHits).forEach(function (key) { +Object.keys(_isFacetRefined).forEach(function (key) { if (key === "default" || key === "__esModule") return; - if (Object.prototype.hasOwnProperty.call(_exportNames, key)) return; - if (key in exports && exports[key] === _createSendEventForHits[key]) return; + if (key in exports && exports[key] === _isFacetRefined[key]) return; Object.defineProperty(exports, key, { enumerable: true, get: function get() { - return _createSendEventForHits[key]; + return _isFacetRefined[key]; } }); }); -var _getAppIdAndApiKey = require("./getAppIdAndApiKey.js"); +var _isFiniteNumber = require("./isFiniteNumber.js"); -var _convertNumericRefinementsToFilters = require("./convertNumericRefinementsToFilters.js"); +Object.keys(_isFiniteNumber).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _isFiniteNumber[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _isFiniteNumber[key]; + } + }); +}); -var _createConcurrentSafePromise = require("./createConcurrentSafePromise.js"); +var _isPlainObject = require("./isPlainObject.js"); -var _debounce = require("./debounce.js"); +Object.keys(_isPlainObject).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _isPlainObject[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _isPlainObject[key]; + } + }); +}); -var _serializer = require("./serializer.js"); +var _isSpecialClick = require("./isSpecialClick.js"); -var _getWidgetAttribute = require("./getWidgetAttribute.js"); +Object.keys(_isSpecialClick).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _isSpecialClick[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _isSpecialClick[key]; + } + }); +}); + +var _logger = require("./logger.js"); + +Object.keys(_logger).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _logger[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _logger[key]; + } + }); +}); + +var _mergeSearchParameters = require("./mergeSearchParameters.js"); + +Object.keys(_mergeSearchParameters).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _mergeSearchParameters[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _mergeSearchParameters[key]; + } + }); +}); + +var _noop = require("./noop.js"); + +Object.keys(_noop).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _noop[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _noop[key]; + } + }); +}); + +var _range = require("./range.js"); + +Object.keys(_range).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _range[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _range[key]; + } + }); +}); + +var _renderArgs = require("./render-args.js"); + +Object.keys(_renderArgs).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _renderArgs[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _renderArgs[key]; + } + }); +}); + +var _resolveSearchParameters = require("./resolveSearchParameters.js"); + +Object.keys(_resolveSearchParameters).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _resolveSearchParameters[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _resolveSearchParameters[key]; + } + }); +}); + +var _reverseHighlightedParts = require("./reverseHighlightedParts.js"); + +Object.keys(_reverseHighlightedParts).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _reverseHighlightedParts[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _reverseHighlightedParts[key]; + } + }); +}); var _safelyRunOnBrowser = require("./safelyRunOnBrowser.js"); -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } \ No newline at end of file +Object.keys(_safelyRunOnBrowser).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _safelyRunOnBrowser[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _safelyRunOnBrowser[key]; + } + }); +}); + +var _serializer = require("./serializer.js"); + +Object.keys(_serializer).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _serializer[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _serializer[key]; + } + }); +}); + +var _toArray = require("./toArray.js"); + +Object.keys(_toArray).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _toArray[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _toArray[key]; + } + }); +}); + +var _uniq = require("./uniq.js"); + +Object.keys(_uniq).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _uniq[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _uniq[key]; + } + }); +}); \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isDomElement.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isDomElement.js index 5e2685a..39eb0e8 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isDomElement.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isDomElement.js @@ -3,11 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.isDomElement = isDomElement; function isDomElement(object) { return object instanceof HTMLElement || Boolean(object) && object.nodeType > 0; -} - -var _default = isDomElement; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isEqual.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isEqual.js index 4347849..e932184 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isEqual.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isEqual.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.isEqual = isEqual; function isPrimitive(obj) { return obj !== Object(obj); @@ -35,7 +35,4 @@ function isEqual(first, second) { } return true; -} - -var _default = isEqual; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFacetRefined.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFacetRefined.js index 49f271a..ea3d6f3 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFacetRefined.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFacetRefined.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = isFacetRefined; +exports.isFacetRefined = isFacetRefined; function isFacetRefined(helper, facet, value) { if (helper.state.isHierarchicalFacet(facet)) { diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFiniteNumber.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFiniteNumber.js index be982c5..a36ec95 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFiniteNumber.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isFiniteNumber.js @@ -3,14 +3,12 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.isFiniteNumber = isFiniteNumber; // This is the `Number.isFinite()` polyfill recommended by MDN. // We do not provide any tests for this function. // See: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isFinite#Polyfill +// @MAJOR Replace with the native `Number.isFinite` method function isFiniteNumber(value) { return typeof value === 'number' && isFinite(value); -} - -var _default = isFiniteNumber; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isPlainObject.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isPlainObject.js index ecc7a67..0608016 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isPlainObject.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isPlainObject.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.isPlainObject = isPlainObject; function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); } @@ -46,7 +46,4 @@ function isPlainObject(value) { } return Object.getPrototypeOf(value) === proto; -} - -var _default = isPlainObject; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isSpecialClick.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isSpecialClick.js index c9f75fb..280af83 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isSpecialClick.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/isSpecialClick.js @@ -3,12 +3,9 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.isSpecialClick = isSpecialClick; function isSpecialClick(event) { var isMiddleClick = event.button === 1; return isMiddleClick || event.altKey || event.ctrlKey || event.metaKey || event.shiftKey; -} - -var _default = isSpecialClick; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/logger.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/logger.js index cebd4c7..6977fd3 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/logger.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/logger.js @@ -5,9 +5,7 @@ Object.defineProperty(exports, "__esModule", { }); exports.warning = exports.deprecate = exports.warn = void 0; -var _noop = _interopRequireDefault(require("./noop.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _noop = require("./noop.js"); /** * Logs a warning when this function is called, in development environment only. @@ -22,14 +20,14 @@ var deprecate = function deprecate(fn, message) { exports.deprecate = deprecate; -var warn = _noop.default; +var warn = _noop.noop; /** * Logs a warning if the condition is not met. * This is used to log issues in development environment only. */ exports.warn = warn; -var _warning = _noop.default; +var _warning = _noop.noop; exports.warning = _warning; if (process.env.NODE_ENV === 'development') { diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/mergeSearchParameters.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/mergeSearchParameters.js index df646e0..2985462 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/mergeSearchParameters.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/mergeSearchParameters.js @@ -3,13 +3,11 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.mergeSearchParameters = void 0; -var _findIndex = _interopRequireDefault(require("./findIndex.js")); +var _findIndex = require("./findIndex.js"); -var _uniq = _interopRequireDefault(require("./uniq.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _uniq = require("./uniq.js"); function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } @@ -53,7 +51,7 @@ var mergeDisjunctiveFacets = function mergeDisjunctiveFacets(left, right) { var mergeHierarchicalFacets = function mergeHierarchicalFacets(left, right) { return left.setQueryParameters({ hierarchicalFacets: right.hierarchicalFacets.reduce(function (facets, facet) { - var index = (0, _findIndex.default)(facets, function (_) { + var index = (0, _findIndex.findIndex)(facets, function (_) { return _.name === facet.name; }); @@ -106,7 +104,7 @@ var mergeHierarchicalFacetsRefinements = function mergeHierarchicalFacetsRefinem }; var mergeRuleContexts = function mergeRuleContexts(left, right) { - var ruleContexts = (0, _uniq.default)([].concat(left.ruleContexts).concat(right.ruleContexts).filter(Boolean)); + var ruleContexts = (0, _uniq.uniq)([].concat(left.ruleContexts).concat(right.ruleContexts).filter(Boolean)); if (ruleContexts.length > 0) { return left.setQueryParameters({ @@ -117,7 +115,7 @@ var mergeRuleContexts = function mergeRuleContexts(left, right) { return left; }; -var merge = function merge() { +var mergeSearchParameters = function mergeSearchParameters() { for (var _len = arguments.length, parameters = new Array(_len), _key = 0; _key < _len; _key++) { parameters[_key] = arguments[_key]; } @@ -137,5 +135,4 @@ var merge = function merge() { }); }; -var _default = merge; -exports.default = _default; \ No newline at end of file +exports.mergeSearchParameters = mergeSearchParameters; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/noop.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/noop.js index 6da6ee9..13ec00d 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/noop.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/noop.js @@ -3,9 +3,6 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.noop = noop; -function noop() {} - -var _default = noop; -exports.default = _default; \ No newline at end of file +function noop() {} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/range.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/range.js index a01a6f4..bd4c32c 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/range.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/range.js @@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.range = range; function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); } @@ -34,7 +34,4 @@ function range(_ref) { return _toConsumableArray(Array(arrayLength)).map(function (_, current) { return start + current * limitStep; }); -} - -var _default = range; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/resolveSearchParameters.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/resolveSearchParameters.js index bfdc556..8b2b839 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/resolveSearchParameters.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/resolveSearchParameters.js @@ -3,9 +3,9 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.resolveSearchParameters = resolveSearchParameters; -var resolveSearchParameters = function resolveSearchParameters(current) { +function resolveSearchParameters(current) { var parent = current.getParent(); var states = [current.getHelper().state]; @@ -15,7 +15,4 @@ var resolveSearchParameters = function resolveSearchParameters(current) { } return states; -}; - -var _default = resolveSearchParameters; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/reverseHighlightedParts.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/reverseHighlightedParts.js index 5c2a65b..a57d0f4 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/reverseHighlightedParts.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/reverseHighlightedParts.js @@ -3,11 +3,9 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = reverseHighlightedParts; +exports.reverseHighlightedParts = reverseHighlightedParts; -var _getHighlightFromSiblings = _interopRequireDefault(require("./getHighlightFromSiblings.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } +var _getHighlightFromSiblings = require("./getHighlightFromSiblings.js"); function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } @@ -28,7 +26,7 @@ function reverseHighlightedParts(parts) { return parts.map(function (part, i) { return _objectSpread(_objectSpread({}, part), {}, { - isHighlighted: !(0, _getHighlightFromSiblings.default)(parts, i) + isHighlighted: !(0, _getHighlightFromSiblings.getHighlightFromSiblings)(parts, i) }); }); } \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/toArray.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/toArray.js index 1d7a954..be4416e 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/toArray.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/toArray.js @@ -3,11 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.toArray = toArray; function toArray(value) { return Array.isArray(value) ? value : [value]; -} - -var _default = toArray; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/uniq.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/uniq.js index 16ef100..ddd189c 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/uniq.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/utils/uniq.js @@ -3,13 +3,10 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.default = void 0; +exports.uniq = uniq; function uniq(array) { return array.filter(function (value, index, self) { return self.indexOf(value) === index; }); -} - -var _default = uniq; -exports.default = _default; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/version.js b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/version.js index 5da26fd..6d51e15 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/lib/version.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/lib/version.js @@ -4,5 +4,5 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; -var _default = '4.40.5'; +var _default = '4.49.0'; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createInsightsMiddleware.js b/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createInsightsMiddleware.js index fcced0a..8e78c31 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createInsightsMiddleware.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createInsightsMiddleware.js @@ -3,16 +3,12 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.createInsightsMiddleware = void 0; +exports.createInsightsMiddleware = createInsightsMiddleware; var _index = require("../helpers/index.js"); var _index2 = require("../lib/utils/index.js"); -var _connectConfigure = _interopRequireDefault(require("../connectors/configure/connectConfigure.js")); - -function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } - function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } @@ -31,7 +27,7 @@ function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !( function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; } -var createInsightsMiddleware = function createInsightsMiddleware(props) { +function createInsightsMiddleware(props) { var _ref = props || {}, _insightsClient = _ref.insightsClient, insightsInitParams = _ref.insightsInitParams, @@ -98,31 +94,28 @@ var createInsightsMiddleware = function createInsightsMiddleware(props) { appId: appId, apiKey: apiKey }, insightsInitParams)); - var createWidget = (0, _connectConfigure.default)(_index2.noop); - var configureClickAnalytics; - var configureUserToken; + var initialParameters; + var helper; return { onStateChange: function onStateChange() {}, - subscribe: function subscribe() { + subscribe: function subscribe() {}, + started: function started() { insightsClient('addAlgoliaAgent', 'insights-middleware'); - configureClickAnalytics = createWidget({ - searchParameters: { - clickAnalytics: true - } - }); - instantSearchInstance.addWidgets([configureClickAnalytics]); + helper = instantSearchInstance.helper; + initialParameters = { + userToken: helper.state.userToken, + clickAnalytics: helper.state.clickAnalytics + }; + helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread(_objectSpread({}, helper.state), {}, { + clickAnalytics: true + })); + instantSearchInstance.scheduleSearch(); var setUserTokenToSearch = function setUserTokenToSearch(userToken) { - if (configureUserToken) { - instantSearchInstance.removeWidgets([configureUserToken]); - } - - configureUserToken = createWidget({ - searchParameters: { - userToken: userToken - } - }); - instantSearchInstance.addWidgets([configureUserToken]); + helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread(_objectSpread({}, helper.state), {}, { + userToken: userToken + })); + instantSearchInstance.scheduleSearch(); }; var anonymousUserToken = (0, _index.getInsightsAnonymousUserTokenInternal)(); @@ -150,9 +143,7 @@ var createInsightsMiddleware = function createInsightsMiddleware(props) { if (onEvent) { onEvent(event, _insightsClient); } else if (event.insightsMethod) { - // At this point, instantSearchInstance must be started and - // it means there is a configure widget (added above). - var hasUserToken = Boolean(instantSearchInstance.renderState[instantSearchInstance.indexName].configure.widgetParams.searchParameters.userToken); + var hasUserToken = Boolean(helper.state.userToken); if (hasUserToken) { insightsClient(event.insightsMethod, event.payload); @@ -166,13 +157,13 @@ var createInsightsMiddleware = function createInsightsMiddleware(props) { }, unsubscribe: function unsubscribe() { insightsClient('onUserTokenChange', undefined); - instantSearchInstance.removeWidgets([configureClickAnalytics, configureUserToken]); - configureClickAnalytics = undefined; - configureUserToken = undefined; instantSearchInstance.sendEventToInsights = _index2.noop; + + if (helper && initialParameters) { + helper.setState(_objectSpread(_objectSpread({}, helper.state), initialParameters)); + instantSearchInstance.scheduleSearch(); + } } }; }; -}; - -exports.createInsightsMiddleware = createInsightsMiddleware; \ No newline at end of file +} \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createMetadataMiddleware.js b/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createMetadataMiddleware.js index 627b754..38ce792 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createMetadataMiddleware.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createMetadataMiddleware.js @@ -9,21 +9,7 @@ exports.createMetadataMiddleware = createMetadataMiddleware; var _index = require("../lib/utils/index.js"); function extractPayload(widgets, instantSearchInstance, payload) { - var parent = instantSearchInstance.mainIndex; - var initOptions = { - instantSearchInstance: instantSearchInstance, - parent: parent, - scopedResults: [], - state: parent.getHelper().state, - helper: parent.getHelper(), - createURL: parent.createURL, - uiState: instantSearchInstance._initialUiState, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }; + var initOptions = (0, _index.createInitArgs)(instantSearchInstance, instantSearchInstance.mainIndex, instantSearchInstance._initialUiState); widgets.forEach(function (widget) { var widgetParams = {}; @@ -94,6 +80,7 @@ function createMetadataMiddleware() { refNode.appendChild(payloadContainer); }, 0); }, + started: function started() {}, unsubscribe: function unsubscribe() { payloadContainer.remove(); } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createRouterMiddleware.js b/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createRouterMiddleware.js index 9bc3dff..bedad55 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createRouterMiddleware.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/middlewares/createRouterMiddleware.js @@ -57,6 +57,7 @@ var createRouterMiddleware = function createRouterMiddleware() { instantSearchInstance.setUiState(stateMapping.routeToState(route)); }); }, + started: function started() {}, unsubscribe: function unsubscribe() { router.dispose(); } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/types/algoliasearch.js b/wp-search-with-algolia/js/instantsearch.js/cjs/types/algoliasearch.js index 9a390c3..86f345f 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/types/algoliasearch.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/types/algoliasearch.js @@ -1 +1,18 @@ -"use strict"; \ No newline at end of file +"use strict"; + +Object.defineProperty(exports, "__esModule", { + value: true +}); + +var _algoliasearch = require("algoliasearch-helper/types/algoliasearch.js"); + +Object.keys(_algoliasearch).forEach(function (key) { + if (key === "default" || key === "__esModule") return; + if (key in exports && exports[key] === _algoliasearch[key]) return; + Object.defineProperty(exports, key, { + enumerable: true, + get: function get() { + return _algoliasearch[key]; + } + }); +}); \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/answers.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/answers.js index cabda41..80bb3f7 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/answers.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/answers.js @@ -7,12 +7,14 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")); var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); var _Answers = _interopRequireDefault(require("../../components/Answers/Answers.js")); @@ -43,7 +45,7 @@ var renderer = function renderer(_ref) { instantSearchInstance = _ref2.instantSearchInstance; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -81,20 +83,20 @@ var answersWidget = function answersWidget(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - emptyRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + emptyRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'empty' }), userCssClasses.emptyRoot), - header: (0, _classnames.default)(suit({ + header: (0, _uiComponentsShared.cx)(suit({ descendantName: 'header' }), userCssClasses.header), - loader: (0, _classnames.default)(suit({ + loader: (0, _uiComponentsShared.cx)(suit({ descendantName: 'loader' }), userCssClasses.loader), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/defaultTemplates.js index c94670d..0cf3c24 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/answers/defaultTemplates.js @@ -5,8 +5,12 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; var defaultTemplates = { - header: '', - loader: '', + header: function header() { + return ''; + }, + loader: function loader() { + return ''; + }, item: function item(_item) { return JSON.stringify(_item); } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/breadcrumb.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/breadcrumb.js index 2a000c8..9aea722 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/breadcrumb.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/breadcrumb.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Breadcrumb = _interopRequireDefault(require("../../components/Breadcrumb/Breadcrumb.js")); @@ -17,6 +17,8 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -45,7 +47,7 @@ var renderer = function renderer(_ref) { refine = _ref2.refine; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -82,24 +84,24 @@ var breadcrumb = function breadcrumb(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - selectedItem: (0, _classnames.default)(suit({ + selectedItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - separator: (0, _classnames.default)(suit({ + separator: (0, _uiComponentsShared.cx)(suit({ descendantName: 'separator' }), userCssClasses.separator), - link: (0, _classnames.default)(suit({ + link: (0, _uiComponentsShared.cx)(suit({ descendantName: 'link' }), userCssClasses.link) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/defaultTemplates.js index 91aae42..5f61c74 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/breadcrumb/defaultTemplates.js @@ -5,8 +5,12 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; var defaultTemplates = { - home: 'Home', - separator: '>' + home: function home() { + return 'Home'; + }, + separator: function separator() { + return '>'; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/clear-refinements.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/clear-refinements.js index 4b9e9ac..69ec157 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/clear-refinements.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/clear-refinements.js @@ -9,7 +9,7 @@ var _preact = require("preact"); var _ClearRefinements = _interopRequireDefault(require("../../components/ClearRefinements/ClearRefinements.js")); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _connectClearRefinements = _interopRequireDefault(require("../../connectors/clear-refinements/connectClearRefinements.js")); @@ -17,6 +17,8 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -39,11 +41,11 @@ var renderer = function renderer(_ref) { templates = _ref.templates; return function (_ref2, isFirstRendering) { var refine = _ref2.refine, - hasRefinements = _ref2.hasRefinements, + canRefine = _ref2.canRefine, instantSearchInstance = _ref2.instantSearchInstance; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -54,7 +56,7 @@ var renderer = function renderer(_ref) { (0, _preact.render)((0, _preact.h)(_ClearRefinements.default, { refine: refine, cssClasses: cssClasses, - hasRefinements: hasRefinements, + hasRefinements: canRefine, templateProps: renderState.templateProps }), containerNode); }; @@ -77,11 +79,11 @@ var clearRefinements = function clearRefinements(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - button: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + button: (0, _uiComponentsShared.cx)(suit({ descendantName: 'button' }), userCssClasses.button), - disabledButton: (0, _classnames.default)(suit({ + disabledButton: (0, _uiComponentsShared.cx)(suit({ descendantName: 'button', modifierName: 'disabled' }), userCssClasses.disabledButton) diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/defaultTemplates.js index d178396..8623029 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/clear-refinements/defaultTemplates.js @@ -5,7 +5,9 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; var defaultTemplates = { - resetLabel: 'Clear refinements' + resetLabel: function resetLabel() { + return 'Clear refinements'; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/current-refinements/current-refinements.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/current-refinements/current-refinements.js index adef34c..1dc9a4b 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/current-refinements/current-refinements.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/current-refinements/current-refinements.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _CurrentRefinements = _interopRequireDefault(require("../../components/CurrentRefinements/CurrentRefinements.js")); @@ -32,7 +32,8 @@ var suit = (0, _suit.component)('CurrentRefinements'); var renderer = function renderer(_ref, isFirstRender) { var items = _ref.items, - widgetParams = _ref.widgetParams; + widgetParams = _ref.widgetParams, + canRefine = _ref.canRefine; if (isFirstRender) { return; @@ -43,7 +44,8 @@ var renderer = function renderer(_ref, isFirstRender) { cssClasses = _ref2.cssClasses; (0, _preact.render)((0, _preact.h)(_CurrentRefinements.default, { cssClasses: cssClasses, - items: items + items: items, + canRefine: canRefine }), container); }; @@ -62,23 +64,26 @@ var currentRefinements = function currentRefinements(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - list: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ + modifierName: 'noRefinement' + }), userCssClasses.noRefinementRoot), + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - label: (0, _classnames.default)(suit({ + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - category: (0, _classnames.default)(suit({ + category: (0, _uiComponentsShared.cx)(suit({ descendantName: 'category' }), userCssClasses.category), - categoryLabel: (0, _classnames.default)(suit({ + categoryLabel: (0, _uiComponentsShared.cx)(suit({ descendantName: 'categoryLabel' }), userCssClasses.categoryLabel), - delete: (0, _classnames.default)(suit({ + delete: (0, _uiComponentsShared.cx)(suit({ descendantName: 'delete' }), userCssClasses.delete) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/GeoSearchRenderer.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/GeoSearchRenderer.js index 10f3bb5..b7e1da4 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/GeoSearchRenderer.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/GeoSearchRenderer.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _index = require("../../lib/utils/index.js"); +var _index = require("../../lib/templating/index.js"); var _GeoSearchControls = _interopRequireDefault(require("../../components/GeoSearchControls/GeoSearchControls.js")); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/createHTMLMarker.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/createHTMLMarker.js index 56012fc..d3ca75e 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/createHTMLMarker.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/createHTMLMarker.js @@ -1,12 +1,14 @@ "use strict"; -function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); } - Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; +var _preact = require("preact"); + +function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); } + function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } @@ -29,7 +31,6 @@ function _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.g function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } -/* global google EventListener */ var createHTMLMarker = function createHTMLMarker(googleReference) { var HTMLMarker = /*#__PURE__*/function (_googleReference$maps) { _inherits(HTMLMarker, _googleReference$maps); @@ -73,7 +74,12 @@ var createHTMLMarker = function createHTMLMarker(googleReference) { _this.element = document.createElement('div'); _this.element.className = className; _this.element.style.position = 'absolute'; - _this.element.innerHTML = template; + + if (_typeof(template) === 'object') { + (0, _preact.render)(template, _this.element); + } else { + _this.element.innerHTML = template; + } _this.setMap(map); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/defaultTemplates.js index 7ba20e7..60d2b35 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/defaultTemplates.js @@ -4,11 +4,24 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _preact = require("preact"); + +var _ref = (0, _preact.h)("p", null, "Your custom HTML Marker"); + var defaultTemplates = { - HTMLMarker: '

Your custom HTML Marker

', - reset: 'Clear the map refinement', - toggle: 'Search as I move the map', - redo: 'Redo search here' + HTMLMarker: function HTMLMarker() { + return _ref; + }, + reset: function reset() { + return 'Clear the map refinement'; + }, + toggle: function toggle() { + return 'Search as I move the map'; + }, + redo: function redo() { + return 'Redo search here'; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/geo-search.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/geo-search.js index e565ce9..5343c56 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/geo-search.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/geo-search/geo-search.js @@ -5,12 +5,14 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _preact = require("preact"); var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); var _connectGeoSearch = _interopRequireDefault(require("../../connectors/geo-search/connectGeoSearch.js")); @@ -100,35 +102,35 @@ var geoSearch = function geoSearch(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), // Required only to mount / unmount the Preact tree tree: suit({ descendantName: 'tree' }), - map: (0, _classnames.default)(suit({ + map: (0, _uiComponentsShared.cx)(suit({ descendantName: 'map' }), userCssClasses.map), - control: (0, _classnames.default)(suit({ + control: (0, _uiComponentsShared.cx)(suit({ descendantName: 'control' }), userCssClasses.control), - label: (0, _classnames.default)(suit({ + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - selectedLabel: (0, _classnames.default)(suit({ + selectedLabel: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label', modifierName: 'selected' }), userCssClasses.selectedLabel), - input: (0, _classnames.default)(suit({ + input: (0, _uiComponentsShared.cx)(suit({ descendantName: 'input' }), userCssClasses.input), - redo: (0, _classnames.default)(suit({ + redo: (0, _uiComponentsShared.cx)(suit({ descendantName: 'redo' }), userCssClasses.redo), - disabledRedo: (0, _classnames.default)(suit({ + disabledRedo: (0, _uiComponentsShared.cx)(suit({ descendantName: 'redo', modifierName: 'disabled' }), userCssClasses.disabledRedo), - reset: (0, _classnames.default)(suit({ + reset: (0, _uiComponentsShared.cx)(suit({ descendantName: 'reset' }), userCssClasses.reset) }; @@ -161,10 +163,10 @@ var geoSearch = function geoSearch(widgetParams) { return new HTMLMarker(_objectSpread(_objectSpread(_objectSpread({}, customHTMLMarker.createOptions(item)), rest), {}, { __id: item.objectID, position: item._geoloc, - className: (0, _classnames.default)(suit({ + className: (0, _uiComponentsShared.cx)(suit({ descendantName: 'marker' })), - template: (0, _index.renderTemplate)({ + template: (0, _index2.renderTemplate)({ templateKey: 'HTMLMarker', templates: templates, data: item diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/defaultTemplates.js index 69d663e..bb50d49 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/defaultTemplates.js @@ -4,9 +4,33 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _preact = require("preact"); + +var _formatNumber = require("../../lib/formatNumber.js"); + +var _uiComponentsShared = require("@algolia/ui-components-shared"); + var defaultTemplates = { - item: '' + '{{label}}' + '{{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}}' + '', - showMoreText: "\n {{#isShowingMore}}\n Show less\n {{/isShowingMore}}\n {{^isShowingMore}}\n Show more\n {{/isShowingMore}}\n " + item: function item(_ref) { + var url = _ref.url, + label = _ref.label, + count = _ref.count, + cssClasses = _ref.cssClasses, + isRefined = _ref.isRefined; + return (0, _preact.h)("a", { + className: (0, _uiComponentsShared.cx)((0, _uiComponentsShared.cx)(cssClasses.link), (0, _uiComponentsShared.cx)(isRefined ? cssClasses.selectedItemLink : undefined)), + href: url + }, (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.label) + }, label), (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.count) + }, (0, _formatNumber.formatNumber)(count))); + }, + showMoreText: function showMoreText(_ref2) { + var isShowingMore = _ref2.isShowingMore; + return isShowingMore ? 'Show less' : 'Show more'; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/hierarchical-menu.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/hierarchical-menu.js index ca7ad11..6368153 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/hierarchical-menu.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hierarchical-menu/hierarchical-menu.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _RefinementList = _interopRequireDefault(require("../../components/RefinementList/RefinementList.js")); @@ -17,6 +17,8 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -48,7 +50,7 @@ var renderer = function renderer(_ref) { canToggleShowMore = _ref2.canToggleShowMore; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -146,41 +148,45 @@ var hierarchicalMenu = function hierarchicalMenu(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - childList: (0, _classnames.default)(suit({ + childList: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list', modifierName: 'child' }), userCssClasses.childList), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - selectedItem: (0, _classnames.default)(suit({ + selectedItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - parentItem: (0, _classnames.default)(suit({ + parentItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'parent' }), userCssClasses.parentItem), - link: (0, _classnames.default)(suit({ + link: (0, _uiComponentsShared.cx)(suit({ descendantName: 'link' }), userCssClasses.link), - label: (0, _classnames.default)(suit({ + selectedItemLink: (0, _uiComponentsShared.cx)(suit({ + descendantName: 'link', + modifierName: 'selected' + }), userCssClasses.selectedItemLink), + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - count: (0, _classnames.default)(suit({ + count: (0, _uiComponentsShared.cx)(suit({ descendantName: 'count' }), userCssClasses.count), - showMore: (0, _classnames.default)(suit({ + showMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'showMore' }), userCssClasses.showMore), - disabledShowMore: (0, _classnames.default)(suit({ + disabledShowMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'showMore', modifierName: 'disabled' }), userCssClasses.disabledShowMore) diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits-per-page/hits-per-page.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits-per-page/hits-per-page.js index 3e8f4b2..9590558 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits-per-page/hits-per-page.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits-per-page/hits-per-page.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Selector = _interopRequireDefault(require("../../components/Selector/Selector.js")); @@ -69,11 +69,11 @@ var hitsPerPage = function hitsPerPage(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - select: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + select: (0, _uiComponentsShared.cx)(suit({ descendantName: 'select' }), userCssClasses.select), - option: (0, _classnames.default)(suit({ + option: (0, _uiComponentsShared.cx)(suit({ descendantName: 'option' }), userCssClasses.option) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/defaultTemplates.js index c9ba195..916eeae 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/defaultTemplates.js @@ -5,7 +5,9 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; var defaultTemplates = { - empty: 'No results', + empty: function empty() { + return 'No results'; + }, item: function item(data) { return JSON.stringify(data, null, 2); } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/hits.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/hits.js index 475cfeb..edc0afd 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/hits.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/hits/hits.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _connectHits = _interopRequireDefault(require("../../connectors/hits/connectHits.js")); @@ -17,9 +17,11 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); -var _index2 = require("../../lib/insights/index.js"); +var _index3 = require("../../lib/insights/index.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -33,7 +35,7 @@ var withUsage = (0, _index.createDocumentationMessageGenerator)({ name: 'hits' }); var suit = (0, _suit.component)('Hits'); -var HitsWithInsightsListener = (0, _index2.withInsightsListener)(_Hits.default); +var HitsWithInsightsListener = (0, _index3.withInsightsListener)(_Hits.default); var renderer = function renderer(_ref) { var renderState = _ref.renderState, @@ -48,7 +50,7 @@ var renderer = function renderer(_ref) { bindEvent = _ref2.bindEvent; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -86,14 +88,14 @@ var hits = function hits(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - emptyRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + emptyRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'empty' }), userCssClasses.emptyRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item) }; @@ -103,7 +105,7 @@ var hits = function hits(widgetParams) { renderState: {}, templates: templates }); - var makeWidget = (0, _index2.withInsights)(_connectHits.default)(specializedRenderer, function () { + var makeWidget = (0, _index3.withInsights)(_connectHits.default)(specializedRenderer, function () { return (0, _preact.render)(null, containerNode); }); return _objectSpread(_objectSpread({}, makeWidget({ diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/index/index.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/index/index.js index c5c9468..6118d1c 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/index/index.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/index/index.js @@ -3,7 +3,6 @@ Object.defineProperty(exports, "__esModule", { value: true }); -exports.isIndexWidget = isIndexWidget; exports.default = void 0; var _algoliasearchHelper = _interopRequireDefault(require("algoliasearch-helper")); @@ -38,16 +37,11 @@ var withUsage = (0, _index.createDocumentationMessageGenerator)({ name: 'index-widget' }); -function isIndexWidget(widget) { - return widget.$$type === 'ais.index'; -} /** * This is the same content as helper._change / setState, but allowing for extra * UiState to be synchronized. * see: https://github.com/algolia/algoliasearch-helper-js/blob/6b835ffd07742f2d6b314022cce6848f5cfecd4a/src/algoliasearch.helper.js#L1311-L1324 */ - - function privateHelperSetState(helper, _ref) { var state = _ref.state, isPageReset = _ref.isPageReset, @@ -67,7 +61,7 @@ function privateHelperSetState(helper, _ref) { function getLocalWidgetsUiState(widgets, widgetStateOptions) { var initialUiState = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {}; return widgets.reduce(function (uiState, widget) { - if (isIndexWidget(widget)) { + if ((0, _index.isIndexWidget)(widget)) { return uiState; } @@ -88,7 +82,7 @@ function getLocalWidgetsSearchParameters(widgets, widgetSearchParametersOptions) rest = _objectWithoutProperties(widgetSearchParametersOptions, ["initialSearchParameters"]); return widgets.filter(function (widget) { - return !isIndexWidget(widget); + return !(0, _index.isIndexWidget)(widget); }).reduce(function (state, widget) { if (!widget.getWidgetSearchParameters) { return state; @@ -99,7 +93,7 @@ function getLocalWidgetsSearchParameters(widgets, widgetSearchParametersOptions) } function resetPageFromWidgets(widgets) { - var indexWidgets = widgets.filter(isIndexWidget); + var indexWidgets = widgets.filter(_index.isIndexWidget); if (indexWidgets.length === 0) { return; @@ -116,7 +110,7 @@ function resetPageFromWidgets(widgets) { } function resolveScopedResultsFromWidgets(widgets) { - var indexWidgets = widgets.filter(isIndexWidget); + var indexWidgets = widgets.filter(_index.isIndexWidget); return indexWidgets.reduce(function (scopedResults, current) { return scopedResults.concat.apply(scopedResults, [{ indexId: current.getIndexId(), @@ -201,20 +195,7 @@ var index = function index(widgetParams) { widgets.forEach(function (widget) { if (widget.getRenderState) { - var renderState = widget.getRenderState(localInstantSearchInstance.renderState[_this.getIndexId()] || {}, { - uiState: localInstantSearchInstance._initialUiState, - helper: _this.getHelper(), - parent: _this, - instantSearchInstance: localInstantSearchInstance, - state: helper.state, - renderState: localInstantSearchInstance.renderState, - templatesConfig: localInstantSearchInstance.templatesConfig, - createURL: _this.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: localInstantSearchInstance._isSearchStalled - } - }); + var renderState = widget.getRenderState(localInstantSearchInstance.renderState[_this.getIndexId()] || {}, (0, _index.createInitArgs)(localInstantSearchInstance, _this, localInstantSearchInstance._initialUiState)); storeRenderState({ renderState: renderState, instantSearchInstance: localInstantSearchInstance, @@ -224,20 +205,7 @@ var index = function index(widgetParams) { }); widgets.forEach(function (widget) { if (widget.init) { - widget.init({ - helper: helper, - parent: _this, - uiState: localInstantSearchInstance._initialUiState, - instantSearchInstance: localInstantSearchInstance, - state: helper.state, - renderState: localInstantSearchInstance.renderState, - templatesConfig: localInstantSearchInstance.templatesConfig, - createURL: _this.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: localInstantSearchInstance._isSearchStalled - } - }); + widget.init((0, _index.createInitArgs)(localInstantSearchInstance, _this, localInstantSearchInstance._initialUiState)); } }); localInstantSearchInstance.scheduleSearch(); @@ -326,7 +294,9 @@ var index = function index(widgetParams) { if (instantSearchInstance.onStateChange) { instantSearchInstance.onStateChange({ uiState: instantSearchInstance.mainIndex.getWidgetUiState({}), - setUiState: instantSearchInstance.setUiState.bind(instantSearchInstance) + setUiState: function setUiState(nextState) { + return instantSearchInstance.setUiState(nextState, false); + } }); // We don't trigger a search when controlled because it becomes the // responsibility of `setUiState`. @@ -401,20 +371,7 @@ var index = function index(widgetParams) { localWidgets.forEach(function (widget) { if (widget.getRenderState) { - var renderState = widget.getRenderState(instantSearchInstance.renderState[_this3.getIndexId()] || {}, { - uiState: uiState, - helper: helper, - parent: _this3, - instantSearchInstance: instantSearchInstance, - state: helper.state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this3.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + var renderState = widget.getRenderState(instantSearchInstance.renderState[_this3.getIndexId()] || {}, (0, _index.createInitArgs)(instantSearchInstance, _this3, uiState)); storeRenderState({ renderState: renderState, instantSearchInstance: instantSearchInstance, @@ -428,20 +385,7 @@ var index = function index(widgetParams) { !widget.getWidgetState || Boolean(widget.getWidgetUiState), 'The `getWidgetState` method is renamed `getWidgetUiState` and will no longer exist under that name in InstantSearch.js 5.x. Please use `getWidgetUiState` instead.') : void 0; if (widget.init) { - widget.init({ - uiState: uiState, - helper: helper, - parent: _this3, - instantSearchInstance: instantSearchInstance, - state: helper.state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this3.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + widget.init((0, _index.createInitArgs)(instantSearchInstance, _this3, uiState)); } }); // Subscribe to the Helper state changes for the `uiState` once widgets // are initialized. Until the first render, state changes are part of the @@ -482,20 +426,7 @@ var index = function index(widgetParams) { localWidgets.forEach(function (widget) { if (widget.getRenderState) { - var renderState = widget.getRenderState(instantSearchInstance.renderState[_this4.getIndexId()] || {}, { - helper: _this4.getHelper(), - parent: _this4, - instantSearchInstance: instantSearchInstance, - results: _this4.getResults(), - scopedResults: _this4.getScopedResults(), - state: _this4.getResults()._state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this4.createURL, - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + var renderState = widget.getRenderState(instantSearchInstance.renderState[_this4.getIndexId()] || {}, (0, _index.createRenderArgs)(instantSearchInstance, _this4)); storeRenderState({ renderState: renderState, instantSearchInstance: instantSearchInstance, @@ -511,20 +442,7 @@ var index = function index(widgetParams) { // be delayed. The render is triggered for the complete tree but some parts do // not have results yet. if (widget.render) { - widget.render({ - helper: helper, - parent: _this4, - instantSearchInstance: instantSearchInstance, - results: _this4.getResults(), - scopedResults: _this4.getScopedResults(), - state: _this4.getResults()._state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this4.createURL, - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + widget.render((0, _index.createRenderArgs)(instantSearchInstance, _this4)); } }); }, @@ -554,9 +472,9 @@ var index = function index(widgetParams) { derivedHelper = null; }, getWidgetUiState: function getWidgetUiState(uiState) { - return localWidgets.filter(isIndexWidget).reduce(function (previousUiState, innerIndex) { + return localWidgets.filter(_index.isIndexWidget).reduce(function (previousUiState, innerIndex) { return innerIndex.getWidgetUiState(previousUiState); - }, _objectSpread(_objectSpread({}, uiState), {}, _defineProperty({}, this.getIndexId(), localUiState))); + }, _objectSpread(_objectSpread({}, uiState), {}, _defineProperty({}, indexId, _objectSpread(_objectSpread({}, uiState[indexId]), localUiState)))); }, getWidgetState: function getWidgetState(uiState) { process.env.NODE_ENV === 'development' ? (0, _index.warning)(false, 'The `getWidgetState` method is renamed `getWidgetUiState` and will no longer exist under that name in InstantSearch.js 5.x. Please use `getWidgetUiState` instead.') : void 0; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/defaultTemplates.js index e62d14e..ed97a0f 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/defaultTemplates.js @@ -5,9 +5,15 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; var defaultTemplates = { - empty: 'No results', - showPreviousText: 'Show previous results', - showMoreText: 'Show more results', + empty: function empty() { + return 'No results'; + }, + showPreviousText: function showPreviousText() { + return 'Show previous results'; + }, + showMoreText: function showMoreText() { + return 'Show more results'; + }, item: function item(data) { return JSON.stringify(data, null, 2); } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/infinite-hits.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/infinite-hits.js index 485aca4..a9577e4 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/infinite-hits.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/infinite-hits/infinite-hits.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _InfiniteHits = _interopRequireDefault(require("../../components/InfiniteHits/InfiniteHits.js")); @@ -15,9 +15,11 @@ var _connectInfiniteHits = _interopRequireDefault(require("../../connectors/infi var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); -var _index2 = require("../../lib/insights/index.js"); +var _index3 = require("../../lib/insights/index.js"); var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")); @@ -33,7 +35,7 @@ var withUsage = (0, _index.createDocumentationMessageGenerator)({ name: 'infinite-hits' }); var suit = (0, _suit.component)('InfiniteHits'); -var InfiniteHitsWithInsightsListener = (0, _index2.withInsightsListener)(_InfiniteHits.default); +var InfiniteHitsWithInsightsListener = (0, _index3.withInsightsListener)(_InfiniteHits.default); var renderer = function renderer(_ref) { var containerNode = _ref.containerNode, @@ -53,7 +55,7 @@ var renderer = function renderer(_ref) { bindEvent = _ref2.bindEvent; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -98,27 +100,27 @@ var infiniteHits = function infiniteHits(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - emptyRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + emptyRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'empty' }), userCssClasses.emptyRoot), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - loadPrevious: (0, _classnames.default)(suit({ + loadPrevious: (0, _uiComponentsShared.cx)(suit({ descendantName: 'loadPrevious' }), userCssClasses.loadPrevious), - disabledLoadPrevious: (0, _classnames.default)(suit({ + disabledLoadPrevious: (0, _uiComponentsShared.cx)(suit({ descendantName: 'loadPrevious', modifierName: 'disabled' }), userCssClasses.disabledLoadPrevious), - loadMore: (0, _classnames.default)(suit({ + loadMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'loadMore' }), userCssClasses.loadMore), - disabledLoadMore: (0, _classnames.default)(suit({ + disabledLoadMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'loadMore', modifierName: 'disabled' }), userCssClasses.disabledLoadMore) @@ -130,7 +132,7 @@ var infiniteHits = function infiniteHits(widgetParams) { showPrevious: showPrevious, renderState: {} }); - var makeWidget = (0, _index2.withInsights)(_connectInfiniteHits.default)(specializedRenderer, function () { + var makeWidget = (0, _index3.withInsights)(_connectInfiniteHits.default)(specializedRenderer, function () { return (0, _preact.render)(null, containerNode); }); return _objectSpread(_objectSpread({}, makeWidget({ diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/defaultTemplates.js index 31f7779..e9592b9 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/defaultTemplates.js @@ -4,9 +4,18 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _formatNumber = require("../../lib/formatNumber.js"); + var defaultTemplates = { - item: '{{label}} ({{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}})', - defaultOption: 'See all' + item: function item(_ref) { + var label = _ref.label, + count = _ref.count; + return "".concat(label, " (").concat((0, _formatNumber.formatNumber)(count), ")"); + }, + defaultOption: function defaultOption() { + return 'See all'; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/menu-select.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/menu-select.js index 153c8f8..a9adcb9 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/menu-select.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu-select/menu-select.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _connectMenu = _interopRequireDefault(require("../../connectors/menu/connectMenu.js")); @@ -17,6 +17,8 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -43,7 +45,7 @@ var renderer = function renderer(_ref) { instantSearchInstance = _ref2.instantSearchInstance; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -80,14 +82,14 @@ var menuSelect = function menuSelect(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - select: (0, _classnames.default)(suit({ + select: (0, _uiComponentsShared.cx)(suit({ descendantName: 'select' }), userCssClasses.select), - option: (0, _classnames.default)(suit({ + option: (0, _uiComponentsShared.cx)(suit({ descendantName: 'option' }), userCssClasses.option) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/defaultTemplates.js index 69d663e..5a163a5 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/defaultTemplates.js @@ -4,9 +4,32 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _preact = require("preact"); + +var _formatNumber = require("../../lib/formatNumber.js"); + +var _uiComponentsShared = require("@algolia/ui-components-shared"); + var defaultTemplates = { - item: '' + '{{label}}' + '{{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}}' + '', - showMoreText: "\n {{#isShowingMore}}\n Show less\n {{/isShowingMore}}\n {{^isShowingMore}}\n Show more\n {{/isShowingMore}}\n " + item: function item(_ref) { + var cssClasses = _ref.cssClasses, + url = _ref.url, + label = _ref.label, + count = _ref.count; + return (0, _preact.h)("a", { + className: (0, _uiComponentsShared.cx)(cssClasses.link), + href: url + }, (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.label) + }, label), (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.count) + }, (0, _formatNumber.formatNumber)(count))); + }, + showMoreText: function showMoreText(_ref2) { + var isShowingMore = _ref2.isShowingMore; + return isShowingMore ? 'Show less' : 'Show more'; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/menu.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/menu.js index 117d893..f915c35 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/menu.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/menu/menu.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _RefinementList = _interopRequireDefault(require("../../components/RefinementList/RefinementList.js")); @@ -17,6 +17,8 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -48,7 +50,7 @@ var renderer = function renderer(_ref) { canToggleShowMore = _ref2.canToggleShowMore; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -95,33 +97,33 @@ var menu = function menu(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - selectedItem: (0, _classnames.default)(suit({ + selectedItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - link: (0, _classnames.default)(suit({ + link: (0, _uiComponentsShared.cx)(suit({ descendantName: 'link' }), userCssClasses.link), - label: (0, _classnames.default)(suit({ + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - count: (0, _classnames.default)(suit({ + count: (0, _uiComponentsShared.cx)(suit({ descendantName: 'count' }), userCssClasses.count), - showMore: (0, _classnames.default)(suit({ + showMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'showMore' }), userCssClasses.showMore), - disabledShowMore: (0, _classnames.default)(suit({ + disabledShowMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'showMore', modifierName: 'disabled' }), userCssClasses.disabledShowMore) diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/defaultTemplates.js index 99019cb..61e9521 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/defaultTemplates.js @@ -4,8 +4,26 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _preact = require("preact"); + var defaultTemplates = { - item: "" + item: function item(_ref) { + var cssClasses = _ref.cssClasses, + attribute = _ref.attribute, + label = _ref.label, + isRefined = _ref.isRefined; + return (0, _preact.h)("label", { + className: cssClasses.label + }, (0, _preact.h)("input", { + type: "radio", + className: cssClasses.radio, + name: attribute, + defaultChecked: isRefined + }), (0, _preact.h)("span", { + className: cssClasses.labelText + }, label)); + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/numeric-menu.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/numeric-menu.js index 9cd567f..e3b9d3d 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/numeric-menu.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/numeric-menu/numeric-menu.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _RefinementList = _interopRequireDefault(require("../../components/RefinementList/RefinementList.js")); @@ -19,6 +19,8 @@ var _index = require("../../lib/utils/index.js"); var _suit = require("../../lib/suit.js"); +var _index2 = require("../../lib/templating/index.js"); + function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } @@ -45,7 +47,7 @@ var renderer = function renderer(_ref) { items = _ref2.items; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -81,27 +83,27 @@ var numericMenu = function numericMenu(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - selectedItem: (0, _classnames.default)(suit({ + selectedItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - label: (0, _classnames.default)(suit({ + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - radio: (0, _classnames.default)(suit({ + radio: (0, _uiComponentsShared.cx)(suit({ descendantName: 'radio' }), userCssClasses.radio), - labelText: (0, _classnames.default)(suit({ + labelText: (0, _uiComponentsShared.cx)(suit({ descendantName: 'labelText' }), userCssClasses.labelText) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/pagination/pagination.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/pagination/pagination.js index ad6f0ec..a57790c 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/pagination/pagination.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/pagination/pagination.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Pagination = _interopRequireDefault(require("../../components/Pagination/Pagination.js")); @@ -109,45 +109,45 @@ var pagination = function pagination(widgetParams) { var scrollTo = userScrollTo === true ? 'body' : userScrollTo; var scrollToNode = scrollTo !== false ? (0, _index.getContainerNode)(scrollTo) : false; var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - firstPageItem: (0, _classnames.default)(suit({ + firstPageItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'firstPage' }), userCssClasses.firstPageItem), - lastPageItem: (0, _classnames.default)(suit({ + lastPageItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'lastPage' }), userCssClasses.lastPageItem), - previousPageItem: (0, _classnames.default)(suit({ + previousPageItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'previousPage' }), userCssClasses.previousPageItem), - nextPageItem: (0, _classnames.default)(suit({ + nextPageItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'nextPage' }), userCssClasses.nextPageItem), - pageItem: (0, _classnames.default)(suit({ + pageItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'page' }), userCssClasses.pageItem), - selectedItem: (0, _classnames.default)(suit({ + selectedItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - disabledItem: (0, _classnames.default)(suit({ + disabledItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'disabled' }), userCssClasses.disabledItem), - link: (0, _classnames.default)(suit({ + link: (0, _uiComponentsShared.cx)(suit({ descendantName: 'link' }), userCssClasses.link) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/panel/panel.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/panel/panel.js index 6f1cb12..4ec7988 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/panel/panel.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/panel/panel.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -74,29 +74,29 @@ var panel = function panel(panelWidgetParams) { return false; }; var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - collapsibleRoot: (0, _classnames.default)(suit({ + collapsibleRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'collapsible' }), userCssClasses.collapsibleRoot), - collapsedRoot: (0, _classnames.default)(suit({ + collapsedRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'collapsed' }), userCssClasses.collapsedRoot), - collapseButton: (0, _classnames.default)(suit({ + collapseButton: (0, _uiComponentsShared.cx)(suit({ descendantName: 'collapseButton' }), userCssClasses.collapseButton), - collapseIcon: (0, _classnames.default)(suit({ + collapseIcon: (0, _uiComponentsShared.cx)(suit({ descendantName: 'collapseIcon' }), userCssClasses.collapseIcon), - body: (0, _classnames.default)(suit({ + body: (0, _uiComponentsShared.cx)(suit({ descendantName: 'body' }), userCssClasses.body), - header: (0, _classnames.default)(suit({ + header: (0, _uiComponentsShared.cx)(suit({ descendantName: 'header' }), userCssClasses.header), - footer: (0, _classnames.default)(suit({ + footer: (0, _uiComponentsShared.cx)(suit({ descendantName: 'footer' }), userCssClasses.footer) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/powered-by/powered-by.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/powered-by/powered-by.js index 1c9fdc1..ce5a576 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/powered-by/powered-by.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/powered-by/powered-by.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _PoweredBy = _interopRequireDefault(require("../../components/PoweredBy/PoweredBy.js")); @@ -64,13 +64,13 @@ var poweredBy = function poweredBy(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), suit({ + root: (0, _uiComponentsShared.cx)(suit(), suit({ modifierName: theme === 'dark' ? 'dark' : 'light' }), userCssClasses.root), - link: (0, _classnames.default)(suit({ + link: (0, _uiComponentsShared.cx)(suit({ descendantName: 'link' }), userCssClasses.link), - logo: (0, _classnames.default)(suit({ + logo: (0, _uiComponentsShared.cx)(suit({ descendantName: 'logo' }), userCssClasses.logo) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/query-rule-custom-data/query-rule-custom-data.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/query-rule-custom-data/query-rule-custom-data.js index a848549..78c2e9f 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/query-rule-custom-data/query-rule-custom-data.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/query-rule-custom-data/query-rule-custom-data.js @@ -7,7 +7,7 @@ exports.default = exports.defaultTemplates = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -68,7 +68,7 @@ var queryRuleCustomData = function queryRuleCustomData(widgetParams) { } var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root) + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root) }; var containerNode = (0, _index.getContainerNode)(container); diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-input/range-input.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-input/range-input.js index ccd5cc0..1c6ebed 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-input/range-input.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-input/range-input.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _RangeInput = _interopRequireDefault(require("../../components/RangeInput/RangeInput.js")); @@ -15,6 +15,8 @@ var _connectRange = _interopRequireDefault(require("../../connectors/range/conne var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -42,8 +44,12 @@ var withUsage = (0, _index.createDocumentationMessageGenerator)({ }); var suit = (0, _suit.component)('RangeInput'); var defaultTemplates = { - separatorText: 'to', - submitText: 'Go' + separatorText: function separatorText() { + return 'to'; + }, + submitText: function submitText() { + return 'Go'; + } }; var renderer = function renderer(_ref) { @@ -59,7 +65,7 @@ var renderer = function renderer(_ref) { instantSearchInstance = _ref2.instantSearchInstance; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: defaultTemplates, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -110,31 +116,31 @@ var rangeInput = function rangeInput(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinement: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinement: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' })), - form: (0, _classnames.default)(suit({ + form: (0, _uiComponentsShared.cx)(suit({ descendantName: 'form' }), userCssClasses.form), - label: (0, _classnames.default)(suit({ + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - input: (0, _classnames.default)(suit({ + input: (0, _uiComponentsShared.cx)(suit({ descendantName: 'input' }), userCssClasses.input), - inputMin: (0, _classnames.default)(suit({ + inputMin: (0, _uiComponentsShared.cx)(suit({ descendantName: 'input', modifierName: 'min' }), userCssClasses.inputMin), - inputMax: (0, _classnames.default)(suit({ + inputMax: (0, _uiComponentsShared.cx)(suit({ descendantName: 'input', modifierName: 'max' }), userCssClasses.inputMax), - separator: (0, _classnames.default)(suit({ + separator: (0, _uiComponentsShared.cx)(suit({ descendantName: 'separator' }), userCssClasses.separator), - submit: (0, _classnames.default)(suit({ + submit: (0, _uiComponentsShared.cx)(suit({ descendantName: 'submit' }), userCssClasses.submit) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-slider/range-slider.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-slider/range-slider.js index bca1a3f..d862c42 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-slider/range-slider.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/range-slider/range-slider.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Slider = _interopRequireDefault(require("../../components/Slider/Slider.js")); @@ -117,8 +117,8 @@ var rangeSlider = function rangeSlider(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - disabledRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + disabledRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'disabled' }), userCssClasses.disabledRoot) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/defaultTemplates.js index 5fdc1d4..e674cd4 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/defaultTemplates.js @@ -4,8 +4,63 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _preact = require("preact"); + +var _formatNumber = require("../../lib/formatNumber.js"); + +var _uiComponentsShared = require("@algolia/ui-components-shared"); + +function ItemWrapper(_ref) { + var children = _ref.children, + count = _ref.count, + value = _ref.value, + url = _ref.url, + cssClasses = _ref.cssClasses; + + if (count) { + return (0, _preact.h)("a", { + className: (0, _uiComponentsShared.cx)(cssClasses.link), + "aria-label": "".concat(value, " & up"), + href: url + }, children); + } + + return (0, _preact.h)("div", { + className: (0, _uiComponentsShared.cx)(cssClasses.link), + "aria-label": "".concat(value, " & up"), + disabled: true + }, children); +} + var defaultTemplates = { - item: "{{#count}}{{/count}}{{^count}}{{/count}}" + item: function item(_ref2) { + var count = _ref2.count, + value = _ref2.value, + url = _ref2.url, + stars = _ref2.stars, + cssClasses = _ref2.cssClasses; + return (0, _preact.h)(ItemWrapper, { + count: count, + value: value, + url: url, + cssClasses: cssClasses + }, stars.map(function (isFull, index) { + return (0, _preact.h)("svg", { + key: index, + className: (0, _uiComponentsShared.cx)(cssClasses.starIcon, isFull ? cssClasses.fullStarIcon : cssClasses.emptyStarIcon), + "aria-hidden": "true", + width: "24", + height: "24" + }, (0, _preact.h)("use", { + xlinkHref: isFull ? '#ais-RatingMenu-starSymbol' : '#ais-RatingMenu-starEmptySymbol' + })); + }), (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.label) + }, "& Up"), count && (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.count) + }, (0, _formatNumber.formatNumber)(count))); + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/rating-menu.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/rating-menu.js index ca91287..3bf5b59 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/rating-menu.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/rating-menu/rating-menu.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _RefinementList = _interopRequireDefault(require("../../components/RefinementList/RefinementList.js")); @@ -17,6 +17,8 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -52,7 +54,7 @@ var renderer = function renderer(_ref) { instantSearchInstance = _ref2.instantSearchInstance; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -126,42 +128,42 @@ var ratingMenu = function ratingMenu(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - selectedItem: (0, _classnames.default)(suit({ + selectedItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - disabledItem: (0, _classnames.default)(suit({ + disabledItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'disabled' }), userCssClasses.disabledItem), - link: (0, _classnames.default)(suit({ + link: (0, _uiComponentsShared.cx)(suit({ descendantName: 'link' }), userCssClasses.link), - starIcon: (0, _classnames.default)(suit({ + starIcon: (0, _uiComponentsShared.cx)(suit({ descendantName: 'starIcon' }), userCssClasses.starIcon), - fullStarIcon: (0, _classnames.default)(suit({ + fullStarIcon: (0, _uiComponentsShared.cx)(suit({ descendantName: 'starIcon', modifierName: 'full' }), userCssClasses.fullStarIcon), - emptyStarIcon: (0, _classnames.default)(suit({ + emptyStarIcon: (0, _uiComponentsShared.cx)(suit({ descendantName: 'starIcon', modifierName: 'empty' }), userCssClasses.emptyStarIcon), - label: (0, _classnames.default)(suit({ + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - count: (0, _classnames.default)(suit({ + count: (0, _uiComponentsShared.cx)(suit({ descendantName: 'count' }), userCssClasses.count) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/defaultTemplates.js index bba09be..545e245 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/defaultTemplates.js @@ -4,10 +4,44 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _preact = require("preact"); + +var _formatNumber = require("../../lib/formatNumber.js"); + +var _uiComponentsShared = require("@algolia/ui-components-shared"); + var defaultTemplates = { - item: "", - showMoreText: "\n {{#isShowingMore}}\n Show less\n {{/isShowingMore}}\n {{^isShowingMore}}\n Show more\n {{/isShowingMore}}\n ", - searchableNoResults: 'No results' + item: function item(_ref) { + var cssClasses = _ref.cssClasses, + count = _ref.count, + value = _ref.value, + highlighted = _ref.highlighted, + isRefined = _ref.isRefined, + isFromSearch = _ref.isFromSearch; + return (0, _preact.h)("label", { + className: (0, _uiComponentsShared.cx)(cssClasses.label) + }, (0, _preact.h)("input", { + type: "checkbox", + className: (0, _uiComponentsShared.cx)(cssClasses.checkbox), + value: value, + defaultChecked: isRefined + }), (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.labelText), + dangerouslySetInnerHTML: isFromSearch ? { + __html: highlighted + } : undefined + }, !isFromSearch && highlighted), (0, _preact.h)("span", { + className: (0, _uiComponentsShared.cx)(cssClasses.count) + }, (0, _formatNumber.formatNumber)(count))); + }, + showMoreText: function showMoreText(_ref2) { + var isShowingMore = _ref2.isShowingMore; + return isShowingMore ? 'Show less' : 'Show more'; + }, + searchableNoResults: function searchableNoResults() { + return 'No results'; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/refinement-list.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/refinement-list.js index 756a4f8..5b2b5fa 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/refinement-list.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/refinement-list/refinement-list.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _RefinementList = _interopRequireDefault(require("../../components/RefinementList/RefinementList.js")); @@ -15,6 +15,8 @@ var _connectRefinementList = _interopRequireDefault(require("../../connectors/re var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); var _defaultTemplates = _interopRequireDefault(require("../search-box/defaultTemplates.js")); @@ -58,12 +60,12 @@ var renderer = function renderer(_ref) { canToggleShowMore = _ref2.canToggleShowMore; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates2.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates }); - renderState.searchBoxTemplateProps = (0, _index.prepareTemplateProps)({ + renderState.searchBoxTemplateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: searchBoxTemplates @@ -140,69 +142,69 @@ var refinementList = function refinementList(widgetParams) { var escapeFacetValues = searchable ? Boolean(searchableEscapeFacetValues) : false; var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - noRefinementRoot: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + noRefinementRoot: (0, _uiComponentsShared.cx)(suit({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: (0, _classnames.default)(suit({ + list: (0, _uiComponentsShared.cx)(suit({ descendantName: 'list' }), userCssClasses.list), - item: (0, _classnames.default)(suit({ + item: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item' }), userCssClasses.item), - selectedItem: (0, _classnames.default)(suit({ + selectedItem: (0, _uiComponentsShared.cx)(suit({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - searchBox: (0, _classnames.default)(suit({ + searchBox: (0, _uiComponentsShared.cx)(suit({ descendantName: 'searchBox' }), userCssClasses.searchBox), - label: (0, _classnames.default)(suit({ + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - checkbox: (0, _classnames.default)(suit({ + checkbox: (0, _uiComponentsShared.cx)(suit({ descendantName: 'checkbox' }), userCssClasses.checkbox), - labelText: (0, _classnames.default)(suit({ + labelText: (0, _uiComponentsShared.cx)(suit({ descendantName: 'labelText' }), userCssClasses.labelText), - count: (0, _classnames.default)(suit({ + count: (0, _uiComponentsShared.cx)(suit({ descendantName: 'count' }), userCssClasses.count), - noResults: (0, _classnames.default)(suit({ + noResults: (0, _uiComponentsShared.cx)(suit({ descendantName: 'noResults' }), userCssClasses.noResults), - showMore: (0, _classnames.default)(suit({ + showMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'showMore' }), userCssClasses.showMore), - disabledShowMore: (0, _classnames.default)(suit({ + disabledShowMore: (0, _uiComponentsShared.cx)(suit({ descendantName: 'showMore', modifierName: 'disabled' }), userCssClasses.disabledShowMore), searchable: { - root: (0, _classnames.default)(searchBoxSuit(), userCssClasses.searchableRoot), - form: (0, _classnames.default)(searchBoxSuit({ + root: (0, _uiComponentsShared.cx)(searchBoxSuit(), userCssClasses.searchableRoot), + form: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'form' }), userCssClasses.searchableForm), - input: (0, _classnames.default)(searchBoxSuit({ + input: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'input' }), userCssClasses.searchableInput), - submit: (0, _classnames.default)(searchBoxSuit({ + submit: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'submit' }), userCssClasses.searchableSubmit), - submitIcon: (0, _classnames.default)(searchBoxSuit({ + submitIcon: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'submitIcon' }), userCssClasses.searchableSubmitIcon), - reset: (0, _classnames.default)(searchBoxSuit({ + reset: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'reset' }), userCssClasses.searchableReset), - resetIcon: (0, _classnames.default)(searchBoxSuit({ + resetIcon: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'resetIcon' }), userCssClasses.searchableResetIcon), - loadingIndicator: (0, _classnames.default)(searchBoxSuit({ + loadingIndicator: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'loadingIndicator' }), userCssClasses.searchableLoadingIndicator), - loadingIcon: (0, _classnames.default)(searchBoxSuit({ + loadingIcon: (0, _uiComponentsShared.cx)(searchBoxSuit({ descendantName: 'loadingIcon' }), userCssClasses.searchableLoadingIcon) } diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/defaultTemplates.js index ae5ffd2..fd1b69a 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/defaultTemplates.js @@ -5,7 +5,9 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; var defaultTemplates = { - text: '', + text: function text() { + return ''; + }, button: function button(_ref) { var isRelevantSorted = _ref.isRelevantSorted; return isRelevantSorted ? 'See all results' : 'See relevant results'; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/relevant-sort.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/relevant-sort.js index 6d75434..28f9562 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/relevant-sort.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/relevant-sort/relevant-sort.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -63,11 +63,11 @@ var relevantSort = function relevantSort(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - text: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + text: (0, _uiComponentsShared.cx)(suit({ descendantName: 'text' }), userCssClasses.text), - button: (0, _classnames.default)(suit({ + button: (0, _uiComponentsShared.cx)(suit({ descendantName: 'button' }), userCssClasses.button) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/defaultTemplates.js index fa48695..0ac4fe7 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/defaultTemplates.js @@ -4,10 +4,68 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; + +var _preact = require("preact"); + +var _ref2 = (0, _preact.h)("path", { + d: "M8.114 10L.944 2.83 0 1.885 1.886 0l.943.943L10 8.113l7.17-7.17.944-.943L20 1.886l-.943.943-7.17 7.17 7.17 7.17.943.944L18.114 20l-.943-.943-7.17-7.17-7.17 7.17-.944.943L0 18.114l.943-.943L8.113 10z" +}); + +var _ref4 = (0, _preact.h)("path", { + d: "M26.804 29.01c-2.832 2.34-6.465 3.746-10.426 3.746C7.333 32.756 0 25.424 0 16.378 0 7.333 7.333 0 16.378 0c9.046 0 16.378 7.333 16.378 16.378 0 3.96-1.406 7.594-3.746 10.426l10.534 10.534c.607.607.61 1.59-.004 2.202-.61.61-1.597.61-2.202.004L26.804 29.01zm-10.426.627c7.323 0 13.26-5.936 13.26-13.26 0-7.32-5.937-13.257-13.26-13.257C9.056 3.12 3.12 9.056 3.12 16.378c0 7.323 5.936 13.26 13.258 13.26z" +}); + +var _ref6 = (0, _preact.h)("g", { + fill: "none", + fillRule: "evenodd" +}, (0, _preact.h)("g", { + transform: "translate(1 1)", + strokeWidth: "2" +}, (0, _preact.h)("circle", { + strokeOpacity: ".5", + cx: "18", + cy: "18", + r: "18" +}), (0, _preact.h)("path", { + d: "M36 18c0-9.94-8.06-18-18-18" +}, (0, _preact.h)("animateTransform", { + attributeName: "transform", + type: "rotate", + from: "0 18 18", + to: "360 18 18", + dur: "1s", + repeatCount: "indefinite" +})))); + var defaultTemplate = { - reset: "\n\n \n\n ", - submit: "\n\n \n\n ", - loadingIndicator: "\n\n \n \n \n \n \n \n \n \n\n " + reset: function reset(_ref) { + var cssClasses = _ref.cssClasses; + return (0, _preact.h)("svg", { + className: cssClasses.resetIcon, + viewBox: "0 0 20 20", + width: "10", + height: "10" + }, _ref2); + }, + submit: function submit(_ref3) { + var cssClasses = _ref3.cssClasses; + return (0, _preact.h)("svg", { + className: cssClasses.submitIcon, + width: "10", + height: "10", + viewBox: "0 0 40 40" + }, _ref4); + }, + loadingIndicator: function loadingIndicator(_ref5) { + var cssClasses = _ref5.cssClasses; + return (0, _preact.h)("svg", { + className: cssClasses.loadingIcon, + width: "16", + height: "16", + viewBox: "0 0 38 38", + stroke: "#444" + }, _ref6); + } }; var _default = defaultTemplate; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/search-box.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/search-box.js index fd99e2d..79cfa51 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/search-box.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/search-box/search-box.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -98,29 +98,29 @@ var searchBox = function searchBox(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - form: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + form: (0, _uiComponentsShared.cx)(suit({ descendantName: 'form' }), userCssClasses.form), - input: (0, _classnames.default)(suit({ + input: (0, _uiComponentsShared.cx)(suit({ descendantName: 'input' }), userCssClasses.input), - submit: (0, _classnames.default)(suit({ + submit: (0, _uiComponentsShared.cx)(suit({ descendantName: 'submit' }), userCssClasses.submit), - submitIcon: (0, _classnames.default)(suit({ + submitIcon: (0, _uiComponentsShared.cx)(suit({ descendantName: 'submitIcon' }), userCssClasses.submitIcon), - reset: (0, _classnames.default)(suit({ + reset: (0, _uiComponentsShared.cx)(suit({ descendantName: 'reset' }), userCssClasses.reset), - resetIcon: (0, _classnames.default)(suit({ + resetIcon: (0, _uiComponentsShared.cx)(suit({ descendantName: 'resetIcon' }), userCssClasses.resetIcon), - loadingIndicator: (0, _classnames.default)(suit({ + loadingIndicator: (0, _uiComponentsShared.cx)(suit({ descendantName: 'loadingIndicator' }), userCssClasses.loadingIndicator), - loadingIcon: (0, _classnames.default)(suit({ + loadingIcon: (0, _uiComponentsShared.cx)(suit({ descendantName: 'loadingIcon' }), userCssClasses.loadingIcon) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/sort-by/sort-by.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/sort-by/sort-by.js index 14cc44b..9b227c7 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/sort-by/sort-by.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/sort-by/sort-by.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Selector = _interopRequireDefault(require("../../components/Selector/Selector.js")); @@ -72,11 +72,11 @@ var sortBy = function sortBy(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - select: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + select: (0, _uiComponentsShared.cx)(suit({ descendantName: 'select' }), userCssClasses.select), - option: (0, _classnames.default)(suit({ + option: (0, _uiComponentsShared.cx)(suit({ descendantName: 'option' }), userCssClasses.option) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/stats/stats.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/stats/stats.js index f4c89e3..82408ae 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/stats/stats.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/stats/stats.js @@ -7,7 +7,7 @@ exports.default = exports.defaultTemplates = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _Stats = _interopRequireDefault(require("../../components/Stats/Stats.js")); @@ -15,8 +15,12 @@ var _connectStats = _interopRequireDefault(require("../../connectors/stats/conne var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); +var _formatNumber = require("../../lib/formatNumber.js"); + function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } @@ -30,28 +34,74 @@ var withUsage = (0, _index.createDocumentationMessageGenerator)({ }); var suit = (0, _suit.component)('Stats'); var defaultTemplates = { - text: "\n {{#areHitsSorted}}\n {{#hasNoSortedResults}}No relevant results{{/hasNoSortedResults}}\n {{#hasOneSortedResults}}1 relevant result{{/hasOneSortedResults}}\n {{#hasManySortedResults}}{{#helpers.formatNumber}}{{nbSortedHits}}{{/helpers.formatNumber}} relevant results{{/hasManySortedResults}}\n sorted out of {{#helpers.formatNumber}}{{nbHits}}{{/helpers.formatNumber}}\n {{/areHitsSorted}}\n {{^areHitsSorted}}\n {{#hasNoResults}}No results{{/hasNoResults}}\n {{#hasOneResult}}1 result{{/hasOneResult}}\n {{#hasManyResults}}{{#helpers.formatNumber}}{{nbHits}}{{/helpers.formatNumber}} results{{/hasManyResults}}\n {{/areHitsSorted}}\n found in {{processingTimeMS}}ms" + text: function text(props) { + return "".concat(props.areHitsSorted ? getSortedResultsSentence(props) : getResultsSentence(props), " found in ").concat(props.processingTimeMS, "ms"); + } }; exports.defaultTemplates = defaultTemplates; -var renderer = function renderer(_ref) { - var renderState = _ref.renderState, - cssClasses = _ref.cssClasses, - containerNode = _ref.containerNode, - templates = _ref.templates; - return function (_ref2, isFirstRendering) { - var hitsPerPage = _ref2.hitsPerPage, - nbHits = _ref2.nbHits, - nbSortedHits = _ref2.nbSortedHits, - areHitsSorted = _ref2.areHitsSorted, - nbPages = _ref2.nbPages, - page = _ref2.page, - processingTimeMS = _ref2.processingTimeMS, - query = _ref2.query, - instantSearchInstance = _ref2.instantSearchInstance; +function getSortedResultsSentence(_ref) { + var nbHits = _ref.nbHits, + hasNoSortedResults = _ref.hasNoSortedResults, + hasOneSortedResults = _ref.hasOneSortedResults, + hasManySortedResults = _ref.hasManySortedResults, + nbSortedHits = _ref.nbSortedHits; + var suffix = "sorted out of ".concat((0, _formatNumber.formatNumber)(nbHits)); + + if (hasNoSortedResults) { + return "No relevant results ".concat(suffix); + } + + if (hasOneSortedResults) { + return "1 relevant result ".concat(suffix); + } + + if (hasManySortedResults) { + return "".concat((0, _formatNumber.formatNumber)(nbSortedHits || 0), " relevant results ").concat(suffix); + } + + return ''; +} + +function getResultsSentence(_ref2) { + var nbHits = _ref2.nbHits, + hasNoResults = _ref2.hasNoResults, + hasOneResult = _ref2.hasOneResult, + hasManyResults = _ref2.hasManyResults; + + if (hasNoResults) { + return 'No results'; + } + + if (hasOneResult) { + return '1 result'; + } + + if (hasManyResults) { + return "".concat((0, _formatNumber.formatNumber)(nbHits), " results"); + } + + return ''; +} + +var renderer = function renderer(_ref3) { + var renderState = _ref3.renderState, + cssClasses = _ref3.cssClasses, + containerNode = _ref3.containerNode, + templates = _ref3.templates; + return function (_ref4, isFirstRendering) { + var hitsPerPage = _ref4.hitsPerPage, + nbHits = _ref4.nbHits, + nbSortedHits = _ref4.nbSortedHits, + areHitsSorted = _ref4.areHitsSorted, + nbPages = _ref4.nbPages, + page = _ref4.page, + processingTimeMS = _ref4.processingTimeMS, + query = _ref4.query, + instantSearchInstance = _ref4.instantSearchInstance; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: defaultTemplates, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -82,12 +132,12 @@ var renderer = function renderer(_ref) { var stats = function stats(widgetParams) { - var _ref3 = widgetParams || {}, - container = _ref3.container, - _ref3$cssClasses = _ref3.cssClasses, - userCssClasses = _ref3$cssClasses === void 0 ? {} : _ref3$cssClasses, - _ref3$templates = _ref3.templates, - templates = _ref3$templates === void 0 ? {} : _ref3$templates; + var _ref5 = widgetParams || {}, + container = _ref5.container, + _ref5$cssClasses = _ref5.cssClasses, + userCssClasses = _ref5$cssClasses === void 0 ? {} : _ref5$cssClasses, + _ref5$templates = _ref5.templates, + templates = _ref5$templates === void 0 ? {} : _ref5$templates; if (!container) { throw new Error(withUsage('The `container` option is required.')); @@ -95,8 +145,8 @@ var stats = function stats(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - text: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + text: (0, _uiComponentsShared.cx)(suit({ descendantName: 'text' }), userCssClasses.text) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/defaultTemplates.js index e1ae8b9..ca4f70c 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/defaultTemplates.js @@ -5,7 +5,10 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; var defaultTemplates = { - labelText: '{{name}}' + labelText: function labelText(_ref) { + var name = _ref.name; + return name; + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/toggle-refinement.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/toggle-refinement.js index 3c3291a..ae7a407 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/toggle-refinement.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/toggle-refinement/toggle-refinement.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _ToggleRefinement = _interopRequireDefault(require("../../components/ToggleRefinement/ToggleRefinement.js")); @@ -17,6 +17,8 @@ var _defaultTemplates = _interopRequireDefault(require("./defaultTemplates.js")) var _index = require("../../lib/utils/index.js"); +var _index2 = require("../../lib/templating/index.js"); + var _suit = require("../../lib/suit.js"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } @@ -43,7 +45,7 @@ var renderer = function renderer(_ref) { instantSearchInstance = _ref2.instantSearchInstance; if (isFirstRendering) { - renderState.templateProps = (0, _index.prepareTemplateProps)({ + renderState.templateProps = (0, _index2.prepareTemplateProps)({ defaultTemplates: _defaultTemplates.default, templatesConfig: instantSearchInstance.templatesConfig, templates: templates @@ -90,14 +92,14 @@ var toggleRefinement = function toggleRefinement(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - label: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + label: (0, _uiComponentsShared.cx)(suit({ descendantName: 'label' }), userCssClasses.label), - checkbox: (0, _classnames.default)(suit({ + checkbox: (0, _uiComponentsShared.cx)(suit({ descendantName: 'checkbox' }), userCssClasses.checkbox), - labelText: (0, _classnames.default)(suit({ + labelText: (0, _uiComponentsShared.cx)(suit({ descendantName: 'labelText' }), userCssClasses.labelText) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/defaultTemplates.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/defaultTemplates.js index 2bcd3a8..805dbe4 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/defaultTemplates.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/defaultTemplates.js @@ -5,22 +5,93 @@ Object.defineProperty(exports, "__esModule", { }); exports.default = void 0; -var getButtonInnerElement = function getButtonInnerElement(status, errorCode, isListening) { +var _preact = require("preact"); + +var _ref2 = (0, _preact.h)(_preact.Fragment, null, (0, _preact.h)("line", { + x1: "1", + y1: "1", + x2: "23", + y2: "23" +}), (0, _preact.h)("path", { + d: "M9 9v3a3 3 0 0 0 5.12 2.12M15 9.34V4a3 3 0 0 0-5.94-.6" +}), (0, _preact.h)("path", { + d: "M17 16.95A7 7 0 0 1 5 12v-2m14 0v2a7 7 0 0 1-.11 1.23" +}), (0, _preact.h)("line", { + x1: "12", + y1: "19", + x2: "12", + y2: "23" +}), (0, _preact.h)("line", { + x1: "8", + y1: "23", + x2: "16", + y2: "23" +})); + +var _ref3 = (0, _preact.h)("path", { + d: "M19 10v2a7 7 0 0 1-14 0v-2" +}); + +var _ref4 = (0, _preact.h)("line", { + x1: "12", + y1: "19", + x2: "12", + y2: "23" +}); + +var _ref5 = (0, _preact.h)("line", { + x1: "8", + y1: "23", + x2: "16", + y2: "23" +}); + +var ButtonInnerElement = function ButtonInnerElement(_ref) { + var status = _ref.status, + errorCode = _ref.errorCode, + isListening = _ref.isListening; + if (status === 'error' && errorCode === 'not-allowed') { - return "\n \n \n \n "; + return _ref2; } - return "\n \n \n \n "); + return (0, _preact.h)(_preact.Fragment, null, (0, _preact.h)("path", { + d: "M12 1a3 3 0 0 0-3 3v8a3 3 0 0 0 6 0V4a3 3 0 0 0-3-3z", + fill: isListening ? 'currentColor' : 'none' + }), _ref3, _ref4, _ref5); }; var defaultTemplates = { - buttonText: function buttonText(_ref) { - var status = _ref.status, - errorCode = _ref.errorCode, - isListening = _ref.isListening; - return "\n ".concat(getButtonInnerElement(status, errorCode, isListening), "\n "); + buttonText: function buttonText(_ref6) { + var status = _ref6.status, + errorCode = _ref6.errorCode, + isListening = _ref6.isListening; + return (0, _preact.h)("svg", { + width: "16", + height: "16", + viewBox: "0 0 24 24", + fill: "none", + stroke: "currentColor" + /* eslint-disable react/no-unknown-property */ + // Preact supports kebab case attributes, and using camel case would + // require using `preact/compat`. + // @TODO: reconsider using the `react` ESLint preset + , + "stroke-width": "2", + "stroke-linecap": "round", + "stroke-linejoin": "round" + /* eslint-enable react/no-unknown-property */ + + }, (0, _preact.h)(ButtonInnerElement, { + status: status, + errorCode: errorCode, + isListening: isListening + })); }, - status: "

{{transcript}}

" + status: function status(_ref7) { + var transcript = _ref7.transcript; + return (0, _preact.h)("p", null, transcript); + } }; var _default = defaultTemplates; exports.default = _default; \ No newline at end of file diff --git a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/voice-search.js b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/voice-search.js index b8c5cc9..837a909 100644 --- a/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/voice-search.js +++ b/wp-search-with-algolia/js/instantsearch.js/cjs/widgets/voice-search/voice-search.js @@ -7,7 +7,7 @@ exports.default = void 0; var _preact = require("preact"); -var _classnames = _interopRequireDefault(require("classnames")); +var _uiComponentsShared = require("@algolia/ui-components-shared"); var _index = require("../../lib/utils/index.js"); @@ -71,11 +71,11 @@ var voiceSearch = function voiceSearch(widgetParams) { var containerNode = (0, _index.getContainerNode)(container); var cssClasses = { - root: (0, _classnames.default)(suit(), userCssClasses.root), - button: (0, _classnames.default)(suit({ + root: (0, _uiComponentsShared.cx)(suit(), userCssClasses.root), + button: (0, _uiComponentsShared.cx)(suit({ descendantName: 'button' }), userCssClasses.button), - status: (0, _classnames.default)(suit({ + status: (0, _uiComponentsShared.cx)(suit({ descendantName: 'status' }), userCssClasses.status) }; diff --git a/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js b/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js index 85a5736..bde6e34 100644 --- a/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js +++ b/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js @@ -1,4 +1,4 @@ -/*! InstantSearch.js 4.40.5 | © Algolia, Inc. and contributors; MIT License | https://github.com/algolia/instantsearch.js */ +/*! InstantSearch.js 4.49.0 | © Algolia, Inc. and contributors; MIT License | https://github.com/algolia/instantsearch.js */ (function (global, factory) { typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : typeof define === 'function' && define.amd ? define(factory) : @@ -297,6 +297,63 @@ throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } + function _createForOfIteratorHelper(o, allowArrayLike) { + var it; + + if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { + if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { + if (it) o = it; + var i = 0; + + var F = function () {}; + + return { + s: F, + n: function () { + if (i >= o.length) return { + done: true + }; + return { + done: false, + value: o[i++] + }; + }, + e: function (e) { + throw e; + }, + f: F + }; + } + + throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); + } + + var normalCompletion = true, + didErr = false, + err; + return { + s: function () { + it = o[Symbol.iterator](); + }, + n: function () { + var step = it.next(); + normalCompletion = step.done; + return step; + }, + e: function (e) { + didErr = true; + err = e; + }, + f: function () { + try { + if (!normalCompletion && it.return != null) it.return(); + } finally { + if (didErr) throw err; + } + } + }; + } + function clone(value) { if (typeof value === 'object' && value !== null) { return _merge(Array.isArray(value) ? [] : {}, value); @@ -2989,7 +3046,7 @@ nextDisjunctiveResult++; }); - // if we have some root level values for hierarchical facets, merge them + // if we have some parent level values for hierarchical facets, merge them state.getRefinedHierarchicalFacets().forEach(function(refinedFacet) { var hierarchicalFacet = state.getHierarchicalFacetByName(refinedFacet); var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet); @@ -3001,47 +3058,49 @@ return; } - var result = results[nextDisjunctiveResult]; - var facets = result && result.facets - ? result.facets - : {}; - Object.keys(facets).forEach(function(dfacet) { - var facetResults = facets[dfacet]; - var position = findIndex(state.hierarchicalFacets, function(f) { - return f.name === hierarchicalFacet.name; - }); - var attributeIndex = findIndex(self.hierarchicalFacets[position], function(f) { - return f.attribute === dfacet; - }); + results.slice(nextDisjunctiveResult).forEach(function(result) { + var facets = result && result.facets + ? result.facets + : {}; - // previous refinements and no results so not able to find it - if (attributeIndex === -1) { - return; - } + Object.keys(facets).forEach(function(dfacet) { + var facetResults = facets[dfacet]; + var position = findIndex(state.hierarchicalFacets, function(f) { + return f.name === hierarchicalFacet.name; + }); + var attributeIndex = findIndex(self.hierarchicalFacets[position], function(f) { + return f.attribute === dfacet; + }); - // when we always get root levels, if the hits refinement is `beers > IPA` (count: 5), - // then the disjunctive values will be `beers` (count: 100), - // but we do not want to display - // | beers (100) - // > IPA (5) - // We want - // | beers (5) - // > IPA (5) - var defaultData = {}; - - if (currentRefinement.length > 0) { - var root = currentRefinement[0].split(separator)[0]; - defaultData[root] = self.hierarchicalFacets[position][attributeIndex].data[root]; - } + // previous refinements and no results so not able to find it + if (attributeIndex === -1) { + return; + } - self.hierarchicalFacets[position][attributeIndex].data = defaultsPure( - defaultData, - facetResults, - self.hierarchicalFacets[position][attributeIndex].data - ); - }); + // when we always get root levels, if the hits refinement is `beers > IPA` (count: 5), + // then the disjunctive values will be `beers` (count: 100), + // but we do not want to display + // | beers (100) + // > IPA (5) + // We want + // | beers (5) + // > IPA (5) + var defaultData = {}; + + if (currentRefinement.length > 0) { + var root = currentRefinement[0].split(separator)[0]; + defaultData[root] = self.hierarchicalFacets[position][attributeIndex].data[root]; + } - nextDisjunctiveResult++; + self.hierarchicalFacets[position][attributeIndex].data = defaultsPure( + defaultData, + facetResults, + self.hierarchicalFacets[position][attributeIndex].data + ); + }); + + nextDisjunctiveResult++; + }); }); // add the excludes @@ -3845,6 +3904,17 @@ var DerivedHelper_1 = DerivedHelper; + function sortObject(obj) { + return Object.keys(obj) + .sort(function(a, b) { + return a.localeCompare(b); + }) + .reduce(function(acc, curr) { + acc[curr] = obj[curr]; + return acc; + }, {}); + } + var requestBuilder = { /** * Get all the queries to send to the client, those queries can used directly @@ -3869,18 +3939,67 @@ }); }); - // maybe more to get the root level of hierarchical facets when activated + // More to get the parent levels of the hierarchical facets when refined state.getRefinedHierarchicalFacets().forEach(function(refinedFacet) { var hierarchicalFacet = state.getHierarchicalFacetByName(refinedFacet); - var currentRefinement = state.getHierarchicalRefinement(refinedFacet); - // if we are deeper than level 0 (starting from `beer > IPA`) - // we want to get the root values var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet); + + // If we are deeper than level 0 (starting from `beer > IPA`) + // we want to get all parent values if (currentRefinement.length > 0 && currentRefinement[0].split(separator).length > 1) { - queries.push({ - indexName: index, - params: requestBuilder._getDisjunctiveFacetSearchParams(state, refinedFacet, true) + // We generate a map of the filters we will use for our facet values queries + var filtersMap = currentRefinement[0].split(separator).slice(0, -1).reduce( + function createFiltersMap(map, segment, level) { + return map.concat({ + attribute: hierarchicalFacet.attributes[level], + value: level === 0 + ? segment + : [map[map.length - 1].value, segment].join(separator) + }); + } + , []); + + filtersMap.forEach(function(filter, level) { + var params = requestBuilder._getDisjunctiveFacetSearchParams( + state, + filter.attribute, + level === 0 + ); + + // Keep facet filters unrelated to current hierarchical attributes + function hasHierarchicalFacetFilter(value) { + return hierarchicalFacet.attributes.some(function(attribute) { + return attribute === value.split(':')[0]; + }); + } + + var filteredFacetFilters = (params.facetFilters || []).reduce(function(acc, facetFilter) { + if (Array.isArray(facetFilter)) { + var filtered = facetFilter.filter(function(filterValue) { + return !hasHierarchicalFacetFilter(filterValue); + }); + + if (filtered.length > 0) { + acc.push(filtered); + } + } + + if (typeof facetFilter === 'string' && !hasHierarchicalFacetFilter(facetFilter)) { + acc.push(facetFilter); + } + + return acc; + }, []); + + var parent = filtersMap[level - 1]; + if (level > 0) { + params.facetFilters = filteredFacetFilters.concat(parent.attribute + ':' + parent.value); + } else { + params.facetFilters = filteredFacetFilters.length > 0 ? filteredFacetFilters : undefined; + } + + queries.push({indexName: index, params: params}); }); } }); @@ -3915,7 +4034,7 @@ additionalParams.numericFilters = numericFilters; } - return merge_1({}, state.getQueryParams(), additionalParams); + return sortObject(merge_1({}, state.getQueryParams(), additionalParams)); }, /** @@ -3930,16 +4049,16 @@ var numericFilters = requestBuilder._getNumericFilters(state, facet); var tagFilters = requestBuilder._getTagFilters(state); var additionalParams = { - hitsPerPage: 1, + hitsPerPage: 0, page: 0, - attributesToRetrieve: [], - attributesToHighlight: [], - attributesToSnippet: [], - tagFilters: tagFilters, analytics: false, clickAnalytics: false }; + if (tagFilters.length > 0) { + additionalParams.tagFilters = tagFilters; + } + var hierarchicalFacet = state.getHierarchicalFacetByName(facet); if (hierarchicalFacet) { @@ -3960,7 +4079,7 @@ additionalParams.facetFilters = facetFilters; } - return merge_1({}, state.getQueryParams(), additionalParams); + return sortObject(merge_1({}, state.getQueryParams(), additionalParams)); }, /** @@ -4153,17 +4272,17 @@ if (typeof maxFacetHits === 'number') { searchForFacetSearchParameters.maxFacetHits = maxFacetHits; } - return merge_1( + return sortObject(merge_1( {}, requestBuilder._getHitsSearchParams(stateForSearchForFacetValues), searchForFacetSearchParameters - ); + )); } }; var requestBuilder_1 = requestBuilder; - var version = '3.8.2'; + var version = '3.11.1'; var escapeFacetValue$3 = escapeFacetValue_1.escapeFacetValue; @@ -4483,20 +4602,51 @@ */ AlgoliaSearchHelper.prototype.searchForFacetValues = function(facet, query, maxFacetHits, userState) { var clientHasSFFV = typeof this.client.searchForFacetValues === 'function'; + var clientHasInitIndex = typeof this.client.initIndex === 'function'; if ( !clientHasSFFV && - typeof this.client.initIndex !== 'function' + !clientHasInitIndex && + typeof this.client.search !== 'function' ) { throw new Error( 'search for facet values (searchable) was called, but this client does not have a function client.searchForFacetValues or client.initIndex(index).searchForFacetValues' ); } + var state = this.state.setQueryParameters(userState || {}); var isDisjunctive = state.isDisjunctiveFacet(facet); var algoliaQuery = requestBuilder_1.getSearchForFacetQuery(facet, query, maxFacetHits, state); this._currentNbQueries++; var self = this; + var searchForFacetValuesPromise; + // newer algoliasearch ^3.27.1 - ~4.0.0 + if (clientHasSFFV) { + searchForFacetValuesPromise = this.client.searchForFacetValues([ + {indexName: state.index, params: algoliaQuery} + ]); + // algoliasearch < 3.27.1 + } else if (clientHasInitIndex) { + searchForFacetValuesPromise = this.client + .initIndex(state.index) + .searchForFacetValues(algoliaQuery); + // algoliasearch ~5.0.0 + } else { + // @MAJOR only use client.search + delete algoliaQuery.facetName; + searchForFacetValuesPromise = this.client + .search([ + { + type: 'facet', + facet: facet, + indexName: state.index, + params: algoliaQuery + } + ]) + .then(function processResponse(response) { + return response.results[0]; + }); + } this.emit('searchForFacetValues', { state: state, @@ -4504,10 +4654,6 @@ query: query }); - var searchForFacetValuesPromise = clientHasSFFV - ? this.client.searchForFacetValues([{indexName: state.index, params: algoliaQuery}]) - : this.client.initIndex(state.index).searchForFacetValues(algoliaQuery); - return searchForFacetValuesPromise.then(function addIsRefined(content) { self._currentNbQueries--; if (self._currentNbQueries === 0) self.emit('searchQueueEmpty'); @@ -5690,1248 +5836,80 @@ return text.toString().charAt(0).toUpperCase() + text.toString().slice(1); } - var nextMicroTask = Promise.resolve(); - - var defer = function defer(callback) { - var progress = null; - var cancelled = false; + function noop() {} - var fn = function fn() { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } + /** + * Logs a warning when this function is called, in development environment only. + */ + var deprecate = function deprecate(fn, message) { + return fn; + }; + /** + * Logs a warning + * This is used to log issues in development environment only. + */ - if (progress !== null) { - return; - } - progress = nextMicroTask.then(function () { - progress = null; + var warn = noop; + /** + * Logs a warning if the condition is not met. + * This is used to log issues in development environment only. + */ - if (cancelled) { - cancelled = false; - return; - } + var _warning = noop; - callback.apply(void 0, args); - }); + { + warn = function warn(message) { + // eslint-disable-next-line no-console + console.warn("[InstantSearch.js]: ".concat(message.trim())); }; - fn.wait = function () { - if (progress === null) { - throw new Error('The deferred function should be called before calling `wait()`'); - } + deprecate = function deprecate(fn, message) { + var hasAlreadyPrinted = false; + return function () { + if (!hasAlreadyPrinted) { + hasAlreadyPrinted = true; + warn(message); + } - return progress; + return fn.apply(void 0, arguments); + }; }; - fn.cancel = function () { - if (progress === null) { + _warning = function warning(condition, message) { + if (condition) { return; } - cancelled = true; - }; + var hasAlreadyPrinted = _warning.cache[message]; - return fn; - }; + if (!hasAlreadyPrinted) { + _warning.cache[message] = true; + warn(message); + } + }; - function isDomElement(object) { - return object instanceof HTMLElement || Boolean(object) && object.nodeType > 0; + _warning.cache = {}; } /** - * Return the container. If it's a string, it is considered a - * css selector and retrieves the first matching element. Otherwise - * test if it validates that it's a correct DOMElement. - * - * @param {string|HTMLElement} selectorOrHTMLElement CSS Selector or container node. - * @return {HTMLElement} Container node - * @throws Error when the type is not correct + * A typed version of Object.keys, to use when looping over a static object + * inspired from https://stackoverflow.com/a/65117465/3185307 */ + var keys = Object.keys; - function getContainerNode(selectorOrHTMLElement) { - var isSelectorString = typeof selectorOrHTMLElement === 'string'; - var domElement = isSelectorString ? document.querySelector(selectorOrHTMLElement) : selectorOrHTMLElement; + // to map them. - if (!isDomElement(domElement)) { - var errorMessage = 'Container must be `string` or `HTMLElement`.'; + function getWidgetNames(connectorName) { + switch (connectorName) { + case 'range': + return []; - if (isSelectorString) { - errorMessage += " Unable to find ".concat(selectorOrHTMLElement); - } + case 'menu': + return ['menu', 'menuSelect']; - throw new Error(errorMessage); - } - - return domElement; - } - - function isSpecialClick(event) { - var isMiddleClick = event.button === 1; - return isMiddleClick || event.altKey || event.ctrlKey || event.metaKey || event.shiftKey; - } - - function uniq(array) { - return array.filter(function (value, index, self) { - return self.indexOf(value) === index; - }); - } - - function prepareTemplates( // can not use = {} here, since the template could have different constraints - defaultTemplates) { - var templates = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; - var allKeys = uniq([].concat(_toConsumableArray(Object.keys(defaultTemplates || {})), _toConsumableArray(Object.keys(templates)))); - return allKeys.reduce(function (config, key) { - var defaultTemplate = defaultTemplates ? defaultTemplates[key] : undefined; - var customTemplate = templates[key]; - var isCustomTemplate = customTemplate !== undefined && customTemplate !== defaultTemplate; - config.templates[key] = isCustomTemplate ? customTemplate // typescript doesn't recognize that this condition asserts customTemplate is defined - : defaultTemplate; - config.useCustomCompileOptions[key] = isCustomTemplate; - return config; - }, { - // eslint-disable-next-line @typescript-eslint/consistent-type-assertions - templates: {}, - // eslint-disable-next-line @typescript-eslint/consistent-type-assertions - useCustomCompileOptions: {} - }); - } - /** - * Prepares an object to be passed to the Template widget - */ - - - function prepareTemplateProps(_ref) { - var defaultTemplates = _ref.defaultTemplates, - templates = _ref.templates, - templatesConfig = _ref.templatesConfig; - var preparedTemplates = prepareTemplates(defaultTemplates, templates); - return _objectSpread2({ - templatesConfig: templatesConfig - }, preparedTemplates); - } - - function createCommonjsModule(fn, module) { - return module = { exports: {} }, fn(module, module.exports), module.exports; - } - - var compiler = createCommonjsModule(function (module, exports) { - /* - * Copyright 2011 Twitter, Inc. - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - - (function (Hogan) { - // Setup regex assignments - // remove whitespace according to Mustache spec - var rIsWhitespace = /\S/, - rQuot = /\"/g, - rNewline = /\n/g, - rCr = /\r/g, - rSlash = /\\/g, - rLineSep = /\u2028/, - rParagraphSep = /\u2029/; - - Hogan.tags = { - '#': 1, '^': 2, '<': 3, '$': 4, - '/': 5, '!': 6, '>': 7, '=': 8, '_v': 9, - '{': 10, '&': 11, '_t': 12 - }; - - Hogan.scan = function scan(text, delimiters) { - var len = text.length, - IN_TEXT = 0, - IN_TAG_TYPE = 1, - IN_TAG = 2, - state = IN_TEXT, - tagType = null, - tag = null, - buf = '', - tokens = [], - seenTag = false, - i = 0, - lineStart = 0, - otag = '{{', - ctag = '}}'; - - function addBuf() { - if (buf.length > 0) { - tokens.push({tag: '_t', text: new String(buf)}); - buf = ''; - } - } - - function lineIsWhitespace() { - var isAllWhitespace = true; - for (var j = lineStart; j < tokens.length; j++) { - isAllWhitespace = - (Hogan.tags[tokens[j].tag] < Hogan.tags['_v']) || - (tokens[j].tag == '_t' && tokens[j].text.match(rIsWhitespace) === null); - if (!isAllWhitespace) { - return false; - } - } - - return isAllWhitespace; - } - - function filterLine(haveSeenTag, noNewLine) { - addBuf(); - - if (haveSeenTag && lineIsWhitespace()) { - for (var j = lineStart, next; j < tokens.length; j++) { - if (tokens[j].text) { - if ((next = tokens[j+1]) && next.tag == '>') { - // set indent to token value - next.indent = tokens[j].text.toString(); - } - tokens.splice(j, 1); - } - } - } else if (!noNewLine) { - tokens.push({tag:'\n'}); - } - - seenTag = false; - lineStart = tokens.length; - } - - function changeDelimiters(text, index) { - var close = '=' + ctag, - closeIndex = text.indexOf(close, index), - delimiters = trim( - text.substring(text.indexOf('=', index) + 1, closeIndex) - ).split(' '); - - otag = delimiters[0]; - ctag = delimiters[delimiters.length - 1]; - - return closeIndex + close.length - 1; - } - - if (delimiters) { - delimiters = delimiters.split(' '); - otag = delimiters[0]; - ctag = delimiters[1]; - } - - for (i = 0; i < len; i++) { - if (state == IN_TEXT) { - if (tagChange(otag, text, i)) { - --i; - addBuf(); - state = IN_TAG_TYPE; - } else { - if (text.charAt(i) == '\n') { - filterLine(seenTag); - } else { - buf += text.charAt(i); - } - } - } else if (state == IN_TAG_TYPE) { - i += otag.length - 1; - tag = Hogan.tags[text.charAt(i + 1)]; - tagType = tag ? text.charAt(i + 1) : '_v'; - if (tagType == '=') { - i = changeDelimiters(text, i); - state = IN_TEXT; - } else { - if (tag) { - i++; - } - state = IN_TAG; - } - seenTag = i; - } else { - if (tagChange(ctag, text, i)) { - tokens.push({tag: tagType, n: trim(buf), otag: otag, ctag: ctag, - i: (tagType == '/') ? seenTag - otag.length : i + ctag.length}); - buf = ''; - i += ctag.length - 1; - state = IN_TEXT; - if (tagType == '{') { - if (ctag == '}}') { - i++; - } else { - cleanTripleStache(tokens[tokens.length - 1]); - } - } - } else { - buf += text.charAt(i); - } - } - } - - filterLine(seenTag, true); - - return tokens; - }; - - function cleanTripleStache(token) { - if (token.n.substr(token.n.length - 1) === '}') { - token.n = token.n.substring(0, token.n.length - 1); - } - } - - function trim(s) { - if (s.trim) { - return s.trim(); - } - - return s.replace(/^\s*|\s*$/g, ''); - } - - function tagChange(tag, text, index) { - if (text.charAt(index) != tag.charAt(0)) { - return false; - } - - for (var i = 1, l = tag.length; i < l; i++) { - if (text.charAt(index + i) != tag.charAt(i)) { - return false; - } - } - - return true; - } - - // the tags allowed inside super templates - var allowedInSuper = {'_t': true, '\n': true, '$': true, '/': true}; - - function buildTree(tokens, kind, stack, customTags) { - var instructions = [], - opener = null, - tail = null, - token = null; - - tail = stack[stack.length - 1]; - - while (tokens.length > 0) { - token = tokens.shift(); - - if (tail && tail.tag == '<' && !(token.tag in allowedInSuper)) { - throw new Error('Illegal content in < super tag.'); - } - - if (Hogan.tags[token.tag] <= Hogan.tags['$'] || isOpener(token, customTags)) { - stack.push(token); - token.nodes = buildTree(tokens, token.tag, stack, customTags); - } else if (token.tag == '/') { - if (stack.length === 0) { - throw new Error('Closing tag without opener: /' + token.n); - } - opener = stack.pop(); - if (token.n != opener.n && !isCloser(token.n, opener.n, customTags)) { - throw new Error('Nesting error: ' + opener.n + ' vs. ' + token.n); - } - opener.end = token.i; - return instructions; - } else if (token.tag == '\n') { - token.last = (tokens.length == 0) || (tokens[0].tag == '\n'); - } - - instructions.push(token); - } - - if (stack.length > 0) { - throw new Error('missing closing tag: ' + stack.pop().n); - } - - return instructions; - } - - function isOpener(token, tags) { - for (var i = 0, l = tags.length; i < l; i++) { - if (tags[i].o == token.n) { - token.tag = '#'; - return true; - } - } - } - - function isCloser(close, open, tags) { - for (var i = 0, l = tags.length; i < l; i++) { - if (tags[i].c == close && tags[i].o == open) { - return true; - } - } - } - - function stringifySubstitutions(obj) { - var items = []; - for (var key in obj) { - items.push('"' + esc(key) + '": function(c,p,t,i) {' + obj[key] + '}'); - } - return "{ " + items.join(",") + " }"; - } - - function stringifyPartials(codeObj) { - var partials = []; - for (var key in codeObj.partials) { - partials.push('"' + esc(key) + '":{name:"' + esc(codeObj.partials[key].name) + '", ' + stringifyPartials(codeObj.partials[key]) + "}"); - } - return "partials: {" + partials.join(",") + "}, subs: " + stringifySubstitutions(codeObj.subs); - } - - Hogan.stringify = function(codeObj, text, options) { - return "{code: function (c,p,i) { " + Hogan.wrapMain(codeObj.code) + " }," + stringifyPartials(codeObj) + "}"; - }; - - var serialNo = 0; - Hogan.generate = function(tree, text, options) { - serialNo = 0; - var context = { code: '', subs: {}, partials: {} }; - Hogan.walk(tree, context); - - if (options.asString) { - return this.stringify(context, text, options); - } - - return this.makeTemplate(context, text, options); - }; - - Hogan.wrapMain = function(code) { - return 'var t=this;t.b(i=i||"");' + code + 'return t.fl();'; - }; - - Hogan.template = Hogan.Template; - - Hogan.makeTemplate = function(codeObj, text, options) { - var template = this.makePartials(codeObj); - template.code = new Function('c', 'p', 'i', this.wrapMain(codeObj.code)); - return new this.template(template, text, this, options); - }; - - Hogan.makePartials = function(codeObj) { - var key, template = {subs: {}, partials: codeObj.partials, name: codeObj.name}; - for (key in template.partials) { - template.partials[key] = this.makePartials(template.partials[key]); - } - for (key in codeObj.subs) { - template.subs[key] = new Function('c', 'p', 't', 'i', codeObj.subs[key]); - } - return template; - }; - - function esc(s) { - return s.replace(rSlash, '\\\\') - .replace(rQuot, '\\\"') - .replace(rNewline, '\\n') - .replace(rCr, '\\r') - .replace(rLineSep, '\\u2028') - .replace(rParagraphSep, '\\u2029'); - } - - function chooseMethod(s) { - return (~s.indexOf('.')) ? 'd' : 'f'; - } - - function createPartial(node, context) { - var prefix = "<" + (context.prefix || ""); - var sym = prefix + node.n + serialNo++; - context.partials[sym] = {name: node.n, partials: {}}; - context.code += 't.b(t.rp("' + esc(sym) + '",c,p,"' + (node.indent || '') + '"));'; - return sym; - } - - Hogan.codegen = { - '#': function(node, context) { - context.code += 'if(t.s(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,1),' + - 'c,p,0,' + node.i + ',' + node.end + ',"' + node.otag + " " + node.ctag + '")){' + - 't.rs(c,p,' + 'function(c,p,t){'; - Hogan.walk(node.nodes, context); - context.code += '});c.pop();}'; - }, - - '^': function(node, context) { - context.code += 'if(!t.s(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,1),c,p,1,0,0,"")){'; - Hogan.walk(node.nodes, context); - context.code += '};'; - }, - - '>': createPartial, - '<': function(node, context) { - var ctx = {partials: {}, code: '', subs: {}, inPartial: true}; - Hogan.walk(node.nodes, ctx); - var template = context.partials[createPartial(node, context)]; - template.subs = ctx.subs; - template.partials = ctx.partials; - }, - - '$': function(node, context) { - var ctx = {subs: {}, code: '', partials: context.partials, prefix: node.n}; - Hogan.walk(node.nodes, ctx); - context.subs[node.n] = ctx.code; - if (!context.inPartial) { - context.code += 't.sub("' + esc(node.n) + '",c,p,i);'; - } - }, - - '\n': function(node, context) { - context.code += write('"\\n"' + (node.last ? '' : ' + i')); - }, - - '_v': function(node, context) { - context.code += 't.b(t.v(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,0)));'; - }, - - '_t': function(node, context) { - context.code += write('"' + esc(node.text) + '"'); - }, - - '{': tripleStache, - - '&': tripleStache - }; - - function tripleStache(node, context) { - context.code += 't.b(t.t(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,0)));'; - } - - function write(s) { - return 't.b(' + s + ');'; - } - - Hogan.walk = function(nodelist, context) { - var func; - for (var i = 0, l = nodelist.length; i < l; i++) { - func = Hogan.codegen[nodelist[i].tag]; - func && func(nodelist[i], context); - } - return context; - }; - - Hogan.parse = function(tokens, text, options) { - options = options || {}; - return buildTree(tokens, '', [], options.sectionTags || []); - }; - - Hogan.cache = {}; - - Hogan.cacheKey = function(text, options) { - return [text, !!options.asString, !!options.disableLambda, options.delimiters, !!options.modelGet].join('||'); - }; - - Hogan.compile = function(text, options) { - options = options || {}; - var key = Hogan.cacheKey(text, options); - var template = this.cache[key]; - - if (template) { - var partials = template.partials; - for (var name in partials) { - delete partials[name].instance; - } - return template; - } - - template = this.generate(this.parse(this.scan(text, options.delimiters), text, options), text, options); - return this.cache[key] = template; - }; - })( exports ); - }); - - var template = createCommonjsModule(function (module, exports) { - - (function (Hogan) { - Hogan.Template = function (codeObj, text, compiler, options) { - codeObj = codeObj || {}; - this.r = codeObj.code || this.r; - this.c = compiler; - this.options = options || {}; - this.text = text || ''; - this.partials = codeObj.partials || {}; - this.subs = codeObj.subs || {}; - this.buf = ''; - }; - - Hogan.Template.prototype = { - // render: replaced by generated code. - r: function (context, partials, indent) { return ''; }, - - // variable escaping - v: hoganEscape, - - // triple stache - t: coerceToString, - - render: function render(context, partials, indent) { - return this.ri([context], partials || {}, indent); - }, - - // render internal -- a hook for overrides that catches partials too - ri: function (context, partials, indent) { - return this.r(context, partials, indent); - }, - - // ensurePartial - ep: function(symbol, partials) { - var partial = this.partials[symbol]; - - // check to see that if we've instantiated this partial before - var template = partials[partial.name]; - if (partial.instance && partial.base == template) { - return partial.instance; - } - - if (typeof template == 'string') { - if (!this.c) { - throw new Error("No compiler available."); - } - template = this.c.compile(template, this.options); - } - - if (!template) { - return null; - } - - // We use this to check whether the partials dictionary has changed - this.partials[symbol].base = template; - - if (partial.subs) { - // Make sure we consider parent template now - if (!partials.stackText) partials.stackText = {}; - for (key in partial.subs) { - if (!partials.stackText[key]) { - partials.stackText[key] = (this.activeSub !== undefined && partials.stackText[this.activeSub]) ? partials.stackText[this.activeSub] : this.text; - } - } - template = createSpecializedPartial(template, partial.subs, partial.partials, - this.stackSubs, this.stackPartials, partials.stackText); - } - this.partials[symbol].instance = template; - - return template; - }, - - // tries to find a partial in the current scope and render it - rp: function(symbol, context, partials, indent) { - var partial = this.ep(symbol, partials); - if (!partial) { - return ''; - } - - return partial.ri(context, partials, indent); - }, - - // render a section - rs: function(context, partials, section) { - var tail = context[context.length - 1]; - - if (!isArray(tail)) { - section(context, partials, this); - return; - } - - for (var i = 0; i < tail.length; i++) { - context.push(tail[i]); - section(context, partials, this); - context.pop(); - } - }, - - // maybe start a section - s: function(val, ctx, partials, inverted, start, end, tags) { - var pass; - - if (isArray(val) && val.length === 0) { - return false; - } - - if (typeof val == 'function') { - val = this.ms(val, ctx, partials, inverted, start, end, tags); - } - - pass = !!val; - - if (!inverted && pass && ctx) { - ctx.push((typeof val == 'object') ? val : ctx[ctx.length - 1]); - } - - return pass; - }, - - // find values with dotted names - d: function(key, ctx, partials, returnFound) { - var found, - names = key.split('.'), - val = this.f(names[0], ctx, partials, returnFound), - doModelGet = this.options.modelGet, - cx = null; - - if (key === '.' && isArray(ctx[ctx.length - 2])) { - val = ctx[ctx.length - 1]; - } else { - for (var i = 1; i < names.length; i++) { - found = findInScope(names[i], val, doModelGet); - if (found !== undefined) { - cx = val; - val = found; - } else { - val = ''; - } - } - } - - if (returnFound && !val) { - return false; - } - - if (!returnFound && typeof val == 'function') { - ctx.push(cx); - val = this.mv(val, ctx, partials); - ctx.pop(); - } - - return val; - }, - - // find values with normal names - f: function(key, ctx, partials, returnFound) { - var val = false, - v = null, - found = false, - doModelGet = this.options.modelGet; - - for (var i = ctx.length - 1; i >= 0; i--) { - v = ctx[i]; - val = findInScope(key, v, doModelGet); - if (val !== undefined) { - found = true; - break; - } - } - - if (!found) { - return (returnFound) ? false : ""; - } - - if (!returnFound && typeof val == 'function') { - val = this.mv(val, ctx, partials); - } - - return val; - }, - - // higher order templates - ls: function(func, cx, partials, text, tags) { - var oldTags = this.options.delimiters; - - this.options.delimiters = tags; - this.b(this.ct(coerceToString(func.call(cx, text)), cx, partials)); - this.options.delimiters = oldTags; - - return false; - }, - - // compile text - ct: function(text, cx, partials) { - if (this.options.disableLambda) { - throw new Error('Lambda features disabled.'); - } - return this.c.compile(text, this.options).render(cx, partials); - }, - - // template result buffering - b: function(s) { this.buf += s; }, - - fl: function() { var r = this.buf; this.buf = ''; return r; }, - - // method replace section - ms: function(func, ctx, partials, inverted, start, end, tags) { - var textSource, - cx = ctx[ctx.length - 1], - result = func.call(cx); - - if (typeof result == 'function') { - if (inverted) { - return true; - } else { - textSource = (this.activeSub && this.subsText && this.subsText[this.activeSub]) ? this.subsText[this.activeSub] : this.text; - return this.ls(result, cx, partials, textSource.substring(start, end), tags); - } - } - - return result; - }, - - // method replace variable - mv: function(func, ctx, partials) { - var cx = ctx[ctx.length - 1]; - var result = func.call(cx); - - if (typeof result == 'function') { - return this.ct(coerceToString(result.call(cx)), cx, partials); - } - - return result; - }, - - sub: function(name, context, partials, indent) { - var f = this.subs[name]; - if (f) { - this.activeSub = name; - f(context, partials, this, indent); - this.activeSub = false; - } - } - - }; - - //Find a key in an object - function findInScope(key, scope, doModelGet) { - var val; - - if (scope && typeof scope == 'object') { - - if (scope[key] !== undefined) { - val = scope[key]; - - // try lookup with get for backbone or similar model data - } else if (doModelGet && scope.get && typeof scope.get == 'function') { - val = scope.get(key); - } - } - - return val; - } - - function createSpecializedPartial(instance, subs, partials, stackSubs, stackPartials, stackText) { - function PartialTemplate() {} PartialTemplate.prototype = instance; - function Substitutions() {} Substitutions.prototype = instance.subs; - var key; - var partial = new PartialTemplate(); - partial.subs = new Substitutions(); - partial.subsText = {}; //hehe. substext. - partial.buf = ''; - - stackSubs = stackSubs || {}; - partial.stackSubs = stackSubs; - partial.subsText = stackText; - for (key in subs) { - if (!stackSubs[key]) stackSubs[key] = subs[key]; - } - for (key in stackSubs) { - partial.subs[key] = stackSubs[key]; - } - - stackPartials = stackPartials || {}; - partial.stackPartials = stackPartials; - for (key in partials) { - if (!stackPartials[key]) stackPartials[key] = partials[key]; - } - for (key in stackPartials) { - partial.partials[key] = stackPartials[key]; - } - - return partial; - } - - var rAmp = /&/g, - rLt = //g, - rApos = /\'/g, - rQuot = /\"/g, - hChars = /[&<>\"\']/; - - function coerceToString(val) { - return String((val === null || val === undefined) ? '' : val); - } - - function hoganEscape(str) { - str = coerceToString(str); - return hChars.test(str) ? - str - .replace(rAmp, '&') - .replace(rLt, '<') - .replace(rGt, '>') - .replace(rApos, ''') - .replace(rQuot, '"') : - str; - } - - var isArray = Array.isArray || function(a) { - return Object.prototype.toString.call(a) === '[object Array]'; - }; - - })( exports ); - }); - - /* - * Copyright 2011 Twitter, Inc. - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - - // This file is for use with Node.js. See dist/ for browser files. - - - compiler.Template = template.Template; - compiler.template = compiler.Template; - var hogan = compiler; - - // We add all our template helper methods to the template as lambdas. Note - // that lambdas in Mustache are supposed to accept a second argument of - // `render` to get the rendered value, not the literal `{{value}}`. But - // this is currently broken (see https://github.com/twitter/hogan.js/issues/222). - function transformHelpersToHogan() { - var helpers = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; - var compileOptions = arguments.length > 1 ? arguments[1] : undefined; - var data = arguments.length > 2 ? arguments[2] : undefined; - return Object.keys(helpers).reduce(function (acc, helperKey) { - return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, helperKey, function () { - var _this = this; - - return function (text) { - var render = function render(value) { - return hogan.compile(value, compileOptions).render(_this); - }; - - return helpers[helperKey].call(data, text, render); - }; - })); - }, {}); - } - - function renderTemplate(_ref) { - var templates = _ref.templates, - templateKey = _ref.templateKey, - compileOptions = _ref.compileOptions, - helpers = _ref.helpers, - data = _ref.data, - bindEvent = _ref.bindEvent; - var template = templates[templateKey]; - - if (typeof template !== 'string' && typeof template !== 'function') { - throw new Error("Template must be 'string' or 'function', was '".concat(_typeof(template), "' (key: ").concat(templateKey, ")")); - } - - if (typeof template === 'function') { - return template(data, bindEvent); - } - - var transformedHelpers = transformHelpersToHogan(helpers, compileOptions, data); - return hogan.compile(template, compileOptions).render(_objectSpread2(_objectSpread2({}, data), {}, { - helpers: transformedHelpers - })).replace(/[ \n\r\t\f\xA0]+/g, function (spaces) { - return spaces.replace(/(^|\xA0+)[^\xA0]+/g, '$1 '); - }).trim(); - } - - // We aren't using the native `Array.prototype.find` because the refactor away from Lodash is not - // published as a major version. - // Relying on the `find` polyfill on user-land, which before was only required for niche use-cases, - // was decided as too risky. - // @MAJOR Replace with the native `Array.prototype.find` method - // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find - function find$1(items, predicate) { - var value; - - for (var i = 0; i < items.length; i++) { - value = items[i]; // inlined for performance: if (Call(predicate, thisArg, [value, i, list])) { - - if (predicate(value, i, items)) { - return value; - } - } - - return undefined; - } - - function unescapeFacetValue$3(value) { - if (typeof value === 'string') { - return value.replace(/^\\-/, '-'); - } - - return value; - } - function escapeFacetValue$4(value) { - if (typeof value === 'number' && value < 0 || typeof value === 'string') { - return String(value).replace(/^-/, '\\-'); - } - - return value; - } - - function getRefinement$1(state, type, attribute, name) { - var resultsFacets = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : []; - var res = { - type: type, - attribute: attribute, - name: name, - escapedValue: escapeFacetValue$4(name) - }; - var facet = find$1(resultsFacets, function (resultsFacet) { - return resultsFacet.name === attribute; - }); - var count; - - if (type === 'hierarchical') { - (function () { - var facetDeclaration = state.getHierarchicalFacetByName(attribute); - var nameParts = name.split(facetDeclaration.separator); - - var getFacetRefinement = function getFacetRefinement(facetData) { - return function (refinementKey) { - return facetData[refinementKey]; - }; - }; - - var _loop = function _loop(i) { - facet = facet && facet.data && find$1(Object.keys(facet.data).map(getFacetRefinement(facet.data)), function (refinement) { - return refinement.name === nameParts[i]; - }); - }; - - for (var i = 0; facet !== undefined && i < nameParts.length; ++i) { - _loop(i); - } - - count = facet && facet.count; - })(); - } else { - count = facet && facet.data && facet.data[res.name]; - } - - if (count !== undefined) { - res.count = count; - } - - if (facet && facet.exhaustive !== undefined) { - res.exhaustive = facet.exhaustive; - } - - return res; - } - - function getRefinements(results, state) { - var includesQuery = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; - var refinements = []; - var _state$facetsRefineme = state.facetsRefinements, - facetsRefinements = _state$facetsRefineme === void 0 ? {} : _state$facetsRefineme, - _state$facetsExcludes = state.facetsExcludes, - facetsExcludes = _state$facetsExcludes === void 0 ? {} : _state$facetsExcludes, - _state$disjunctiveFac = state.disjunctiveFacetsRefinements, - disjunctiveFacetsRefinements = _state$disjunctiveFac === void 0 ? {} : _state$disjunctiveFac, - _state$hierarchicalFa = state.hierarchicalFacetsRefinements, - hierarchicalFacetsRefinements = _state$hierarchicalFa === void 0 ? {} : _state$hierarchicalFa, - _state$numericRefinem = state.numericRefinements, - numericRefinements = _state$numericRefinem === void 0 ? {} : _state$numericRefinem, - _state$tagRefinements = state.tagRefinements, - tagRefinements = _state$tagRefinements === void 0 ? [] : _state$tagRefinements; - Object.keys(facetsRefinements).forEach(function (attribute) { - var refinementNames = facetsRefinements[attribute]; - refinementNames.forEach(function (refinementName) { - refinements.push(getRefinement$1(state, 'facet', attribute, refinementName, results.facets)); - }); - }); - Object.keys(facetsExcludes).forEach(function (attribute) { - var refinementNames = facetsExcludes[attribute]; - refinementNames.forEach(function (refinementName) { - refinements.push({ - type: 'exclude', - attribute: attribute, - name: refinementName, - exclude: true - }); - }); - }); - Object.keys(disjunctiveFacetsRefinements).forEach(function (attribute) { - var refinementNames = disjunctiveFacetsRefinements[attribute]; - refinementNames.forEach(function (refinementName) { - refinements.push(getRefinement$1(state, 'disjunctive', attribute, // We unescape any disjunctive refined values with `unescapeFacetValue` because - // they can be escaped on negative numeric values with `escapeFacetValue`. - unescapeFacetValue$3(refinementName), results.disjunctiveFacets)); - }); - }); - Object.keys(hierarchicalFacetsRefinements).forEach(function (attribute) { - var refinementNames = hierarchicalFacetsRefinements[attribute]; - refinementNames.forEach(function (refinement) { - refinements.push(getRefinement$1(state, 'hierarchical', attribute, refinement, results.hierarchicalFacets)); - }); - }); - Object.keys(numericRefinements).forEach(function (attribute) { - var operators = numericRefinements[attribute]; - Object.keys(operators).forEach(function (operatorOriginal) { - var operator = operatorOriginal; - var valueOrValues = operators[operator]; - var refinementNames = Array.isArray(valueOrValues) ? valueOrValues : [valueOrValues]; - refinementNames.forEach(function (refinementName) { - refinements.push({ - type: 'numeric', - attribute: attribute, - name: "".concat(refinementName), - numericValue: refinementName, - operator: operator - }); - }); - }); - }); - tagRefinements.forEach(function (refinementName) { - refinements.push({ - type: 'tag', - attribute: '_tags', - name: refinementName - }); - }); - - if (includesQuery && state.query && state.query.trim()) { - refinements.push({ - attribute: 'query', - type: 'query', - name: state.query, - query: state.query - }); - } - - return refinements; - } - - /** - * Clears the refinements of a SearchParameters object based on rules provided. - * The included attributes list is applied before the excluded attributes list. If the list - * is not provided, this list of all the currently refined attributes is used as included attributes. - * @param {object} $0 parameters - * @param {Helper} $0.helper instance of the Helper - * @param {string[]} [$0.attributesToClear = []] list of parameters to clear - * @returns {SearchParameters} search parameters with refinements cleared - */ - function clearRefinements(_ref) { - var helper = _ref.helper, - _ref$attributesToClea = _ref.attributesToClear, - attributesToClear = _ref$attributesToClea === void 0 ? [] : _ref$attributesToClea; - var finalState = helper.state.setPage(0); - finalState = attributesToClear.reduce(function (state, attribute) { - if (finalState.isNumericRefined(attribute)) { - return state.removeNumericRefinement(attribute); - } - - if (finalState.isHierarchicalFacet(attribute)) { - return state.removeHierarchicalFacetRefinement(attribute); - } - - if (finalState.isDisjunctiveFacet(attribute)) { - return state.removeDisjunctiveFacetRefinement(attribute); - } - - if (finalState.isConjunctiveFacet(attribute)) { - return state.removeFacetRefinement(attribute); - } - - return state; - }, finalState); - - if (attributesToClear.indexOf('query') !== -1) { - finalState = finalState.setQuery(''); - } - - return finalState; - } - - function getObjectType(object) { - return Object.prototype.toString.call(object).slice(8, -1); - } - - function checkRendering(rendering, usage) { - if (rendering === undefined || typeof rendering !== 'function') { - throw new Error("The render function is not valid (received type ".concat(getObjectType(rendering), ").\n\n").concat(usage)); - } - } - - function noop() {} - - /** - * Logs a warning when this function is called, in development environment only. - */ - var deprecate = function deprecate(fn, message) { - return fn; - }; - /** - * Logs a warning - * This is used to log issues in development environment only. - */ - - - var warn = noop; - /** - * Logs a warning if the condition is not met. - * This is used to log issues in development environment only. - */ - - var _warning = noop; - - { - warn = function warn(message) { - // eslint-disable-next-line no-console - console.warn("[InstantSearch.js]: ".concat(message.trim())); - }; - - deprecate = function deprecate(fn, message) { - var hasAlreadyPrinted = false; - return function () { - if (!hasAlreadyPrinted) { - hasAlreadyPrinted = true; - warn(message); - } - - return fn.apply(void 0, arguments); - }; - }; - - _warning = function warning(condition, message) { - if (condition) { - return; - } - - var hasAlreadyPrinted = _warning.cache[message]; - - if (!hasAlreadyPrinted) { - _warning.cache[message] = true; - warn(message); - } - }; - - _warning.cache = {}; - } - - /** - * A typed version of Object.keys, to use when looping over a static object - * inspired from https://stackoverflow.com/a/65117465/3185307 - */ - var keys = Object.keys; - - // to map them. - - function getWidgetNames(connectorName) { - switch (connectorName) { - case 'range': - return []; - - case 'menu': - return ['menu', 'menuSelect']; - - default: - return [connectorName]; + default: + return [connectorName]; } } @@ -7065,110 +6043,52 @@ }).join(',\n '), "\n]);\n```\n\nIf you're using custom widgets that do set these query parameters, we recommend using connectors instead.\n\nSee https://www.algolia.com/doc/guides/building-search-ui/widgets/customize-an-existing-widget/js/#customize-the-complete-ui-of-the-widgets")) ; } - function getPropertyByPath(object, path) { - var parts = Array.isArray(path) ? path : path.split('.'); - return parts.reduce(function (current, key) { - return current && current[key]; - }, object); - } - - // This is the `Number.isFinite()` polyfill recommended by MDN. - // We do not provide any tests for this function. - // See: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isFinite#Polyfill - function isFiniteNumber(value) { - return typeof value === 'number' && isFinite(value); + function getObjectType(object) { + return Object.prototype.toString.call(object).slice(8, -1); } - /** - * This implementation is taken from Lodash implementation. - * See: https://github.com/lodash/lodash/blob/master/isPlainObject.js - */ - function getTag(value) { - if (value === null) { - return value === undefined ? '[object Undefined]' : '[object Null]'; + function checkRendering(rendering, usage) { + if (rendering === undefined || typeof rendering !== 'function') { + throw new Error("The render function is not valid (received type ".concat(getObjectType(rendering), ").\n\n").concat(usage)); } - - return Object.prototype.toString.call(value); } - function isObjectLike(value) { - return _typeof(value) === 'object' && value !== null; - } /** - * Checks if `value` is a plain object. - * - * A plain object is an object created by the `Object` - * constructor or with a `[[Prototype]]` of `null`. + * Clears the refinements of a SearchParameters object based on rules provided. + * The included attributes list is applied before the excluded attributes list. If the list + * is not provided, this list of all the currently refined attributes is used as included attributes. + * @returns search parameters with refinements cleared */ + function clearRefinements(_ref) { + var helper = _ref.helper, + _ref$attributesToClea = _ref.attributesToClear, + attributesToClear = _ref$attributesToClea === void 0 ? [] : _ref$attributesToClea; + var finalState = helper.state.setPage(0); + finalState = attributesToClear.reduce(function (state, attribute) { + if (finalState.isNumericRefined(attribute)) { + return state.removeNumericRefinement(attribute); + } + if (finalState.isHierarchicalFacet(attribute)) { + return state.removeHierarchicalFacetRefinement(attribute); + } - function isPlainObject(value) { - if (!isObjectLike(value) || getTag(value) !== '[object Object]') { - return false; - } - - if (Object.getPrototypeOf(value) === null) { - return true; - } - - var proto = value; - - while (Object.getPrototypeOf(proto) !== null) { - proto = Object.getPrototypeOf(proto); - } - - return Object.getPrototypeOf(value) === proto; - } - - function range(_ref) { - var _ref$start = _ref.start, - start = _ref$start === void 0 ? 0 : _ref$start, - end = _ref.end, - _ref$step = _ref.step, - step = _ref$step === void 0 ? 1 : _ref$step; - // We can't divide by 0 so we re-assign the step to 1 if it happens. - var limitStep = step === 0 ? 1 : step; // In some cases the array to create has a decimal length. - // We therefore need to round the value. - // Example: - // { start: 1, end: 5000, step: 500 } - // => Array length = (5000 - 1) / 500 = 9.998 - - var arrayLength = Math.round((end - start) / limitStep); - return _toConsumableArray(Array(arrayLength)).map(function (_, current) { - return start + current * limitStep; - }); - } - - function isPrimitive(obj) { - return obj !== Object(obj); - } - - function isEqual(first, second) { - if (first === second) { - return true; - } - - if (isPrimitive(first) || isPrimitive(second) || typeof first === 'function' || typeof second === 'function') { - return first === second; - } - - if (Object.keys(first).length !== Object.keys(second).length) { - return false; - } - - for (var _i = 0, _Object$keys = Object.keys(first); _i < _Object$keys.length; _i++) { - var key = _Object$keys[_i]; - - if (!(key in second)) { - return false; + if (finalState.isDisjunctiveFacet(attribute)) { + return state.removeDisjunctiveFacetRefinement(attribute); } - if (!isEqual(first[key], second[key])) { - return false; + if (finalState.isConjunctiveFacet(attribute)) { + return state.removeFacetRefinement(attribute); } + + return state; + }, finalState); + + if (attributesToClear.indexOf('query') !== -1) { + finalState = finalState.setQuery(''); } - return true; + return finalState; } /** @@ -7176,7 +6096,7 @@ * See: https://github.com/lodash/lodash/blob/4.17.11-npm/escape.js */ // Used to map characters to HTML entities. - var htmlEscapes = { + var htmlEntities = { '&': '&', '<': '<', '>': '>', @@ -7193,16 +6113,16 @@ function escape$1(value) { return value && regexHasUnescapedHtml.test(value) ? value.replace(regexUnescapedHtml, function (character) { - return htmlEscapes[character]; + return htmlEntities[character]; }) : value; } - /** * This implementation is taken from Lodash implementation. * See: https://github.com/lodash/lodash/blob/4.17.11-npm/unescape.js */ // Used to map HTML entities to characters. - var htmlEscapes$1 = { + + var htmlCharacters = { '&': '&', '<': '<', '>': '>', @@ -7217,10 +6137,51 @@ * characters. */ - function unescape$1(value) { - return value && regexHasEscapedHtml.test(value) ? value.replace(regexEscapedHtml, function (character) { - return htmlEscapes$1[character]; - }) : value; + function unescape$1(value) { + return value && regexHasEscapedHtml.test(value) ? value.replace(regexEscapedHtml, function (character) { + return htmlCharacters[character]; + }) : value; + } + + /** + * This implementation is taken from Lodash implementation. + * See: https://github.com/lodash/lodash/blob/master/isPlainObject.js + */ + function getTag(value) { + if (value === null) { + return value === undefined ? '[object Undefined]' : '[object Null]'; + } + + return Object.prototype.toString.call(value); + } + + function isObjectLike(value) { + return _typeof(value) === 'object' && value !== null; + } + /** + * Checks if `value` is a plain object. + * + * A plain object is an object created by the `Object` + * constructor or with a `[[Prototype]]` of `null`. + */ + + + function isPlainObject(value) { + if (!isObjectLike(value) || getTag(value) !== '[object Object]') { + return false; + } + + if (Object.getPrototypeOf(value) === null) { + return true; + } + + var proto = value; + + while (Object.getPrototypeOf(proto) !== null) { + proto = Object.getPrototypeOf(proto); + } + + return Object.getPrototypeOf(value) === proto; } var TAG_PLACEHOLDER = { @@ -7290,224 +6251,364 @@ }).join(''); } - function getHighlightedParts(highlightedValue) { - var highlightPostTag = TAG_REPLACEMENT.highlightPostTag, - highlightPreTag = TAG_REPLACEMENT.highlightPreTag; - var splitByPreTag = highlightedValue.split(highlightPreTag); - var firstValue = splitByPreTag.shift(); - var elements = !firstValue ? [] : [{ - value: firstValue, - isHighlighted: false - }]; - splitByPreTag.forEach(function (split) { - var splitByPostTag = split.split(highlightPostTag); - elements.push({ - value: splitByPostTag[0], - isHighlighted: true + // copied from + // https://github.com/algolia/autocomplete.js/blob/307a7acc4283e10a19cb7d067f04f1bea79dc56f/packages/autocomplete-core/src/utils/createConcurrentSafePromise.ts#L1:L1 + + /** + * Creates a runner that executes promises in a concurrent-safe way. + * + * This is useful to prevent older promises to resolve after a newer promise, + * otherwise resulting in stale resolved values. + */ + function createConcurrentSafePromise() { + var basePromiseId = -1; + var latestResolvedId = -1; + var latestResolvedValue = undefined; + return function runConcurrentSafePromise(promise) { + var currentPromiseId = ++basePromiseId; + return Promise.resolve(promise).then(function (x) { + // The promise might take too long to resolve and get outdated. This would + // result in resolving stale values. + // When this happens, we ignore the promise value and return the one + // coming from the latest resolved value. + // + // +----------------------------------+ + // | 100ms | + // | run(1) +---> R1 | + // | 300ms | + // | run(2) +-------------> R2 (SKIP) | + // | 200ms | + // | run(3) +--------> R3 | + // +----------------------------------+ + if (latestResolvedValue && currentPromiseId < latestResolvedId) { + return latestResolvedValue; + } + + latestResolvedId = currentPromiseId; + latestResolvedValue = x; + return x; }); + }; + } - if (splitByPostTag[1] !== '') { - elements.push({ - value: splitByPostTag[1], - isHighlighted: false - }); + function isFacetRefined(helper, facet, value) { + if (helper.state.isHierarchicalFacet(facet)) { + return helper.state.isHierarchicalFacetRefined(facet, value); + } else if (helper.state.isConjunctiveFacet(facet)) { + return helper.state.isFacetRefined(facet, value); + } else { + return helper.state.isDisjunctiveFacetRefined(facet, value); + } + } + + function createSendEventForFacet(_ref) { + var instantSearchInstance = _ref.instantSearchInstance, + helper = _ref.helper, + attr = _ref.attribute, + widgetType = _ref.widgetType; + + var sendEventForFacet = function sendEventForFacet() { + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; } - }); - return elements; + + var eventType = args[0], + facetValue = args[1], + _args$ = args[2], + eventName = _args$ === void 0 ? 'Filter Applied' : _args$; + var attribute = typeof attr === 'string' ? attr : attr(facetValue); + + if (args.length === 1 && _typeof(args[0]) === 'object') { + instantSearchInstance.sendEventToInsights(args[0]); + } else if (eventType === 'click' && (args.length === 2 || args.length === 3)) { + if (!isFacetRefined(helper, attribute, facetValue)) { + // send event only when the facet is being checked "ON" + instantSearchInstance.sendEventToInsights({ + insightsMethod: 'clickedFilters', + widgetType: widgetType, + eventType: eventType, + payload: { + eventName: eventName, + index: helper.getIndex(), + filters: ["".concat(attribute, ":").concat(facetValue)] + }, + attribute: attribute + }); + } + } else { + throw new Error("You need to pass two arguments like:\n sendEvent('click', facetValue);\n\nIf you want to send a custom payload, you can pass one object: sendEvent(customPayload);\n"); + } + }; + + return sendEventForFacet; } - var hasAlphanumeric = new RegExp(/\w/i); - function getHighlightFromSiblings(parts, i) { - var _parts, _parts2; + function serializePayload(payload) { + return btoa(encodeURIComponent(JSON.stringify(payload))); + } + function deserializePayload(serialized) { + return JSON.parse(decodeURIComponent(atob(serialized))); + } - var current = parts[i]; - var isNextHighlighted = ((_parts = parts[i + 1]) === null || _parts === void 0 ? void 0 : _parts.isHighlighted) || true; - var isPreviousHighlighted = ((_parts2 = parts[i - 1]) === null || _parts2 === void 0 ? void 0 : _parts2.isHighlighted) || true; + function chunk(arr) { + var chunkSize = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 20; + var chunks = []; - if (!hasAlphanumeric.test(unescape$1(current.value)) && isPreviousHighlighted === isNextHighlighted) { - return isPreviousHighlighted; + for (var i = 0; i < Math.ceil(arr.length / chunkSize); i++) { + chunks.push(arr.slice(i * chunkSize, (i + 1) * chunkSize)); } - return current.isHighlighted; + return chunks; } - function reverseHighlightedParts(parts) { - if (!parts.some(function (part) { - return part.isHighlighted; - })) { - return parts.map(function (part) { - return _objectSpread2(_objectSpread2({}, part), {}, { - isHighlighted: false - }); - }); + var buildPayloads = function buildPayloads(_ref) { + var index = _ref.index, + widgetType = _ref.widgetType, + methodName = _ref.methodName, + args = _ref.args, + isSearchStalled = _ref.isSearchStalled; + + // when there's only one argument, that means it's custom + if (args.length === 1 && _typeof(args[0]) === 'object') { + return [args[0]]; } - return parts.map(function (part, i) { - return _objectSpread2(_objectSpread2({}, part), {}, { - isHighlighted: !getHighlightFromSiblings(parts, i) + var eventType = args[0]; + var hits = args[1]; + var eventName = args[2]; + + if (!hits) { + { + throw new Error("You need to pass hit or hits as the second argument like:\n ".concat(methodName, "(eventType, hit);\n ")); + } + } + + if ((eventType === 'click' || eventType === 'conversion') && !eventName) { + { + throw new Error("You need to pass eventName as the third argument for 'click' or 'conversion' events like:\n ".concat(methodName, "('click', hit, 'Product Purchased');\n\n To learn more about event naming: https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/in-depth/clicks-conversions-best-practices/\n ")); + } + } + + var hitsArray = Array.isArray(hits) ? removeEscapedFromHits(hits) : [hits]; + + if (hitsArray.length === 0) { + return []; + } + + var queryID = hitsArray[0].__queryID; + var hitsChunks = chunk(hitsArray); + var objectIDsByChunk = hitsChunks.map(function (batch) { + return batch.map(function (hit) { + return hit.objectID; + }); + }); + var positionsByChunk = hitsChunks.map(function (batch) { + return batch.map(function (hit) { + return hit.__position; }); }); - } - // We aren't using the native `Array.prototype.findIndex` because the refactor away from Lodash is not - // published as a major version. - // Relying on the `findIndex` polyfill on user-land, which before was only required for niche use-cases, - // was decided as too risky. - // @MAJOR Replace with the native `Array.prototype.findIndex` method - // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/findIndex - function findIndex$1(array, comparator) { - if (!Array.isArray(array)) { - return -1; + if (eventType === 'view') { + if (isSearchStalled) { + return []; + } + + return hitsChunks.map(function (batch, i) { + return { + insightsMethod: 'viewedObjectIDs', + widgetType: widgetType, + eventType: eventType, + payload: { + eventName: eventName || 'Hits Viewed', + index: index, + objectIDs: objectIDsByChunk[i] + }, + hits: batch + }; + }); + } else if (eventType === 'click') { + return hitsChunks.map(function (batch, i) { + return { + insightsMethod: 'clickedObjectIDsAfterSearch', + widgetType: widgetType, + eventType: eventType, + payload: { + eventName: eventName, + index: index, + queryID: queryID, + objectIDs: objectIDsByChunk[i], + positions: positionsByChunk[i] + }, + hits: batch + }; + }); + } else if (eventType === 'conversion') { + return hitsChunks.map(function (batch, i) { + return { + insightsMethod: 'convertedObjectIDsAfterSearch', + widgetType: widgetType, + eventType: eventType, + payload: { + eventName: eventName, + index: index, + queryID: queryID, + objectIDs: objectIDsByChunk[i] + }, + hits: batch + }; + }); + } else { + throw new Error("eventType(\"".concat(eventType, "\") is not supported.\n If you want to send a custom payload, you can pass one object: ").concat(methodName, "(customPayload);\n ")); } + }; - for (var i = 0; i < array.length; i++) { - if (comparator(array[i])) { - return i; + function removeEscapedFromHits(hits) { + // remove `hits.__escaped` without mutating + return hits.slice(); + } + + function createSendEventForHits(_ref2) { + var instantSearchInstance = _ref2.instantSearchInstance, + index = _ref2.index, + widgetType = _ref2.widgetType; + + var sendEventForHits = function sendEventForHits() { + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; } - } - return -1; - } + var payloads = buildPayloads({ + widgetType: widgetType, + index: index, + methodName: 'sendEvent', + args: args, + isSearchStalled: instantSearchInstance.status === 'stalled' + }); + payloads.forEach(function (payload) { + return instantSearchInstance.sendEventToInsights(payload); + }); + }; - var mergeWithRest = function mergeWithRest(left, right) { - var facets = right.facets, - disjunctiveFacets = right.disjunctiveFacets, - facetsRefinements = right.facetsRefinements, - facetsExcludes = right.facetsExcludes, - disjunctiveFacetsRefinements = right.disjunctiveFacetsRefinements, - numericRefinements = right.numericRefinements, - tagRefinements = right.tagRefinements, - hierarchicalFacets = right.hierarchicalFacets, - hierarchicalFacetsRefinements = right.hierarchicalFacetsRefinements, - ruleContexts = right.ruleContexts, - rest = _objectWithoutProperties(right, ["facets", "disjunctiveFacets", "facetsRefinements", "facetsExcludes", "disjunctiveFacetsRefinements", "numericRefinements", "tagRefinements", "hierarchicalFacets", "hierarchicalFacetsRefinements", "ruleContexts"]); + return sendEventForHits; + } + function createBindEventForHits(_ref3) { + var index = _ref3.index, + widgetType = _ref3.widgetType; - return left.setQueryParameters(rest); - }; // Merge facets + var bindEventForHits = function bindEventForHits() { + for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) { + args[_key2] = arguments[_key2]; + } + var payloads = buildPayloads({ + widgetType: widgetType, + index: index, + methodName: 'bindEvent', + args: args, + isSearchStalled: false + }); + return payloads.length ? "data-insights-event=".concat(serializePayload(payloads)) : ''; + }; - var mergeFacets = function mergeFacets(left, right) { - return right.facets.reduce(function (_, name) { - return _.addFacet(name); - }, left); - }; + return bindEventForHits; + } - var mergeDisjunctiveFacets = function mergeDisjunctiveFacets(left, right) { - return right.disjunctiveFacets.reduce(function (_, name) { - return _.addDisjunctiveFacet(name); - }, left); - }; + function isIndexWidget(widget) { + return widget.$$type === 'ais.index'; + } - var mergeHierarchicalFacets = function mergeHierarchicalFacets(left, right) { - return left.setQueryParameters({ - hierarchicalFacets: right.hierarchicalFacets.reduce(function (facets, facet) { - var index = findIndex$1(facets, function (_) { - return _.name === facet.name; - }); + function setIndexHelperState(finalUiState, indexWidget) { + var nextIndexUiState = finalUiState[indexWidget.getIndexId()] || {}; - if (index === -1) { - return facets.concat(facet); - } + { + checkIndexUiState({ + index: indexWidget, + indexUiState: nextIndexUiState + }); + } - var nextFacets = facets.slice(); - nextFacets.splice(index, 1, facet); - return nextFacets; - }, left.hierarchicalFacets) + indexWidget.getHelper().setState(indexWidget.getWidgetSearchParameters(indexWidget.getHelper().state, { + uiState: nextIndexUiState + })); + indexWidget.getWidgets().filter(isIndexWidget).forEach(function (widget) { + return setIndexHelperState(finalUiState, widget); }); - }; // Merge facet refinements + } + // Debounce a function call to the trailing edge. + // The debounced function returns a promise. + function debounce(func, wait) { + var lastTimeout = null; + return function () { + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; + } - var mergeTagRefinements = function mergeTagRefinements(left, right) { - return right.tagRefinements.reduce(function (_, value) { - return _.addTagRefinement(value); - }, left); - }; + return new Promise(function (resolve, reject) { + if (lastTimeout) { + clearTimeout(lastTimeout); + } - var mergeFacetRefinements = function mergeFacetRefinements(left, right) { - return left.setQueryParameters({ - facetsRefinements: _objectSpread2(_objectSpread2({}, left.facetsRefinements), right.facetsRefinements) - }); - }; + lastTimeout = setTimeout(function () { + lastTimeout = null; + Promise.resolve(func.apply(void 0, args)).then(resolve).catch(reject); + }, wait); + }); + }; + } - var mergeFacetsExcludes = function mergeFacetsExcludes(left, right) { - return left.setQueryParameters({ - facetsExcludes: _objectSpread2(_objectSpread2({}, left.facetsExcludes), right.facetsExcludes) - }); - }; + var nextMicroTask = Promise.resolve(); + function defer(callback) { + var progress = null; + var cancelled = false; - var mergeDisjunctiveFacetsRefinements = function mergeDisjunctiveFacetsRefinements(left, right) { - return left.setQueryParameters({ - disjunctiveFacetsRefinements: _objectSpread2(_objectSpread2({}, left.disjunctiveFacetsRefinements), right.disjunctiveFacetsRefinements) - }); - }; + var fn = function fn() { + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; + } - var mergeNumericRefinements = function mergeNumericRefinements(left, right) { - return left.setQueryParameters({ - numericRefinements: _objectSpread2(_objectSpread2({}, left.numericRefinements), right.numericRefinements) - }); - }; + if (progress !== null) { + return; + } - var mergeHierarchicalFacetsRefinements = function mergeHierarchicalFacetsRefinements(left, right) { - return left.setQueryParameters({ - hierarchicalFacetsRefinements: _objectSpread2(_objectSpread2({}, left.hierarchicalFacetsRefinements), right.hierarchicalFacetsRefinements) - }); - }; + progress = nextMicroTask.then(function () { + progress = null; - var mergeRuleContexts = function mergeRuleContexts(left, right) { - var ruleContexts = uniq([].concat(left.ruleContexts).concat(right.ruleContexts).filter(Boolean)); + if (cancelled) { + cancelled = false; + return; + } - if (ruleContexts.length > 0) { - return left.setQueryParameters({ - ruleContexts: ruleContexts + callback.apply(void 0, args); }); - } - - return left; - }; - - var merge$1 = function merge() { - for (var _len = arguments.length, parameters = new Array(_len), _key = 0; _key < _len; _key++) { - parameters[_key] = arguments[_key]; - } + }; - return parameters.reduce(function (left, right) { - var hierarchicalFacetsRefinementsMerged = mergeHierarchicalFacetsRefinements(left, right); - var hierarchicalFacetsMerged = mergeHierarchicalFacets(hierarchicalFacetsRefinementsMerged, right); - var tagRefinementsMerged = mergeTagRefinements(hierarchicalFacetsMerged, right); - var numericRefinementsMerged = mergeNumericRefinements(tagRefinementsMerged, right); - var disjunctiveFacetsRefinementsMerged = mergeDisjunctiveFacetsRefinements(numericRefinementsMerged, right); - var facetsExcludesMerged = mergeFacetsExcludes(disjunctiveFacetsRefinementsMerged, right); - var facetRefinementsMerged = mergeFacetRefinements(facetsExcludesMerged, right); - var disjunctiveFacetsMerged = mergeDisjunctiveFacets(facetRefinementsMerged, right); - var ruleContextsMerged = mergeRuleContexts(disjunctiveFacetsMerged, right); - var facetsMerged = mergeFacets(ruleContextsMerged, right); - return mergeWithRest(facetsMerged, right); - }); - }; + fn.wait = function () { + if (progress === null) { + throw new Error('The deferred function should be called before calling `wait()`'); + } - var resolveSearchParameters = function resolveSearchParameters(current) { - var parent = current.getParent(); - var states = [current.getHelper().state]; + return progress; + }; - while (parent !== null) { - states = [parent.getHelper().state].concat(states); - parent = parent.getParent(); - } + fn.cancel = function () { + if (progress === null) { + return; + } - return states; - }; + cancelled = true; + }; - function toArray(value) { - return Array.isArray(value) ? value : [value]; + return fn; } - var createDocumentationLink = function createDocumentationLink(_ref) { + function createDocumentationLink(_ref) { var name = _ref.name, _ref$connector = _ref.connector, connector = _ref$connector === void 0 ? false : _ref$connector; return ['https://www.algolia.com/doc/api-reference/widgets/', name, '/js/', connector ? '#connector' : ''].join(''); - }; - var createDocumentationMessageGenerator = function createDocumentationMessageGenerator() { + } + function createDocumentationMessageGenerator() { for (var _len = arguments.length, widgets = new Array(_len), _key = 0; _key < _len; _key++) { widgets[_key] = arguments[_key]; } @@ -7518,7 +6619,62 @@ return function (message) { return [message, "See documentation: ".concat(links)].filter(Boolean).join('\n\n'); }; - }; + } + + function unescapeFacetValue$3(value) { + if (typeof value === 'string') { + return value.replace(/^\\-/, '-'); + } + + return value; + } + function escapeFacetValue$4(value) { + if (typeof value === 'number' && value < 0 || typeof value === 'string') { + return String(value).replace(/^-/, '\\-'); + } + + return value; + } + + // We aren't using the native `Array.prototype.find` because the refactor away from Lodash is not + // published as a major version. + // Relying on the `find` polyfill on user-land, which before was only required for niche use-cases, + // was decided as too risky. + // @MAJOR Replace with the native `Array.prototype.find` method + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find + function find$1(items, predicate) { + var value; + + for (var i = 0; i < items.length; i++) { + value = items[i]; // inlined for performance: if (Call(predicate, thisArg, [value, i, list])) { + + if (predicate(value, i, items)) { + return value; + } + } + + return undefined; + } + + // We aren't using the native `Array.prototype.findIndex` because the refactor away from Lodash is not + // published as a major version. + // Relying on the `findIndex` polyfill on user-land, which before was only required for niche use-cases, + // was decided as too risky. + // @MAJOR Replace with the native `Array.prototype.findIndex` method + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/findIndex + function findIndex$1(array, comparator) { + if (!Array.isArray(array)) { + return -1; + } + + for (var i = 0; i < array.length; i++) { + if (comparator(array[i])) { + return i; + } + } + + return -1; + } var latLngRegExp = /^(-?\d+(?:\.\d+)?),\s*(-?\d+(?:\.\d+)?)$/; function aroundLatLngToPosition(value) { @@ -7591,12 +6747,267 @@ }; } - function insideBoundingBoxToBoundingBox(value) { - if (Array.isArray(value)) { - return insideBoundingBoxArrayToBoundingBox(value); + function insideBoundingBoxToBoundingBox(value) { + if (Array.isArray(value)) { + return insideBoundingBoxArrayToBoundingBox(value); + } + + return insideBoundingBoxStringToBoundingBox(value); + } + + // typed as any, since it accepts the _real_ js clients, not the interface we otherwise expect + function getAppIdAndApiKey(searchClient) { + if (searchClient.transporter) { + // searchClient v4 + var _searchClient$transpo = searchClient.transporter, + headers = _searchClient$transpo.headers, + queryParameters = _searchClient$transpo.queryParameters; + var APP_ID = 'x-algolia-application-id'; + var API_KEY = 'x-algolia-api-key'; + var appId = headers[APP_ID] || queryParameters[APP_ID]; + var apiKey = headers[API_KEY] || queryParameters[API_KEY]; + return [appId, apiKey]; + } else { + // searchClient v3 + return [searchClient.applicationID, searchClient.apiKey]; + } + } + + function isDomElement(object) { + return object instanceof HTMLElement || Boolean(object) && object.nodeType > 0; + } + + /** + * Return the container. If it's a string, it is considered a + * css selector and retrieves the first matching element. Otherwise + * test if it validates that it's a correct DOMElement. + * + * @param {string|HTMLElement} selectorOrHTMLElement CSS Selector or container node. + * @return {HTMLElement} Container node + * @throws Error when the type is not correct + */ + + function getContainerNode(selectorOrHTMLElement) { + var isSelectorString = typeof selectorOrHTMLElement === 'string'; + var domElement = isSelectorString ? document.querySelector(selectorOrHTMLElement) : selectorOrHTMLElement; + + if (!isDomElement(domElement)) { + var errorMessage = 'Container must be `string` or `HTMLElement`.'; + + if (isSelectorString) { + errorMessage += " Unable to find ".concat(selectorOrHTMLElement); + } + + throw new Error(errorMessage); + } + + return domElement; + } + + function getHighlightedParts(highlightedValue) { + var highlightPostTag = TAG_REPLACEMENT.highlightPostTag, + highlightPreTag = TAG_REPLACEMENT.highlightPreTag; + var splitByPreTag = highlightedValue.split(highlightPreTag); + var firstValue = splitByPreTag.shift(); + var elements = !firstValue ? [] : [{ + value: firstValue, + isHighlighted: false + }]; + splitByPreTag.forEach(function (split) { + var splitByPostTag = split.split(highlightPostTag); + elements.push({ + value: splitByPostTag[0], + isHighlighted: true + }); + + if (splitByPostTag[1] !== '') { + elements.push({ + value: splitByPostTag[1], + isHighlighted: false + }); + } + }); + return elements; + } + + var hasAlphanumeric = new RegExp(/\w/i); + function getHighlightFromSiblings(parts, i) { + var _parts, _parts2; + + var current = parts[i]; + var isNextHighlighted = ((_parts = parts[i + 1]) === null || _parts === void 0 ? void 0 : _parts.isHighlighted) || true; + var isPreviousHighlighted = ((_parts2 = parts[i - 1]) === null || _parts2 === void 0 ? void 0 : _parts2.isHighlighted) || true; + + if (!hasAlphanumeric.test(unescape$1(current.value)) && isPreviousHighlighted === isNextHighlighted) { + return isPreviousHighlighted; + } + + return current.isHighlighted; + } + + function getPropertyByPath(object, path) { + var parts = Array.isArray(path) ? path : path.split('.'); + return parts.reduce(function (current, key) { + return current && current[key]; + }, object); + } + + function getRefinement$1(state, type, attribute, name) { + var resultsFacets = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : []; + var res = { + type: type, + attribute: attribute, + name: name, + escapedValue: escapeFacetValue$4(name) + }; + var facet = find$1(resultsFacets, function (resultsFacet) { + return resultsFacet.name === attribute; + }); + var count; + + if (type === 'hierarchical') { + (function () { + var facetDeclaration = state.getHierarchicalFacetByName(attribute); + var nameParts = name.split(facetDeclaration.separator); + + var getFacetRefinement = function getFacetRefinement(facetData) { + return function (refinementKey) { + return facetData[refinementKey]; + }; + }; + + var _loop = function _loop(i) { + facet = facet && facet.data && find$1(Object.keys(facet.data).map(getFacetRefinement(facet.data)), function (refinement) { + return refinement.name === nameParts[i]; + }); + }; + + for (var i = 0; facet !== undefined && i < nameParts.length; ++i) { + _loop(i); + } + + count = facet && facet.count; + })(); + } else { + count = facet && facet.data && facet.data[res.name]; + } + + if (count !== undefined) { + res.count = count; + } + + if (facet && facet.exhaustive !== undefined) { + res.exhaustive = facet.exhaustive; + } + + return res; + } + + function getRefinements(results, state) { + var includesQuery = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; + var refinements = []; + var _state$facetsRefineme = state.facetsRefinements, + facetsRefinements = _state$facetsRefineme === void 0 ? {} : _state$facetsRefineme, + _state$facetsExcludes = state.facetsExcludes, + facetsExcludes = _state$facetsExcludes === void 0 ? {} : _state$facetsExcludes, + _state$disjunctiveFac = state.disjunctiveFacetsRefinements, + disjunctiveFacetsRefinements = _state$disjunctiveFac === void 0 ? {} : _state$disjunctiveFac, + _state$hierarchicalFa = state.hierarchicalFacetsRefinements, + hierarchicalFacetsRefinements = _state$hierarchicalFa === void 0 ? {} : _state$hierarchicalFa, + _state$numericRefinem = state.numericRefinements, + numericRefinements = _state$numericRefinem === void 0 ? {} : _state$numericRefinem, + _state$tagRefinements = state.tagRefinements, + tagRefinements = _state$tagRefinements === void 0 ? [] : _state$tagRefinements; + Object.keys(facetsRefinements).forEach(function (attribute) { + var refinementNames = facetsRefinements[attribute]; + refinementNames.forEach(function (refinementName) { + refinements.push(getRefinement$1(state, 'facet', attribute, refinementName, results.facets)); + }); + }); + Object.keys(facetsExcludes).forEach(function (attribute) { + var refinementNames = facetsExcludes[attribute]; + refinementNames.forEach(function (refinementName) { + refinements.push({ + type: 'exclude', + attribute: attribute, + name: refinementName, + exclude: true + }); + }); + }); + Object.keys(disjunctiveFacetsRefinements).forEach(function (attribute) { + var refinementNames = disjunctiveFacetsRefinements[attribute]; + refinementNames.forEach(function (refinementName) { + refinements.push(getRefinement$1(state, 'disjunctive', attribute, // We unescape any disjunctive refined values with `unescapeFacetValue` because + // they can be escaped on negative numeric values with `escapeFacetValue`. + unescapeFacetValue$3(refinementName), results.disjunctiveFacets)); + }); + }); + Object.keys(hierarchicalFacetsRefinements).forEach(function (attribute) { + var refinementNames = hierarchicalFacetsRefinements[attribute]; + refinementNames.forEach(function (refinement) { + refinements.push(getRefinement$1(state, 'hierarchical', attribute, refinement, results.hierarchicalFacets)); + }); + }); + Object.keys(numericRefinements).forEach(function (attribute) { + var operators = numericRefinements[attribute]; + Object.keys(operators).forEach(function (operatorOriginal) { + var operator = operatorOriginal; + var valueOrValues = operators[operator]; + var refinementNames = Array.isArray(valueOrValues) ? valueOrValues : [valueOrValues]; + refinementNames.forEach(function (refinementName) { + refinements.push({ + type: 'numeric', + attribute: attribute, + name: "".concat(refinementName), + numericValue: refinementName, + operator: operator + }); + }); + }); + }); + tagRefinements.forEach(function (refinementName) { + refinements.push({ + type: 'tag', + attribute: '_tags', + name: refinementName + }); + }); + + if (includesQuery && state.query && state.query.trim()) { + refinements.push({ + attribute: 'query', + type: 'query', + name: state.query, + query: state.query + }); + } + + return refinements; + } + + function getWidgetAttribute(widget, initOptions) { + var _widget$getWidgetRend; + + var renderState = (_widget$getWidgetRend = widget.getWidgetRenderState) === null || _widget$getWidgetRend === void 0 ? void 0 : _widget$getWidgetRend.call(widget, initOptions); + var attribute = null; + + if (renderState && renderState.widgetParams) { + // casting as widgetParams is checked just before + var widgetParams = renderState.widgetParams; + + if (widgetParams.attribute) { + attribute = widgetParams.attribute; + } else if (Array.isArray(widgetParams.attributes)) { + attribute = widgetParams.attributes[0]; + } + } + + if (typeof attribute !== 'string') { + throw new Error("Could not find the attribute of the widget:\n\n".concat(JSON.stringify(widget), "\n\nPlease check whether the widget's getWidgetRenderState returns widgetParams.attribute correctly.")); } - return insideBoundingBoxStringToBoundingBox(value); + return attribute; } function addAbsolutePosition(hits, page, hitsPerPage) { @@ -7619,353 +7030,265 @@ }); } - function isFacetRefined(helper, facet, value) { - if (helper.state.isHierarchicalFacet(facet)) { - return helper.state.isHierarchicalFacetRefined(facet, value); - } else if (helper.state.isConjunctiveFacet(facet)) { - return helper.state.isFacetRefined(facet, value); - } else { - return helper.state.isDisjunctiveFacetRefined(facet, value); - } + function isPrimitive(obj) { + return obj !== Object(obj); } - function createSendEventForFacet(_ref) { - var instantSearchInstance = _ref.instantSearchInstance, - helper = _ref.helper, - attribute = _ref.attribute, - widgetType = _ref.widgetType; + function isEqual(first, second) { + if (first === second) { + return true; + } - var sendEventForFacet = function sendEventForFacet() { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } + if (isPrimitive(first) || isPrimitive(second) || typeof first === 'function' || typeof second === 'function') { + return first === second; + } - var eventType = args[0], - facetValue = args[1], - _args$ = args[2], - eventName = _args$ === void 0 ? 'Filter Applied' : _args$; + if (Object.keys(first).length !== Object.keys(second).length) { + return false; + } - if (args.length === 1 && _typeof(args[0]) === 'object') { - instantSearchInstance.sendEventToInsights(args[0]); - } else if (eventType === 'click' && (args.length === 2 || args.length === 3)) { - if (!isFacetRefined(helper, attribute, facetValue)) { - // send event only when the facet is being checked "ON" - instantSearchInstance.sendEventToInsights({ - insightsMethod: 'clickedFilters', - widgetType: widgetType, - eventType: eventType, - payload: { - eventName: eventName, - index: helper.getIndex(), - filters: ["".concat(attribute, ":").concat(facetValue)] - }, - attribute: attribute - }); - } - } else { - throw new Error("You need to pass two arguments like:\n sendEvent('click', facetValue);\n\nIf you want to send a custom payload, you can pass one object: sendEvent(customPayload);\n"); + for (var _i = 0, _Object$keys = Object.keys(first); _i < _Object$keys.length; _i++) { + var key = _Object$keys[_i]; + + if (!(key in second)) { + return false; } - }; - return sendEventForFacet; - } + if (!isEqual(first[key], second[key])) { + return false; + } + } - function serializePayload(payload) { - return btoa(encodeURIComponent(JSON.stringify(payload))); - } - function deserializePayload(serialized) { - return JSON.parse(decodeURIComponent(atob(serialized))); + return true; } - function chunk(arr) { - var chunkSize = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 20; - var chunks = []; + // This is the `Number.isFinite()` polyfill recommended by MDN. + // We do not provide any tests for this function. + // See: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isFinite#Polyfill + // @MAJOR Replace with the native `Number.isFinite` method + function isFiniteNumber(value) { + return typeof value === 'number' && isFinite(value); + } - for (var i = 0; i < Math.ceil(arr.length / chunkSize); i++) { - chunks.push(arr.slice(i * chunkSize, (i + 1) * chunkSize)); - } + function isSpecialClick(event) { + var isMiddleClick = event.button === 1; + return isMiddleClick || event.altKey || event.ctrlKey || event.metaKey || event.shiftKey; + } - return chunks; + function uniq(array) { + return array.filter(function (value, index, self) { + return self.indexOf(value) === index; + }); } - var buildPayloads = function buildPayloads(_ref) { - var index = _ref.index, - widgetType = _ref.widgetType, - methodName = _ref.methodName, - args = _ref.args; + var mergeWithRest = function mergeWithRest(left, right) { + var facets = right.facets, + disjunctiveFacets = right.disjunctiveFacets, + facetsRefinements = right.facetsRefinements, + facetsExcludes = right.facetsExcludes, + disjunctiveFacetsRefinements = right.disjunctiveFacetsRefinements, + numericRefinements = right.numericRefinements, + tagRefinements = right.tagRefinements, + hierarchicalFacets = right.hierarchicalFacets, + hierarchicalFacetsRefinements = right.hierarchicalFacetsRefinements, + ruleContexts = right.ruleContexts, + rest = _objectWithoutProperties(right, ["facets", "disjunctiveFacets", "facetsRefinements", "facetsExcludes", "disjunctiveFacetsRefinements", "numericRefinements", "tagRefinements", "hierarchicalFacets", "hierarchicalFacetsRefinements", "ruleContexts"]); - // when there's only one argument, that means it's custom - if (args.length === 1 && _typeof(args[0]) === 'object') { - return [args[0]]; - } + return left.setQueryParameters(rest); + }; // Merge facets - var eventType = args[0]; - var hits = args[1]; - var eventName = args[2]; - if (!hits) { - { - throw new Error("You need to pass hit or hits as the second argument like:\n ".concat(methodName, "(eventType, hit);\n ")); - } - } + var mergeFacets = function mergeFacets(left, right) { + return right.facets.reduce(function (_, name) { + return _.addFacet(name); + }, left); + }; - if ((eventType === 'click' || eventType === 'conversion') && !eventName) { - { - throw new Error("You need to pass eventName as the third argument for 'click' or 'conversion' events like:\n ".concat(methodName, "('click', hit, 'Product Purchased');\n\n To learn more about event naming: https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/in-depth/clicks-conversions-best-practices/\n ")); - } - } + var mergeDisjunctiveFacets = function mergeDisjunctiveFacets(left, right) { + return right.disjunctiveFacets.reduce(function (_, name) { + return _.addDisjunctiveFacet(name); + }, left); + }; - var hitsArray = Array.isArray(hits) ? removeEscapedFromHits(hits) : [hits]; + var mergeHierarchicalFacets = function mergeHierarchicalFacets(left, right) { + return left.setQueryParameters({ + hierarchicalFacets: right.hierarchicalFacets.reduce(function (facets, facet) { + var index = findIndex$1(facets, function (_) { + return _.name === facet.name; + }); - if (hitsArray.length === 0) { - return []; - } + if (index === -1) { + return facets.concat(facet); + } - var queryID = hitsArray[0].__queryID; - var hitsChunks = chunk(hitsArray); - var objectIDsByChunk = hitsChunks.map(function (batch) { - return batch.map(function (hit) { - return hit.objectID; - }); - }); - var positionsByChunk = hitsChunks.map(function (batch) { - return batch.map(function (hit) { - return hit.__position; - }); + var nextFacets = facets.slice(); + nextFacets.splice(index, 1, facet); + return nextFacets; + }, left.hierarchicalFacets) }); + }; // Merge facet refinements - if (eventType === 'view') { - return hitsChunks.map(function (batch, i) { - return { - insightsMethod: 'viewedObjectIDs', - widgetType: widgetType, - eventType: eventType, - payload: { - eventName: eventName || 'Hits Viewed', - index: index, - objectIDs: objectIDsByChunk[i] - }, - hits: batch - }; - }); - } else if (eventType === 'click') { - return hitsChunks.map(function (batch, i) { - return { - insightsMethod: 'clickedObjectIDsAfterSearch', - widgetType: widgetType, - eventType: eventType, - payload: { - eventName: eventName, - index: index, - queryID: queryID, - objectIDs: objectIDsByChunk[i], - positions: positionsByChunk[i] - }, - hits: batch - }; - }); - } else if (eventType === 'conversion') { - return hitsChunks.map(function (batch, i) { - return { - insightsMethod: 'convertedObjectIDsAfterSearch', - widgetType: widgetType, - eventType: eventType, - payload: { - eventName: eventName, - index: index, - queryID: queryID, - objectIDs: objectIDsByChunk[i] - }, - hits: batch - }; - }); - } else { - throw new Error("eventType(\"".concat(eventType, "\") is not supported.\n If you want to send a custom payload, you can pass one object: ").concat(methodName, "(customPayload);\n ")); - } - }; - - function removeEscapedFromHits(hits) { - // remove `hits.__escaped` without mutating - return hits.slice(); - } - function createSendEventForHits(_ref2) { - var instantSearchInstance = _ref2.instantSearchInstance, - index = _ref2.index, - widgetType = _ref2.widgetType; + var mergeTagRefinements = function mergeTagRefinements(left, right) { + return right.tagRefinements.reduce(function (_, value) { + return _.addTagRefinement(value); + }, left); + }; - var sendEventForHits = function sendEventForHits() { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } + var mergeFacetRefinements = function mergeFacetRefinements(left, right) { + return left.setQueryParameters({ + facetsRefinements: _objectSpread2(_objectSpread2({}, left.facetsRefinements), right.facetsRefinements) + }); + }; - var payloads = buildPayloads({ - widgetType: widgetType, - index: index, - methodName: 'sendEvent', - args: args - }); - payloads.forEach(function (payload) { - return instantSearchInstance.sendEventToInsights(payload); - }); - }; + var mergeFacetsExcludes = function mergeFacetsExcludes(left, right) { + return left.setQueryParameters({ + facetsExcludes: _objectSpread2(_objectSpread2({}, left.facetsExcludes), right.facetsExcludes) + }); + }; - return sendEventForHits; - } - function createBindEventForHits(_ref3) { - var index = _ref3.index, - widgetType = _ref3.widgetType; + var mergeDisjunctiveFacetsRefinements = function mergeDisjunctiveFacetsRefinements(left, right) { + return left.setQueryParameters({ + disjunctiveFacetsRefinements: _objectSpread2(_objectSpread2({}, left.disjunctiveFacetsRefinements), right.disjunctiveFacetsRefinements) + }); + }; - var bindEventForHits = function bindEventForHits() { - for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) { - args[_key2] = arguments[_key2]; - } + var mergeNumericRefinements = function mergeNumericRefinements(left, right) { + return left.setQueryParameters({ + numericRefinements: _objectSpread2(_objectSpread2({}, left.numericRefinements), right.numericRefinements) + }); + }; - var payloads = buildPayloads({ - widgetType: widgetType, - index: index, - methodName: 'bindEvent', - args: args - }); - return payloads.length ? "data-insights-event=".concat(serializePayload(payloads)) : ''; - }; + var mergeHierarchicalFacetsRefinements = function mergeHierarchicalFacetsRefinements(left, right) { + return left.setQueryParameters({ + hierarchicalFacetsRefinements: _objectSpread2(_objectSpread2({}, left.hierarchicalFacetsRefinements), right.hierarchicalFacetsRefinements) + }); + }; - return bindEventForHits; - } + var mergeRuleContexts = function mergeRuleContexts(left, right) { + var ruleContexts = uniq([].concat(left.ruleContexts).concat(right.ruleContexts).filter(Boolean)); - // typed as any, since it accepts the _real_ js clients, not the interface we otherwise expect - function getAppIdAndApiKey(searchClient) { - if (searchClient.transporter) { - // searchClient v4 - var _searchClient$transpo = searchClient.transporter, - headers = _searchClient$transpo.headers, - queryParameters = _searchClient$transpo.queryParameters; - var APP_ID = 'x-algolia-application-id'; - var API_KEY = 'x-algolia-api-key'; - var appId = headers[APP_ID] || queryParameters[APP_ID]; - var apiKey = headers[API_KEY] || queryParameters[API_KEY]; - return [appId, apiKey]; - } else { - // searchClient v3 - return [searchClient.applicationID, searchClient.apiKey]; + if (ruleContexts.length > 0) { + return left.setQueryParameters({ + ruleContexts: ruleContexts + }); } - } - function convertNumericRefinementsToFilters(state, attribute) { - if (!state) { - return null; + return left; + }; + + var mergeSearchParameters = function mergeSearchParameters() { + for (var _len = arguments.length, parameters = new Array(_len), _key = 0; _key < _len; _key++) { + parameters[_key] = arguments[_key]; } - var filtersObj = state.numericRefinements[attribute]; - /* - filtersObj === { - "<=": [10], - "=": [], - ">=": [5] - } - */ - - var filters = []; - Object.keys(filtersObj).filter(function (operator) { - return Array.isArray(filtersObj[operator]) && filtersObj[operator].length > 0; - }).forEach(function (operator) { - filtersObj[operator].forEach(function (value) { - filters.push("".concat(attribute).concat(operator).concat(value)); - }); + return parameters.reduce(function (left, right) { + var hierarchicalFacetsRefinementsMerged = mergeHierarchicalFacetsRefinements(left, right); + var hierarchicalFacetsMerged = mergeHierarchicalFacets(hierarchicalFacetsRefinementsMerged, right); + var tagRefinementsMerged = mergeTagRefinements(hierarchicalFacetsMerged, right); + var numericRefinementsMerged = mergeNumericRefinements(tagRefinementsMerged, right); + var disjunctiveFacetsRefinementsMerged = mergeDisjunctiveFacetsRefinements(numericRefinementsMerged, right); + var facetsExcludesMerged = mergeFacetsExcludes(disjunctiveFacetsRefinementsMerged, right); + var facetRefinementsMerged = mergeFacetRefinements(facetsExcludesMerged, right); + var disjunctiveFacetsMerged = mergeDisjunctiveFacets(facetRefinementsMerged, right); + var ruleContextsMerged = mergeRuleContexts(disjunctiveFacetsMerged, right); + var facetsMerged = mergeFacets(ruleContextsMerged, right); + return mergeWithRest(facetsMerged, right); }); - return filters; - } + }; - // copied from - // https://github.com/algolia/autocomplete.js/blob/307a7acc4283e10a19cb7d067f04f1bea79dc56f/packages/autocomplete-core/src/utils/createConcurrentSafePromise.ts#L1:L1 + function range(_ref) { + var _ref$start = _ref.start, + start = _ref$start === void 0 ? 0 : _ref$start, + end = _ref.end, + _ref$step = _ref.step, + step = _ref$step === void 0 ? 1 : _ref$step; + // We can't divide by 0 so we re-assign the step to 1 if it happens. + var limitStep = step === 0 ? 1 : step; // In some cases the array to create has a decimal length. + // We therefore need to round the value. + // Example: + // { start: 1, end: 5000, step: 500 } + // => Array length = (5000 - 1) / 500 = 9.998 - /** - * Creates a runner that executes promises in a concurrent-safe way. - * - * This is useful to prevent older promises to resolve after a newer promise, - * otherwise resulting in stale resolved values. - */ - function createConcurrentSafePromise() { - var basePromiseId = -1; - var latestResolvedId = -1; - var latestResolvedValue = undefined; - return function runConcurrentSafePromise(promise) { - var currentPromiseId = ++basePromiseId; - return Promise.resolve(promise).then(function (x) { - // The promise might take too long to resolve and get outdated. This would - // result in resolving stale values. - // When this happens, we ignore the promise value and return the one - // coming from the latest resolved value. - // - // +----------------------------------+ - // | 100ms | - // | run(1) +---> R1 | - // | 300ms | - // | run(2) +-------------> R2 (SKIP) | - // | 200ms | - // | run(3) +--------> R3 | - // +----------------------------------+ - if (latestResolvedValue && currentPromiseId < latestResolvedId) { - return latestResolvedValue; - } + var arrayLength = Math.round((end - start) / limitStep); + return _toConsumableArray(Array(arrayLength)).map(function (_, current) { + return start + current * limitStep; + }); + } - latestResolvedId = currentPromiseId; - latestResolvedValue = x; - return x; - }); + function createInitArgs(instantSearchInstance, parent, uiState) { + var helper = parent.getHelper(); + return { + uiState: uiState, + helper: helper, + parent: parent, + instantSearchInstance: instantSearchInstance, + state: helper.state, + renderState: instantSearchInstance.renderState, + templatesConfig: instantSearchInstance.templatesConfig, + createURL: parent.createURL, + scopedResults: [], + searchMetadata: { + isSearchStalled: instantSearchInstance.status === 'stalled' + }, + status: instantSearchInstance.status, + error: instantSearchInstance.error }; } - - // Debounce a function call to the trailing edge. - // The debounced function returns a promise. - function debounce(func, wait) { - var lastTimeout = null; - return function () { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } - - return new Promise(function (resolve, reject) { - if (lastTimeout) { - clearTimeout(lastTimeout); - } - - lastTimeout = setTimeout(function () { - lastTimeout = null; - Promise.resolve(func.apply(void 0, args)).then(resolve).catch(reject); - }, wait); - }); + function createRenderArgs(instantSearchInstance, parent) { + var results = parent.getResults(); + return { + helper: parent.getHelper(), + parent: parent, + instantSearchInstance: instantSearchInstance, + results: results, + scopedResults: parent.getScopedResults(), + state: results._state, + renderState: instantSearchInstance.renderState, + templatesConfig: instantSearchInstance.templatesConfig, + createURL: parent.createURL, + searchMetadata: { + isSearchStalled: instantSearchInstance.status === 'stalled' + }, + status: instantSearchInstance.status, + error: instantSearchInstance.error }; } - function getWidgetAttribute(widget, initOptions) { - var _widget$getWidgetRend; - - var renderState = (_widget$getWidgetRend = widget.getWidgetRenderState) === null || _widget$getWidgetRend === void 0 ? void 0 : _widget$getWidgetRend.call(widget, initOptions); - var attribute = null; - - if (renderState && renderState.widgetParams) { - // casting as widgetParams is checked just before - var widgetParams = renderState.widgetParams; + function resolveSearchParameters(current) { + var parent = current.getParent(); + var states = [current.getHelper().state]; - if (widgetParams.attribute) { - attribute = widgetParams.attribute; - } else if (Array.isArray(widgetParams.attributes)) { - attribute = widgetParams.attributes[0]; - } + while (parent !== null) { + states = [parent.getHelper().state].concat(states); + parent = parent.getParent(); } - if (typeof attribute !== 'string') { - throw new Error("Could not find the attribute of the widget:\n\n".concat(JSON.stringify(widget), "\n\nPlease check whether the widget's getWidgetRenderState returns widgetParams.attribute correctly.")); + return states; + } + + function reverseHighlightedParts(parts) { + if (!parts.some(function (part) { + return part.isHighlighted; + })) { + return parts.map(function (part) { + return _objectSpread2(_objectSpread2({}, part), {}, { + isHighlighted: false + }); + }); } - return attribute; + return parts.map(function (part, i) { + return _objectSpread2(_objectSpread2({}, part), {}, { + isHighlighted: !getHighlightFromSiblings(parts, i) + }); + }); } // eslint-disable-next-line no-restricted-globals /** - * Runs code on browser enviromnents safely. + * Runs code on browser environments safely. */ function safelyRunOnBrowser(callback) { var _ref = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : { @@ -7986,18 +7309,19 @@ }); } + function toArray(value) { + return Array.isArray(value) ? value : [value]; + } + var withUsage = createDocumentationMessageGenerator({ name: 'index-widget' }); - function isIndexWidget(widget) { - return widget.$$type === 'ais.index'; - } + /** * This is the same content as helper._change / setState, but allowing for extra * UiState to be synchronized. * see: https://github.com/algolia/algoliasearch-helper-js/blob/6b835ffd07742f2d6b314022cce6848f5cfecd4a/src/algoliasearch.helper.js#L1311-L1324 */ - function privateHelperSetState(helper, _ref) { var state = _ref.state, isPageReset = _ref.isPageReset, @@ -8151,20 +7475,7 @@ widgets.forEach(function (widget) { if (widget.getRenderState) { - var renderState = widget.getRenderState(localInstantSearchInstance.renderState[_this.getIndexId()] || {}, { - uiState: localInstantSearchInstance._initialUiState, - helper: _this.getHelper(), - parent: _this, - instantSearchInstance: localInstantSearchInstance, - state: helper.state, - renderState: localInstantSearchInstance.renderState, - templatesConfig: localInstantSearchInstance.templatesConfig, - createURL: _this.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: localInstantSearchInstance._isSearchStalled - } - }); + var renderState = widget.getRenderState(localInstantSearchInstance.renderState[_this.getIndexId()] || {}, createInitArgs(localInstantSearchInstance, _this, localInstantSearchInstance._initialUiState)); storeRenderState({ renderState: renderState, instantSearchInstance: localInstantSearchInstance, @@ -8174,20 +7485,7 @@ }); widgets.forEach(function (widget) { if (widget.init) { - widget.init({ - helper: helper, - parent: _this, - uiState: localInstantSearchInstance._initialUiState, - instantSearchInstance: localInstantSearchInstance, - state: helper.state, - renderState: localInstantSearchInstance.renderState, - templatesConfig: localInstantSearchInstance.templatesConfig, - createURL: _this.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: localInstantSearchInstance._isSearchStalled - } - }); + widget.init(createInitArgs(localInstantSearchInstance, _this, localInstantSearchInstance._initialUiState)); } }); localInstantSearchInstance.scheduleSearch(); @@ -8276,7 +7574,9 @@ if (instantSearchInstance.onStateChange) { instantSearchInstance.onStateChange({ uiState: instantSearchInstance.mainIndex.getWidgetUiState({}), - setUiState: instantSearchInstance.setUiState.bind(instantSearchInstance) + setUiState: function setUiState(nextState) { + return instantSearchInstance.setUiState(nextState, false); + } }); // We don't trigger a search when controlled because it becomes the // responsibility of `setUiState`. @@ -8297,7 +7597,7 @@ }; derivedHelper = mainHelper.derive(function () { - return merge$1.apply(void 0, _toConsumableArray(resolveSearchParameters(_this3))); + return mergeSearchParameters.apply(void 0, _toConsumableArray(resolveSearchParameters(_this3))); }); var indexInitialResults = (_instantSearchInstanc = instantSearchInstance._initialResults) === null || _instantSearchInstanc === void 0 ? void 0 : _instantSearchInstanc[this.getIndexId()]; @@ -8351,20 +7651,7 @@ localWidgets.forEach(function (widget) { if (widget.getRenderState) { - var renderState = widget.getRenderState(instantSearchInstance.renderState[_this3.getIndexId()] || {}, { - uiState: uiState, - helper: helper, - parent: _this3, - instantSearchInstance: instantSearchInstance, - state: helper.state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this3.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + var renderState = widget.getRenderState(instantSearchInstance.renderState[_this3.getIndexId()] || {}, createInitArgs(instantSearchInstance, _this3, uiState)); storeRenderState({ renderState: renderState, instantSearchInstance: instantSearchInstance, @@ -8378,20 +7665,7 @@ !widget.getWidgetState || Boolean(widget.getWidgetUiState), 'The `getWidgetState` method is renamed `getWidgetUiState` and will no longer exist under that name in InstantSearch.js 5.x. Please use `getWidgetUiState` instead.') ; if (widget.init) { - widget.init({ - uiState: uiState, - helper: helper, - parent: _this3, - instantSearchInstance: instantSearchInstance, - state: helper.state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this3.createURL, - scopedResults: [], - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + widget.init(createInitArgs(instantSearchInstance, _this3, uiState)); } }); // Subscribe to the Helper state changes for the `uiState` once widgets // are initialized. Until the first render, state changes are part of the @@ -8432,20 +7706,7 @@ localWidgets.forEach(function (widget) { if (widget.getRenderState) { - var renderState = widget.getRenderState(instantSearchInstance.renderState[_this4.getIndexId()] || {}, { - helper: _this4.getHelper(), - parent: _this4, - instantSearchInstance: instantSearchInstance, - results: _this4.getResults(), - scopedResults: _this4.getScopedResults(), - state: _this4.getResults()._state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this4.createURL, - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + var renderState = widget.getRenderState(instantSearchInstance.renderState[_this4.getIndexId()] || {}, createRenderArgs(instantSearchInstance, _this4)); storeRenderState({ renderState: renderState, instantSearchInstance: instantSearchInstance, @@ -8461,20 +7722,7 @@ // be delayed. The render is triggered for the complete tree but some parts do // not have results yet. if (widget.render) { - widget.render({ - helper: helper, - parent: _this4, - instantSearchInstance: instantSearchInstance, - results: _this4.getResults(), - scopedResults: _this4.getScopedResults(), - state: _this4.getResults()._state, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - createURL: _this4.createURL, - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }); + widget.render(createRenderArgs(instantSearchInstance, _this4)); } }); }, @@ -8506,7 +7754,7 @@ getWidgetUiState: function getWidgetUiState(uiState) { return localWidgets.filter(isIndexWidget).reduce(function (previousUiState, innerIndex) { return innerIndex.getWidgetUiState(previousUiState); - }, _objectSpread2(_objectSpread2({}, uiState), {}, _defineProperty({}, this.getIndexId(), localUiState))); + }, _objectSpread2(_objectSpread2({}, uiState), {}, _defineProperty({}, indexId, _objectSpread2(_objectSpread2({}, uiState[indexId]), localUiState)))); }, getWidgetState: function getWidgetState(uiState) { _warning(false, 'The `getWidgetState` method is renamed `getWidgetUiState` and will no longer exist under that name in InstantSearch.js 5.x. Please use `getWidgetUiState` instead.') ; @@ -8536,7 +7784,7 @@ instantSearchInstance.renderState = _objectSpread2(_objectSpread2({}, instantSearchInstance.renderState), {}, _defineProperty({}, parentIndexName, _objectSpread2(_objectSpread2({}, instantSearchInstance.renderState[parentIndexName]), renderState))); } - var version$1 = '4.40.5'; + var version$1 = '4.49.0'; var NAMESPACE = 'ais'; var component = function component(componentName) { @@ -8721,11 +7969,15 @@ return getCookie(ANONYMOUS_TOKEN_COOKIE_KEY); } + function formatNumber(value, numberLocale) { + return value.toLocaleString(numberLocale); + } + function hoganHelpers(_ref) { var numberLocale = _ref.numberLocale; return { - formatNumber: function formatNumber(value, render) { - return Number(render(value)).toLocaleString(numberLocale); + formatNumber: function formatNumber$1(value, render) { + return formatNumber(Number(render(value)), numberLocale); }, highlight: function highlight$1(options, render) { try { @@ -8873,7 +8125,7 @@ return obj; }; - var merge$2 = function merge(target, source, options) { + var merge$1 = function merge(target, source, options) { /* eslint no-param-reassign: 0 */ if (!source) { return target; @@ -9076,7 +8328,7 @@ isBuffer: isBuffer, isRegExp: isRegExp, maybeMap: maybeMap, - merge: merge$2 + merge: merge$1 }; var has$1 = Object.prototype.hasOwnProperty; @@ -9921,6 +9173,7 @@ instantSearchInstance.setUiState(stateMapping.routeToState(route)); }); }, + started: function started() {}, unsubscribe: function unsubscribe() { router.dispose(); } @@ -9929,21 +9182,7 @@ }; function extractPayload(widgets, instantSearchInstance, payload) { - var parent = instantSearchInstance.mainIndex; - var initOptions = { - instantSearchInstance: instantSearchInstance, - parent: parent, - scopedResults: [], - state: parent.getHelper().state, - helper: parent.getHelper(), - createURL: parent.createURL, - uiState: instantSearchInstance._initialUiState, - renderState: instantSearchInstance.renderState, - templatesConfig: instantSearchInstance.templatesConfig, - searchMetadata: { - isSearchStalled: instantSearchInstance._isSearchStalled - } - }; + var initOptions = createInitArgs(instantSearchInstance, instantSearchInstance.mainIndex, instantSearchInstance._initialUiState); widgets.forEach(function (widget) { var widgetParams = {}; @@ -10013,6 +9252,7 @@ refNode.appendChild(payloadContainer); }, 0); }, + started: function started() {}, unsubscribe: function unsubscribe() { payloadContainer.remove(); } @@ -10026,10 +9266,9 @@ function defaultCreateURL() { return '#'; - } - /** - * Global options for an InstantSearch instance. - */ + } // this purposely breaks typescript's type inference to ensure it's not used + // as it's used for a default parameter for example + // source: https://github.com/Microsoft/TypeScript/issues/14829#issuecomment-504042546 /** @@ -10047,7 +9286,7 @@ _classCallCheck(this, InstantSearch); - _this = _super.call(this); + _this = _super.call(this); // prevent `render` event listening from causing a warning _defineProperty(_assertThisInitialized(_this), "client", void 0); @@ -10073,8 +9312,6 @@ _defineProperty(_assertThisInitialized(_this), "_searchStalledTimer", void 0); - _defineProperty(_assertThisInitialized(_this), "_isSearchStalled", void 0); - _defineProperty(_assertThisInitialized(_this), "_initialUiState", void 0); _defineProperty(_assertThisInitialized(_this), "_initialResults", void 0); @@ -10089,6 +9326,10 @@ _defineProperty(_assertThisInitialized(_this), "sendEventToInsights", void 0); + _defineProperty(_assertThisInitialized(_this), "status", 'idle'); + + _defineProperty(_assertThisInitialized(_this), "error", undefined); + _defineProperty(_assertThisInitialized(_this), "scheduleSearch", defer(function () { if (_this.started) { _this.mainHelper.search(); @@ -10096,10 +9337,16 @@ })); _defineProperty(_assertThisInitialized(_this), "scheduleRender", defer(function () { + var shouldResetStatus = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : true; + if (!_this.mainHelper.hasPendingRequests()) { clearTimeout(_this._searchStalledTimer); _this._searchStalledTimer = null; - _this._isSearchStalled = false; + + if (shouldResetStatus) { + _this.status = 'idle'; + _this.error = undefined; + } } _this.mainIndex.render({ @@ -10120,6 +9367,8 @@ }); })); + _this.setMaxListeners(100); + var _options$indexName = options.indexName, indexName = _options$indexName === void 0 ? null : _options$indexName, numberLocale = options.numberLocale, @@ -10180,7 +9429,6 @@ }; _this._stalledSearchDelay = stalledSearchDelay; _this._searchStalledTimer = null; - _this._isSearchStalled = false; _this._createURL = defaultCreateURL; _this._initialUiState = initialUiState; _this._initialResults = null; @@ -10209,6 +9457,25 @@ _createClass(InstantSearch, [{ + key: "_isSearchStalled", + get: + /** + * The status of the search. Can be "idle", "loading", "stalled", or "error". + */ + + /** + * The last returned error from the Search API. + * The error gets cleared when the next valid search response is rendered. + */ + + /** + * @deprecated use `status === 'stalled'` instead + */ + function get() { + _warning(false, "`InstantSearch._isSearchStalled` is deprecated and will be removed in InstantSearch.js 5.0.\n\nUse `InstantSearch.status === \"stalled\"` instead.") ; + return this.status === 'stalled'; + } + }, { key: "use", value: function use() { var _this2 = this; @@ -10220,6 +9487,7 @@ var newMiddlewareList = middleware.map(function (fn) { var newMiddleware = _objectSpread2({ subscribe: noop, + started: noop, unsubscribe: noop, onStateChange: noop }, fn({ @@ -10238,6 +9506,7 @@ if (this.started) { newMiddlewareList.forEach(function (m) { m.subscribe(); + m.started(); }); } @@ -10298,316 +9567,770 @@ throw new Error(withUsage$1('The `addWidgets` method expects an array of widgets. Please use `addWidget`.')); } - if (widgets.some(function (widget) { - return typeof widget.init !== 'function' && typeof widget.render !== 'function'; - })) { - throw new Error(withUsage$1('The widget definition expects a `render` and/or an `init` method.')); + if (widgets.some(function (widget) { + return typeof widget.init !== 'function' && typeof widget.render !== 'function'; + })) { + throw new Error(withUsage$1('The widget definition expects a `render` and/or an `init` method.')); + } + + this.mainIndex.addWidgets(widgets); + return this; + } + /** + * Removes a widget from the search instance. + * @deprecated This method will still be supported in 4.x releases, but not further. It is replaced by `removeWidgets([widget])` + * @param widget The widget instance to remove from InstantSearch. + * + * The widget must implement a `dispose()` method to clear its state. + */ + + }, { + key: "removeWidget", + value: function removeWidget(widget) { + _warning(false, 'removeWidget will still be supported in 4.x releases, but not further. It is replaced by `removeWidgets([widget])`') ; + return this.removeWidgets([widget]); + } + /** + * Removes multiple widgets from the search instance. + * @param widgets Array of widgets instances to remove from InstantSearch. + * + * The widgets must implement a `dispose()` method to clear their states. + */ + + }, { + key: "removeWidgets", + value: function removeWidgets(widgets) { + if (!Array.isArray(widgets)) { + throw new Error(withUsage$1('The `removeWidgets` method expects an array of widgets. Please use `removeWidget`.')); + } + + if (widgets.some(function (widget) { + return typeof widget.dispose !== 'function'; + })) { + throw new Error(withUsage$1('The widget definition expects a `dispose` method.')); + } + + this.mainIndex.removeWidgets(widgets); + return this; + } + /** + * Ends the initialization of InstantSearch.js and triggers the + * first search. This method should be called after all widgets have been added + * to the instance of InstantSearch.js. InstantSearch.js also supports adding and removing + * widgets after the start as an **EXPERIMENTAL** feature. + */ + + }, { + key: "start", + value: function start() { + var _this3 = this; + + if (this.started) { + throw new Error(withUsage$1('The `start` method has already been called once.')); + } // This Helper is used for the queries, we don't care about its state. The + // states are managed at the `index` level. We use this Helper to create + // DerivedHelper scoped into the `index` widgets. + // In Vue InstantSearch' hydrate, a main helper gets set before start, so + // we need to respect this helper as a way to keep all listeners correct. + + + var mainHelper = this.mainHelper || algoliasearchHelper_1(this.client, this.indexName); + + mainHelper.search = function () { + _this3.status = 'loading'; // @MAJOR: use scheduleRender here + // For now, widgets don't expect to be rendered at the start of `loading`, + // so it would be a breaking change to add an extra render. We don't have + // these guarantees about the render event, thus emitting it once more + // isn't a breaking change. + + _this3.emit('render'); // This solution allows us to keep the exact same API for the users but + // under the hood, we have a different implementation. It should be + // completely transparent for the rest of the codebase. Only this module + // is impacted. + + + return mainHelper.searchOnlyWithDerivedHelpers(); + }; + + if (this._searchFunction) { + // this client isn't used to actually search, but required for the helper + // to not throw errors + var fakeClient = { + search: function search() { + return new Promise(noop); + } + }; + this._mainHelperSearch = mainHelper.search.bind(mainHelper); + + mainHelper.search = function () { + var mainIndexHelper = _this3.mainIndex.getHelper(); + + var searchFunctionHelper = algoliasearchHelper_1(fakeClient, mainIndexHelper.state.index, mainIndexHelper.state); + searchFunctionHelper.once('search', function (_ref2) { + var state = _ref2.state; + mainIndexHelper.overrideStateWithoutTriggeringChangeEvent(state); + + _this3._mainHelperSearch(); + }); // Forward state changes from `searchFunctionHelper` to `mainIndexHelper` + + searchFunctionHelper.on('change', function (_ref3) { + var state = _ref3.state; + mainIndexHelper.setState(state); + }); + + _this3._searchFunction(searchFunctionHelper); + + return mainHelper; + }; + } // Only the "main" Helper emits the `error` event vs the one for `search` + // and `results` that are also emitted on the derived one. + + + mainHelper.on('error', function (_ref4) { + var error = _ref4.error; + + if (!(error instanceof Error)) { + // typescript lies here, error is in some cases { name: string, message: string } + var err = error; + error = Object.keys(err).reduce(function (acc, key) { + acc[key] = err[key]; + return acc; + }, new Error(err.message)); + } // If an error is emitted, it is re-thrown by events. In previous versions + // we emitted {error}, which is thrown as: + // "Uncaught, unspecified \"error\" event. ([object Object])" + // To avoid breaking changes, we make the error available in both + // `error` and `error.error` + // @MAJOR emit only error + + + error.error = error; + _this3.error = error; + _this3.status = 'error'; + + _this3.scheduleRender(false); // This needs to execute last because it throws the error. + + + _this3.emit('error', error); + }); + this.mainHelper = mainHelper; + this.middleware.forEach(function (_ref5) { + var instance = _ref5.instance; + instance.subscribe(); + }); + this.mainIndex.init({ + instantSearchInstance: this, + parent: null, + uiState: this._initialUiState + }); + + if (this._initialResults) { + var originalScheduleSearch = this.scheduleSearch; // We don't schedule a first search when initial results are provided + // because we already have the results to render. This skips the initial + // network request on the browser on `start`. + + this.scheduleSearch = defer(noop); // We also skip the initial network request when widgets are dynamically + // added in the first tick (that's the case in all the framework-based flavors). + // When we add a widget to `index`, it calls `scheduleSearch`. We can rely + // on our `defer` util to restore the original `scheduleSearch` value once + // widgets are added to hook back to the regular lifecycle. + + defer(function () { + _this3.scheduleSearch = originalScheduleSearch; + })(); + } // We only schedule a search when widgets have been added before `start()` + // because there are listeners that can use these results. + // This is especially useful in framework-based flavors that wait for + // dynamically-added widgets to trigger a network request. It avoids + // having to batch this initial network request with the one coming from + // `addWidgets()`. + // Later, we could also skip `index()` widgets and widgets that don't read + // the results, but this is an optimization that has a very low impact for now. + else if (this.mainIndex.getWidgets().length > 0) { + this.scheduleSearch(); + } // Keep the previous reference for legacy purpose, some pattern use + // the direct Helper access `search.helper` (e.g multi-index). + + + this.helper = this.mainIndex.getHelper(); // track we started the search if we add more widgets, + // to init them directly after add + + this.started = true; + this.middleware.forEach(function (_ref6) { + var instance = _ref6.instance; + instance.started(); + }); + } + /** + * Removes all widgets without triggering a search afterwards. This is an **EXPERIMENTAL** feature, + * if you find an issue with it, please + * [open an issue](https://github.com/algolia/instantsearch.js/issues/new?title=Problem%20with%20dispose). + * @return {undefined} This method does not return anything + */ + + }, { + key: "dispose", + value: function dispose() { + this.scheduleSearch.cancel(); + this.scheduleRender.cancel(); + clearTimeout(this._searchStalledTimer); + this.removeWidgets(this.mainIndex.getWidgets()); + this.mainIndex.dispose(); // You can not start an instance two times, therefore a disposed instance + // needs to set started as false otherwise this can not be restarted at a + // later point. + + this.started = false; // The helper needs to be reset to perform the next search from a fresh state. + // If not reset, it would use the state stored before calling `dispose()`. + + this.removeAllListeners(); + this.mainHelper.removeAllListeners(); + this.mainHelper = null; + this.helper = null; + this.middleware.forEach(function (_ref7) { + var instance = _ref7.instance; + instance.unsubscribe(); + }); + } + }, { + key: "scheduleStalledRender", + value: function scheduleStalledRender() { + var _this4 = this; + + if (!this._searchStalledTimer) { + this._searchStalledTimer = setTimeout(function () { + _this4.status = 'stalled'; + + _this4.scheduleRender(); + }, this._stalledSearchDelay); + } + } + /** + * Set the UI state and trigger a search. + * @param uiState The next UI state or a function computing it from the current state + * @param callOnStateChange private parameter used to know if the method is called from a state change + */ + + }, { + key: "setUiState", + value: function setUiState(uiState) { + var _this5 = this; + + var callOnStateChange = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; + + if (!this.mainHelper) { + throw new Error(withUsage$1('The `start` method needs to be called before `setUiState`.')); + } // We refresh the index UI state to update the local UI state that the + // main index passes to the function form of `setUiState`. + + + this.mainIndex.refreshUiState(); + var nextUiState = typeof uiState === 'function' ? uiState(this.mainIndex.getWidgetUiState({})) : uiState; + + if (this.onStateChange && callOnStateChange) { + this.onStateChange({ + uiState: nextUiState, + setUiState: function setUiState(finalUiState) { + setIndexHelperState(typeof finalUiState === 'function' ? finalUiState(nextUiState) : finalUiState, _this5.mainIndex); + + _this5.scheduleSearch(); + + _this5.onInternalStateChange(); + } + }); + } else { + setIndexHelperState(nextUiState, this.mainIndex); + this.scheduleSearch(); + this.onInternalStateChange(); + } + } + }, { + key: "getUiState", + value: function getUiState() { + if (this.started) { + // We refresh the index UI state to make sure changes from `refine` are taken in account + this.mainIndex.refreshUiState(); + } + + return this.mainIndex.getWidgetUiState({}); + } + }, { + key: "createURL", + value: function createURL() { + var nextState = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; + + if (!this.started) { + throw new Error(withUsage$1('The `start` method needs to be called before `createURL`.')); } - this.mainIndex.addWidgets(widgets); - return this; + return this._createURL(nextState); } - /** - * Removes a widget from the search instance. - * @deprecated This method will still be supported in 4.x releases, but not further. It is replaced by `removeWidgets([widget])` - * @param widget The widget instance to remove from InstantSearch. - * - * The widget must implement a `dispose()` method to clear its state. - */ - }, { - key: "removeWidget", - value: function removeWidget(widget) { - _warning(false, 'removeWidget will still be supported in 4.x releases, but not further. It is replaced by `removeWidgets([widget])`') ; - return this.removeWidgets([widget]); + key: "refresh", + value: function refresh() { + if (!this.mainHelper) { + throw new Error(withUsage$1('The `start` method needs to be called before `refresh`.')); + } + + this.mainHelper.clearCache().search(); } - /** - * Removes multiple widgets from the search instance. - * @param widgets Array of widgets instances to remove from InstantSearch. - * - * The widgets must implement a `dispose()` method to clear their states. - */ + }]); - }, { - key: "removeWidgets", - value: function removeWidgets(widgets) { - if (!Array.isArray(widgets)) { - throw new Error(withUsage$1('The `removeWidgets` method expects an array of widgets. Please use `removeWidget`.')); - } + return InstantSearch; + }(events); - if (widgets.some(function (widget) { - return typeof widget.dispose !== 'function'; - })) { - throw new Error(withUsage$1('The widget definition expects a `dispose` method.')); - } + var withUsage$2 = createDocumentationMessageGenerator({ + name: 'clear-refinements', + connector: true + }); - this.mainIndex.removeWidgets(widgets); - return this; + var connectClearRefinements = function connectClearRefinements(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$2()); + return function (widgetParams) { + var _ref = widgetParams || {}, + _ref$includedAttribut = _ref.includedAttributes, + includedAttributes = _ref$includedAttribut === void 0 ? [] : _ref$includedAttribut, + _ref$excludedAttribut = _ref.excludedAttributes, + excludedAttributes = _ref$excludedAttribut === void 0 ? ['query'] : _ref$excludedAttribut, + _ref$transformItems = _ref.transformItems, + transformItems = _ref$transformItems === void 0 ? function (items) { + return items; + } : _ref$transformItems; + + if (widgetParams && widgetParams.includedAttributes && widgetParams.excludedAttributes) { + throw new Error(withUsage$2('The options `includedAttributes` and `excludedAttributes` cannot be used together.')); } - /** - * Ends the initialization of InstantSearch.js and triggers the - * first search. This method should be called after all widgets have been added - * to the instance of InstantSearch.js. InstantSearch.js also supports adding and removing - * widgets after the start as an **EXPERIMENTAL** feature. - */ - }, { - key: "start", - value: function start() { - var _this3 = this; + var connectorState = { + refine: noop, + createURL: function createURL() { + return ''; + }, + attributesToClear: [] + }; - if (this.started) { - throw new Error(withUsage$1('The `start` method has already been called once.')); - } // This Helper is used for the queries, we don't care about its state. The - // states are managed at the `index` level. We use this Helper to create - // DerivedHelper scoped into the `index` widgets. - // In Vue InstantSearch' hydrate, a main helper gets set before start, so - // we need to respect this helper as a way to keep all listeners correct. + var cachedRefine = function cachedRefine() { + return connectorState.refine(); + }; + var cachedCreateURL = function cachedCreateURL() { + return connectorState.createURL(); + }; - var mainHelper = this.mainHelper || algoliasearchHelper_1(this.client, this.indexName); + return { + $$type: 'ais.clearRefinements', + init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), false); + }, + dispose: function dispose() { + unmountFn(); + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + clearRefinements: this.getWidgetRenderState(renderOptions) + }); + }, + getWidgetRenderState: function getWidgetRenderState(_ref2) { + var createURL = _ref2.createURL, + scopedResults = _ref2.scopedResults, + results = _ref2.results; + connectorState.attributesToClear = scopedResults.reduce(function (attributesToClear, scopedResult) { + return attributesToClear.concat(getAttributesToClear({ + scopedResult: scopedResult, + includedAttributes: includedAttributes, + excludedAttributes: excludedAttributes, + transformItems: transformItems, + results: results + })); + }, []); - mainHelper.search = function () { - // This solution allows us to keep the exact same API for the users but - // under the hood, we have a different implementation. It should be - // completely transparent for the rest of the codebase. Only this module - // is impacted. - return mainHelper.searchOnlyWithDerivedHelpers(); - }; + connectorState.refine = function () { + connectorState.attributesToClear.forEach(function (_ref3) { + var indexHelper = _ref3.helper, + items = _ref3.items; + indexHelper.setState(clearRefinements({ + helper: indexHelper, + attributesToClear: items + })).search(); + }); + }; + + connectorState.createURL = function () { + return createURL(mergeSearchParameters.apply(void 0, _toConsumableArray(connectorState.attributesToClear.map(function (_ref4) { + var indexHelper = _ref4.helper, + items = _ref4.items; + return clearRefinements({ + helper: indexHelper, + attributesToClear: items + }); + })))); + }; + + var canRefine = connectorState.attributesToClear.some(function (attributeToClear) { + return attributeToClear.items.length > 0; + }); + return { + canRefine: canRefine, + hasRefinements: canRefine, + refine: cachedRefine, + createURL: cachedCreateURL, + widgetParams: widgetParams + }; + } + }; + }; + }; + + function getAttributesToClear(_ref5) { + var scopedResult = _ref5.scopedResult, + includedAttributes = _ref5.includedAttributes, + excludedAttributes = _ref5.excludedAttributes, + transformItems = _ref5.transformItems, + results = _ref5.results; + var includesQuery = includedAttributes.indexOf('query') !== -1 || excludedAttributes.indexOf('query') === -1; + return { + helper: scopedResult.helper, + items: transformItems(uniq(getRefinements(scopedResult.results, scopedResult.helper.state, includesQuery).map(function (refinement) { + return refinement.attribute; + }).filter(function (attribute) { + return (// If the array is empty (default case), we keep all the attributes + includedAttributes.length === 0 || // Otherwise, only add the specified attributes + includedAttributes.indexOf(attribute) !== -1 + ); + }).filter(function (attribute) { + return (// If the query is included, we ignore the default `excludedAttributes = ['query']` + attribute === 'query' && includesQuery || // Otherwise, ignore the excluded attributes + excludedAttributes.indexOf(attribute) === -1 + ); + })), { + results: results + }) + }; + } + + var withUsage$3 = createDocumentationMessageGenerator({ + name: 'current-refinements', + connector: true + }); + + var connectCurrentRefinements = function connectCurrentRefinements(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$3()); + return function (widgetParams) { + if ((widgetParams || {}).includedAttributes && (widgetParams || {}).excludedAttributes) { + throw new Error(withUsage$3('The options `includedAttributes` and `excludedAttributes` cannot be used together.')); + } + + var _ref = widgetParams || {}, + includedAttributes = _ref.includedAttributes, + _ref$excludedAttribut = _ref.excludedAttributes, + excludedAttributes = _ref$excludedAttribut === void 0 ? ['query'] : _ref$excludedAttribut, + _ref$transformItems = _ref.transformItems, + transformItems = _ref$transformItems === void 0 ? function (items) { + return items; + } : _ref$transformItems; + + return { + $$type: 'ais.currentRefinements', + init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), false); + }, + dispose: function dispose() { + unmountFn(); + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + currentRefinements: this.getWidgetRenderState(renderOptions) + }); + }, + getWidgetRenderState: function getWidgetRenderState(_ref2) { + var results = _ref2.results, + scopedResults = _ref2.scopedResults, + _createURL = _ref2.createURL, + helper = _ref2.helper; - if (this._searchFunction) { - // this client isn't used to actually search, but required for the helper - // to not throw errors - var fakeClient = { - search: function search() { - return new Promise(noop); + function getItems() { + if (!results) { + return transformItems(getRefinementsItems({ + results: {}, + helper: helper, + includedAttributes: includedAttributes, + excludedAttributes: excludedAttributes + }), { + results: results + }); } - }; - this._mainHelperSearch = mainHelper.search.bind(mainHelper); - - mainHelper.search = function () { - var mainIndexHelper = _this3.mainIndex.getHelper(); - - var searchFunctionHelper = algoliasearchHelper_1(fakeClient, mainIndexHelper.state.index, mainIndexHelper.state); - searchFunctionHelper.once('search', function (_ref2) { - var state = _ref2.state; - mainIndexHelper.overrideStateWithoutTriggeringChangeEvent(state); - _this3._mainHelperSearch(); - }); // Forward state changes from `searchFunctionHelper` to `mainIndexHelper` - - searchFunctionHelper.on('change', function (_ref3) { - var state = _ref3.state; - mainIndexHelper.setState(state); - }); - - _this3._searchFunction(searchFunctionHelper); + return scopedResults.reduce(function (accResults, scopedResult) { + return accResults.concat(transformItems(getRefinementsItems({ + results: scopedResult.results, + helper: scopedResult.helper, + includedAttributes: includedAttributes, + excludedAttributes: excludedAttributes + }), { + results: results + })); + }, []); + } - return mainHelper; + var items = getItems(); + return { + items: items, + canRefine: items.length > 0, + refine: function refine(refinement) { + return clearRefinement(helper, refinement); + }, + createURL: function createURL(refinement) { + return _createURL(clearRefinementFromState(helper.state, refinement)); + }, + widgetParams: widgetParams }; - } // Only the "main" Helper emits the `error` event vs the one for `search` - // and `results` that are also emitted on the derived one. - - - mainHelper.on('error', function (_ref4) { - var error = _ref4.error; - // If an error is emitted, it is re-thrown by events. In previous versions - // we emitted {error}, which is thrown as: - // "Uncaught, unspecified \"error\" event. ([object Object])" - // To avoid breaking changes, we make the error available in both - // `error` and `error.error` - // @MAJOR emit only error - error.error = error; - - _this3.emit('error', error); - }); - this.mainHelper = mainHelper; - this.middleware.forEach(function (_ref5) { - var instance = _ref5.instance; - instance.subscribe(); - }); - this.mainIndex.init({ - instantSearchInstance: this, - parent: null, - uiState: this._initialUiState - }); - - if (this._initialResults) { - var originalScheduleSearch = this.scheduleSearch; // We don't schedule a first search when initial results are provided - // because we already have the results to render. This skips the initial - // network request on the browser on `start`. + } + }; + }; + }; - this.scheduleSearch = defer(noop); // We also skip the initial network request when widgets are dynamically - // added in the first tick (that's the case in all the framework-based flavors). - // When we add a widget to `index`, it calls `scheduleSearch`. We can rely - // on our `defer` util to restore the original `scheduleSearch` value once - // widgets are added to hook back to the regular lifecycle. + function getRefinementsItems(_ref3) { + var results = _ref3.results, + helper = _ref3.helper, + includedAttributes = _ref3.includedAttributes, + excludedAttributes = _ref3.excludedAttributes; + var includesQuery = (includedAttributes || []).indexOf('query') !== -1 || (excludedAttributes || []).indexOf('query') === -1; + var filterFunction = includedAttributes ? function (item) { + return includedAttributes.indexOf(item.attribute) !== -1; + } : function (item) { + return excludedAttributes.indexOf(item.attribute) === -1; + }; + var items = getRefinements(results, helper.state, includesQuery).map(normalizeRefinement).filter(filterFunction); + return items.reduce(function (allItems, currentItem) { + return [].concat(_toConsumableArray(allItems.filter(function (item) { + return item.attribute !== currentItem.attribute; + })), [{ + indexName: helper.state.index, + attribute: currentItem.attribute, + label: currentItem.attribute, + refinements: items.filter(function (result) { + return result.attribute === currentItem.attribute; + }) // We want to keep the order of refinements except the numeric ones. + .sort(function (a, b) { + return a.type === 'numeric' ? a.value - b.value : 0; + }), + refine: function refine(refinement) { + return clearRefinement(helper, refinement); + } + }]); + }, []); + } - defer(function () { - _this3.scheduleSearch = originalScheduleSearch; - })(); - } else { - this.scheduleSearch(); - } // Keep the previous reference for legacy purpose, some pattern use - // the direct Helper access `search.helper` (e.g multi-index). + function clearRefinementFromState(state, refinement) { + state = state.resetPage(); + switch (refinement.type) { + case 'facet': + return state.removeFacetRefinement(refinement.attribute, String(refinement.value)); - this.helper = this.mainIndex.getHelper(); // track we started the search if we add more widgets, - // to init them directly after add + case 'disjunctive': + return state.removeDisjunctiveFacetRefinement(refinement.attribute, String(refinement.value)); - this.started = true; - } - /** - * Removes all widgets without triggering a search afterwards. This is an **EXPERIMENTAL** feature, - * if you find an issue with it, please - * [open an issue](https://github.com/algolia/instantsearch.js/issues/new?title=Problem%20with%20dispose). - * @return {undefined} This method does not return anything - */ + case 'hierarchical': + return state.removeHierarchicalFacetRefinement(refinement.attribute); - }, { - key: "dispose", - value: function dispose() { - this.scheduleSearch.cancel(); - this.scheduleRender.cancel(); - clearTimeout(this._searchStalledTimer); - this.removeWidgets(this.mainIndex.getWidgets()); - this.mainIndex.dispose(); // You can not start an instance two times, therefore a disposed instance - // needs to set started as false otherwise this can not be restarted at a - // later point. + case 'exclude': + return state.removeExcludeRefinement(refinement.attribute, String(refinement.value)); - this.started = false; // The helper needs to be reset to perform the next search from a fresh state. - // If not reset, it would use the state stored before calling `dispose()`. + case 'numeric': + return state.removeNumericRefinement(refinement.attribute, refinement.operator, String(refinement.value)); - this.removeAllListeners(); - this.mainHelper.removeAllListeners(); - this.mainHelper = null; - this.helper = null; - this.middleware.forEach(function (_ref6) { - var instance = _ref6.instance; - instance.unsubscribe(); - }); - } - }, { - key: "scheduleStalledRender", - value: function scheduleStalledRender() { - var _this4 = this; + case 'tag': + return state.removeTagRefinement(String(refinement.value)); - if (!this._searchStalledTimer) { - this._searchStalledTimer = setTimeout(function () { - _this4._isSearchStalled = true; + case 'query': + return state.setQueryParameter('query', ''); - _this4.scheduleRender(); - }, this._stalledSearchDelay); - } - } - }, { - key: "setUiState", - value: function setUiState(uiState) { - if (!this.mainHelper) { - throw new Error(withUsage$1('The `start` method needs to be called before `setUiState`.')); - } // We refresh the index UI state to update the local UI state that the - // main index passes to the function form of `setUiState`. + default: + _warning(false, "The refinement type \"".concat(refinement.type, "\" does not exist and cannot be cleared from the current refinements.")) ; + return state; + } + } + function clearRefinement(helper, refinement) { + helper.setState(clearRefinementFromState(helper.state, refinement)).search(); + } - this.mainIndex.refreshUiState(); - var nextUiState = typeof uiState === 'function' ? uiState(this.mainIndex.getWidgetUiState({})) : uiState; + function getOperatorSymbol(operator) { + switch (operator) { + case '>=': + return '≥'; - var setIndexHelperState = function setIndexHelperState(indexWidget) { - var nextIndexUiState = nextUiState[indexWidget.getIndexId()] || {}; + case '<=': + return '≤'; - { - checkIndexUiState({ - index: indexWidget, - indexUiState: nextIndexUiState - }); - } + default: + return operator; + } + } - indexWidget.getHelper().setState(indexWidget.getWidgetSearchParameters(indexWidget.getHelper().state, { - uiState: nextIndexUiState - })); - indexWidget.getWidgets().filter(isIndexWidget).forEach(setIndexHelperState); - }; + function normalizeRefinement(refinement) { + var value = getValue(refinement); + var label = refinement.operator ? "".concat(getOperatorSymbol(refinement.operator), " ").concat(refinement.name) : refinement.name; + var normalizedRefinement = { + attribute: refinement.attribute, + type: refinement.type, + value: value, + label: label + }; - setIndexHelperState(this.mainIndex); - this.scheduleSearch(); - this.onInternalStateChange(); - } - }, { - key: "getUiState", - value: function getUiState() { - if (this.started) { - // We refresh the index UI state to make sure changes from `refine` are taken in account - this.mainIndex.refreshUiState(); - } + if (refinement.operator !== undefined) { + normalizedRefinement.operator = refinement.operator; + } - return this.mainIndex.getWidgetUiState({}); - } - }, { - key: "createURL", - value: function createURL() { - var nextState = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; + if (refinement.count !== undefined) { + normalizedRefinement.count = refinement.count; + } - if (!this.started) { - throw new Error(withUsage$1('The `start` method needs to be called before `createURL`.')); - } + if (refinement.exhaustive !== undefined) { + normalizedRefinement.exhaustive = refinement.exhaustive; + } - return this._createURL(nextState); - } - }, { - key: "refresh", - value: function refresh() { - if (!this.mainHelper) { - throw new Error(withUsage$1('The `start` method needs to be called before `refresh`.')); - } + return normalizedRefinement; + } - this.mainHelper.clearCache().search(); - } - }]); + function getValue(refinement) { + if (refinement.type === 'numeric') { + return Number(refinement.name); + } - return InstantSearch; - }(events); + if ('escapedValue' in refinement) { + return refinement.escapedValue; + } - var withUsage$2 = createDocumentationMessageGenerator({ - name: 'clear-refinements', + return refinement.name; + } + + var withUsage$4 = createDocumentationMessageGenerator({ + name: 'hierarchical-menu', connector: true }); + var DEFAULT_SORT = ['name:asc']; - var connectClearRefinements = function connectClearRefinements(renderFn) { + /** + * **HierarchicalMenu** connector provides the logic to build a custom widget + * that will give the user the ability to explore facets in a tree-like structure. + * + * This is commonly used for multi-level categorization of products on e-commerce + * websites. From a UX point of view, we suggest not displaying more than two + * levels deep. + * + * @type {Connector} + * @param {function(HierarchicalMenuRenderingOptions, boolean)} renderFn Rendering function for the custom **HierarchicalMenu** widget. + * @param {function} unmountFn Unmount function called when the widget is disposed. + * @return {function(CustomHierarchicalMenuWidgetParams)} Re-usable widget factory for a custom **HierarchicalMenu** widget. + */ + var connectHierarchicalMenu = function connectHierarchicalMenu(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$2()); + checkRendering(renderFn, withUsage$4()); return function (widgetParams) { var _ref = widgetParams || {}, - _ref$includedAttribut = _ref.includedAttributes, - includedAttributes = _ref$includedAttribut === void 0 ? [] : _ref$includedAttribut, - _ref$excludedAttribut = _ref.excludedAttributes, - excludedAttributes = _ref$excludedAttribut === void 0 ? ['query'] : _ref$excludedAttribut, + attributes = _ref.attributes, + _ref$separator = _ref.separator, + separator = _ref$separator === void 0 ? ' > ' : _ref$separator, + _ref$rootPath = _ref.rootPath, + rootPath = _ref$rootPath === void 0 ? null : _ref$rootPath, + _ref$showParentLevel = _ref.showParentLevel, + showParentLevel = _ref$showParentLevel === void 0 ? true : _ref$showParentLevel, + _ref$limit = _ref.limit, + limit = _ref$limit === void 0 ? 10 : _ref$limit, + _ref$showMore = _ref.showMore, + showMore = _ref$showMore === void 0 ? false : _ref$showMore, + _ref$showMoreLimit = _ref.showMoreLimit, + showMoreLimit = _ref$showMoreLimit === void 0 ? 20 : _ref$showMoreLimit, + _ref$sortBy = _ref.sortBy, + sortBy = _ref$sortBy === void 0 ? DEFAULT_SORT : _ref$sortBy, _ref$transformItems = _ref.transformItems, transformItems = _ref$transformItems === void 0 ? function (items) { return items; } : _ref$transformItems; - if (widgetParams && widgetParams.includedAttributes && widgetParams.excludedAttributes) { - throw new Error(withUsage$2('The options `includedAttributes` and `excludedAttributes` cannot be used together.')); + if (!attributes || !Array.isArray(attributes) || attributes.length === 0) { + throw new Error(withUsage$4('The `attributes` option expects an array of strings.')); } - var connectorState = { - refine: noop, - createURL: function createURL() { - return ''; - }, - attributesToClear: [] - }; + if (showMore === true && showMoreLimit <= limit) { + throw new Error(withUsage$4('The `showMoreLimit` option must be greater than `limit`.')); + } - var cachedRefine = function cachedRefine() { - return connectorState.refine(); - }; + // we need to provide a hierarchicalFacet name for the search state + // so that we can always map $hierarchicalFacetName => real attributes + // we use the first attribute name + var _attributes = _slicedToArray(attributes, 1), + hierarchicalFacetName = _attributes[0]; - var cachedCreateURL = function cachedCreateURL() { - return connectorState.createURL(); - }; + var sendEvent; // Provide the same function to the `renderFn` so that way the user + // has to only bind it once when `isFirstRendering` for instance + + var toggleShowMore = function toggleShowMore() {}; + + function cachedToggleShowMore() { + toggleShowMore(); + } + + var _refine; + + var isShowingMore = false; + + function createToggleShowMore(renderOptions, widget) { + return function () { + isShowingMore = !isShowingMore; + widget.render(renderOptions); + }; + } + + function getLimit() { + return isShowingMore ? showMoreLimit : limit; + } + + function _prepareFacetValues(facetValues) { + return facetValues.slice(0, getLimit()).map(function (_ref2) { + var label = _ref2.name, + value = _ref2.escapedValue, + data = _ref2.data, + path = _ref2.path, + subValue = _objectWithoutProperties(_ref2, ["name", "escapedValue", "data", "path"]); + + var item = _objectSpread2(_objectSpread2({}, subValue), {}, { + value: value, + label: label, + data: null + }); + + if (Array.isArray(data)) { + item.data = _prepareFacetValues(data); + } + + return item; + }); + } return { - $$type: 'ais.clearRefinements', + $$type: 'ais.hierarchicalMenu', init: function init(initOptions) { var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { @@ -10616,1284 +10339,1400 @@ }, render: function render(renderOptions) { var instantSearchInstance = renderOptions.instantSearchInstance; + toggleShowMore = createToggleShowMore(renderOptions, this); renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { instantSearchInstance: instantSearchInstance }), false); }, - dispose: function dispose() { + dispose: function dispose(_ref3) { + var state = _ref3.state; unmountFn(); + return state.removeHierarchicalFacet(hierarchicalFacetName).setQueryParameter('maxValuesPerFacet', undefined); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - clearRefinements: this.getWidgetRenderState(renderOptions) + hierarchicalMenu: _objectSpread2(_objectSpread2({}, renderState.hierarchicalMenu), {}, _defineProperty({}, hierarchicalFacetName, this.getWidgetRenderState(renderOptions))) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref2) { - var createURL = _ref2.createURL, - scopedResults = _ref2.scopedResults, - results = _ref2.results; - connectorState.attributesToClear = scopedResults.reduce(function (attributesToClear, scopedResult) { - return attributesToClear.concat(getAttributesToClear({ - scopedResult: scopedResult, - includedAttributes: includedAttributes, - excludedAttributes: excludedAttributes, - transformItems: transformItems, + getWidgetRenderState: function getWidgetRenderState(_ref4) { + var results = _ref4.results, + state = _ref4.state, + createURL = _ref4.createURL, + instantSearchInstance = _ref4.instantSearchInstance, + helper = _ref4.helper; + var items = []; + var canToggleShowMore = false; // Bind createURL to this specific attribute + + function _createURL(facetValue) { + return createURL(state.resetPage().toggleFacetRefinement(hierarchicalFacetName, facetValue)); + } + + if (!sendEvent) { + sendEvent = createSendEventForFacet({ + instantSearchInstance: instantSearchInstance, + helper: helper, + attribute: function attribute(facetValue) { + var index = facetValue.split(separator).length - 1; + return attributes[index]; + }, + widgetType: this.$$type + }); + } + + if (!_refine) { + _refine = function _refine(facetValue) { + sendEvent('click', facetValue); + helper.toggleFacetRefinement(hierarchicalFacetName, facetValue).search(); + }; + } + + if (results) { + var facetValues = results.getFacetValues(hierarchicalFacetName, { + sortBy: sortBy, + facetOrdering: sortBy === DEFAULT_SORT + }); + var facetItems = facetValues && !Array.isArray(facetValues) && facetValues.data ? facetValues.data : []; // If the limit is the max number of facet retrieved it is impossible to know + // if the facets are exhaustive. The only moment we are sure it is exhaustive + // is when it is strictly under the number requested unless we know that another + // widget has requested more values (maxValuesPerFacet > getLimit()). + // Because this is used for making the search of facets unable or not, it is important + // to be conservative here. + + var hasExhaustiveItems = (state.maxValuesPerFacet || 0) > getLimit() ? facetItems.length <= getLimit() : facetItems.length < getLimit(); + canToggleShowMore = showMore && (isShowingMore || !hasExhaustiveItems); + items = transformItems(_prepareFacetValues(facetItems), { results: results - })); - }, []); + }); + } + + return { + items: items, + refine: _refine, + canRefine: items.length > 0, + createURL: _createURL, + sendEvent: sendEvent, + widgetParams: widgetParams, + isShowingMore: isShowingMore, + toggleShowMore: cachedToggleShowMore, + canToggleShowMore: canToggleShowMore + }; + }, + getWidgetUiState: function getWidgetUiState(uiState, _ref5) { + var searchParameters = _ref5.searchParameters; + var path = searchParameters.getHierarchicalFacetBreadcrumb(hierarchicalFacetName); + + if (!path.length) { + return uiState; + } + + return _objectSpread2(_objectSpread2({}, uiState), {}, { + hierarchicalMenu: _objectSpread2(_objectSpread2({}, uiState.hierarchicalMenu), {}, _defineProperty({}, hierarchicalFacetName, path)) + }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { + var uiState = _ref6.uiState; + var values = uiState.hierarchicalMenu && uiState.hierarchicalMenu[hierarchicalFacetName]; + + if (searchParameters.isHierarchicalFacet(hierarchicalFacetName)) { + var facet = searchParameters.getHierarchicalFacetByName(hierarchicalFacetName); + _warning(isEqual(facet.attributes, attributes) && facet.separator === separator && facet.rootPath === rootPath, 'Using Breadcrumb and HierarchicalMenu on the same facet with different options overrides the configuration of the HierarchicalMenu.') ; + } + + var withFacetConfiguration = searchParameters.removeHierarchicalFacet(hierarchicalFacetName).addHierarchicalFacet({ + name: hierarchicalFacetName, + attributes: attributes, + separator: separator, + rootPath: rootPath, + showParentLevel: showParentLevel + }); + var currentMaxValuesPerFacet = withFacetConfiguration.maxValuesPerFacet || 0; + var nextMaxValuesPerFacet = Math.max(currentMaxValuesPerFacet, showMore ? showMoreLimit : limit); + var withMaxValuesPerFacet = withFacetConfiguration.setQueryParameter('maxValuesPerFacet', nextMaxValuesPerFacet); - connectorState.refine = function () { - connectorState.attributesToClear.forEach(function (_ref3) { - var indexHelper = _ref3.helper, - items = _ref3.items; - indexHelper.setState(clearRefinements({ - helper: indexHelper, - attributesToClear: items - })).search(); + if (!values) { + return withMaxValuesPerFacet.setQueryParameters({ + hierarchicalFacetsRefinements: _objectSpread2(_objectSpread2({}, withMaxValuesPerFacet.hierarchicalFacetsRefinements), {}, _defineProperty({}, hierarchicalFacetName, [])) }); - }; - - connectorState.createURL = function () { - return createURL(merge$1.apply(void 0, _toConsumableArray(connectorState.attributesToClear.map(function (_ref4) { - var indexHelper = _ref4.helper, - items = _ref4.items; - return clearRefinements({ - helper: indexHelper, - attributesToClear: items - }); - })))); - }; + } - var canRefine = connectorState.attributesToClear.some(function (attributeToClear) { - return attributeToClear.items.length > 0; - }); - return { - canRefine: canRefine, - hasRefinements: canRefine, - refine: cachedRefine, - createURL: cachedCreateURL, - widgetParams: widgetParams - }; + return withMaxValuesPerFacet.addHierarchicalFacetRefinement(hierarchicalFacetName, values.join(separator)); } }; }; }; - function getAttributesToClear(_ref5) { - var scopedResult = _ref5.scopedResult, - includedAttributes = _ref5.includedAttributes, - excludedAttributes = _ref5.excludedAttributes, - transformItems = _ref5.transformItems, - results = _ref5.results; - var includesQuery = includedAttributes.indexOf('query') !== -1 || excludedAttributes.indexOf('query') === -1; - return { - helper: scopedResult.helper, - items: transformItems(uniq(getRefinements(scopedResult.results, scopedResult.helper.state, includesQuery).map(function (refinement) { - return refinement.attribute; - }).filter(function (attribute) { - return (// If the array is empty (default case), we keep all the attributes - includedAttributes.length === 0 || // Otherwise, only add the specified attributes - includedAttributes.indexOf(attribute) !== -1 - ); - }).filter(function (attribute) { - return (// If the query is included, we ignore the default `excludedAttributes = ['query']` - attribute === 'query' && includesQuery || // Otherwise, ignore the excluded attributes - excludedAttributes.indexOf(attribute) === -1 - ); - })), { - results: results - }) - }; - } - - var withUsage$3 = createDocumentationMessageGenerator({ - name: 'current-refinements', + var withUsage$5 = createDocumentationMessageGenerator({ + name: 'hits', connector: true }); - var connectCurrentRefinements = function connectCurrentRefinements(renderFn) { + var connectHits = function connectHits(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$3()); + checkRendering(renderFn, withUsage$5()); return function (widgetParams) { - if ((widgetParams || {}).includedAttributes && (widgetParams || {}).excludedAttributes) { - throw new Error(withUsage$3('The options `includedAttributes` and `excludedAttributes` cannot be used together.')); - } - var _ref = widgetParams || {}, - includedAttributes = _ref.includedAttributes, - _ref$excludedAttribut = _ref.excludedAttributes, - excludedAttributes = _ref$excludedAttribut === void 0 ? ['query'] : _ref$excludedAttribut, + _ref$escapeHTML = _ref.escapeHTML, + escapeHTML = _ref$escapeHTML === void 0 ? true : _ref$escapeHTML, _ref$transformItems = _ref.transformItems, transformItems = _ref$transformItems === void 0 ? function (items) { return items; } : _ref$transformItems; + var sendEvent; + var bindEvent; return { - $$type: 'ais.currentRefinements', + $$type: 'ais.hits', init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance + instantSearchInstance: initOptions.instantSearchInstance }), true); }, render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance + var renderState = this.getWidgetRenderState(renderOptions); + renderFn(_objectSpread2(_objectSpread2({}, renderState), {}, { + instantSearchInstance: renderOptions.instantSearchInstance }), false); - }, - dispose: function dispose() { - unmountFn(); + renderState.sendEvent('view', renderState.hits); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - currentRefinements: this.getWidgetRenderState(renderOptions) + hits: this.getWidgetRenderState(renderOptions) }); }, getWidgetRenderState: function getWidgetRenderState(_ref2) { var results = _ref2.results, - scopedResults = _ref2.scopedResults, - _createURL = _ref2.createURL, - helper = _ref2.helper; + helper = _ref2.helper, + instantSearchInstance = _ref2.instantSearchInstance; - function getItems() { - if (!results) { - return transformItems(getRefinementsItems({ - results: {}, - helper: helper, - includedAttributes: includedAttributes, - excludedAttributes: excludedAttributes - }), { - results: results - }); - } + if (!sendEvent) { + sendEvent = createSendEventForHits({ + instantSearchInstance: instantSearchInstance, + index: helper.getIndex(), + widgetType: this.$$type + }); + } - return scopedResults.reduce(function (accResults, scopedResult) { - return accResults.concat(transformItems(getRefinementsItems({ - results: scopedResult.results, - helper: scopedResult.helper, - includedAttributes: includedAttributes, - excludedAttributes: excludedAttributes - }), { - results: results - })); - }, []); + if (!bindEvent) { + bindEvent = createBindEventForHits({ + index: helper.getIndex(), + widgetType: this.$$type + }); } - var items = getItems(); + if (!results) { + return { + hits: [], + results: undefined, + sendEvent: sendEvent, + bindEvent: bindEvent, + widgetParams: widgetParams + }; + } + + if (escapeHTML && results.hits.length > 0) { + results.hits = escapeHits(results.hits); + } + + var hitsWithAbsolutePosition = addAbsolutePosition(results.hits, results.page, results.hitsPerPage); + var hitsWithAbsolutePositionAndQueryID = addQueryID(hitsWithAbsolutePosition, results.queryID); + var transformedHits = transformItems(hitsWithAbsolutePositionAndQueryID, { + results: results + }); return { - items: items, - canRefine: items.length > 0, - refine: function refine(refinement) { - return clearRefinement(helper, refinement); - }, - createURL: function createURL(refinement) { - return _createURL(clearRefinementFromState(helper.state, refinement)); - }, + hits: transformedHits, + results: results, + sendEvent: sendEvent, + bindEvent: bindEvent, widgetParams: widgetParams }; + }, + dispose: function dispose(_ref3) { + var state = _ref3.state; + unmountFn(); + + if (!escapeHTML) { + return state; + } + + return state.setQueryParameters(Object.keys(TAG_PLACEHOLDER).reduce(function (acc, key) { + return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, key, undefined)); + }, {})); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(state) { + if (!escapeHTML) { + return state; + } + + return state.setQueryParameters(TAG_PLACEHOLDER); } }; }; }; - function getRefinementsItems(_ref3) { - var results = _ref3.results, - helper = _ref3.helper, - includedAttributes = _ref3.includedAttributes, - excludedAttributes = _ref3.excludedAttributes; - var includesQuery = (includedAttributes || []).indexOf('query') !== -1 || (excludedAttributes || []).indexOf('query') === -1; - var filterFunction = includedAttributes ? function (item) { - return includedAttributes.indexOf(item.attribute) !== -1; - } : function (item) { - return excludedAttributes.indexOf(item.attribute) === -1; - }; - var items = getRefinements(results, helper.state, includesQuery).map(normalizeRefinement).filter(filterFunction); - return items.reduce(function (allItems, currentItem) { - return [].concat(_toConsumableArray(allItems.filter(function (item) { - return item.attribute !== currentItem.attribute; - })), [{ - indexName: helper.state.index, - attribute: currentItem.attribute, - label: currentItem.attribute, - refinements: items.filter(function (result) { - return result.attribute === currentItem.attribute; - }) // We want to keep the order of refinements except the numeric ones. - .sort(function (a, b) { - return a.type === 'numeric' ? a.value - b.value : 0; - }), - refine: function refine(refinement) { - return clearRefinement(helper, refinement); + var getSelectedHits = function getSelectedHits(hits, selectedObjectIDs) { + return selectedObjectIDs.map(function (objectID) { + var hit = find$1(hits, function (h) { + return h.objectID === objectID; + }); + + if (typeof hit === 'undefined') { + throw new Error("Could not find objectID \"".concat(objectID, "\" passed to `clickedObjectIDsAfterSearch` in the returned hits. This is necessary to infer the absolute position and the query ID.")); + } + + return hit; + }); + }; + + var getQueryID = function getQueryID(selectedHits) { + var queryIDs = uniq(selectedHits.map(function (hit) { + return hit.__queryID; + })); + + if (queryIDs.length > 1) { + throw new Error('Insights currently allows a single `queryID`. The `objectIDs` provided map to multiple `queryID`s.'); + } + + var queryID = queryIDs[0]; + + if (typeof queryID !== 'string') { + throw new Error("Could not infer `queryID`. Ensure InstantSearch `clickAnalytics: true` was added with the Configure widget.\n\nSee: https://alg.li/lNiZZ7"); + } + + return queryID; + }; + + var getPositions = function getPositions(selectedHits) { + return selectedHits.map(function (hit) { + return hit.__position; + }); + }; + + var inferPayload = function inferPayload(_ref) { + var method = _ref.method, + results = _ref.results, + hits = _ref.hits, + objectIDs = _ref.objectIDs; + var index = results.index; + var selectedHits = getSelectedHits(hits, objectIDs); + var queryID = getQueryID(selectedHits); + + switch (method) { + case 'clickedObjectIDsAfterSearch': + { + var positions = getPositions(selectedHits); + return { + index: index, + queryID: queryID, + objectIDs: objectIDs, + positions: positions + }; } - }]); - }, []); - } - function clearRefinementFromState(state, refinement) { - switch (refinement.type) { - case 'facet': - return state.removeFacetRefinement(refinement.attribute, String(refinement.value)); + case 'convertedObjectIDsAfterSearch': + return { + index: index, + queryID: queryID, + objectIDs: objectIDs + }; - case 'disjunctive': - return state.removeDisjunctiveFacetRefinement(refinement.attribute, String(refinement.value)); + default: + throw new Error("Unsupported method passed to insights: \"".concat(method, "\".")); + } + }; - case 'hierarchical': - return state.removeHierarchicalFacetRefinement(refinement.attribute); + var wrapInsightsClient = function wrapInsightsClient(aa, results, hits) { + return function (method) { + for (var _len = arguments.length, payloads = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { + payloads[_key - 1] = arguments[_key]; + } - case 'exclude': - return state.removeExcludeRefinement(refinement.attribute, String(refinement.value)); + var payload = payloads[0]; + _warning(false, "`insights` function has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the `insights` middleware.\n\nFor more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/") ; - case 'numeric': - return state.removeNumericRefinement(refinement.attribute, refinement.operator, String(refinement.value)); + if (!aa) { + var withInstantSearchUsage = createDocumentationMessageGenerator({ + name: 'instantsearch' + }); + throw new Error(withInstantSearchUsage('The `insightsClient` option has not been provided to `instantsearch`.')); + } - case 'tag': - return state.removeTagRefinement(String(refinement.value)); + if (!Array.isArray(payload.objectIDs)) { + throw new TypeError('Expected `objectIDs` to be an array.'); + } - case 'query': - return state.setQueryParameter('query', ''); + var inferredPayload = inferPayload({ + method: method, + results: results, + hits: hits, + objectIDs: payload.objectIDs + }); + aa(method, _objectSpread2(_objectSpread2({}, inferredPayload), payload)); + }; + }; + /** + * @deprecated This function will be still supported in 4.x releases, but not further. It is replaced by the `insights` middleware. For more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/ + * It passes `insights` to `HitsWithInsightsListener` and `InfiniteHitsWithInsightsListener`. + */ - default: - _warning(false, "The refinement type \"".concat(refinement.type, "\" does not exist and cannot be cleared from the current refinements.")) ; - return state; - } - } - function clearRefinement(helper, refinement) { - helper.setState(clearRefinementFromState(helper.state, refinement)).search(); + function withInsights(connector) { + return function (renderFn, unmountFn) { + return connector(function (renderOptions, isFirstRender) { + var results = renderOptions.results, + hits = renderOptions.hits, + instantSearchInstance = renderOptions.instantSearchInstance; + + if (results && hits && instantSearchInstance) { + var insights = wrapInsightsClient(instantSearchInstance.insightsClient, results, hits); + return renderFn(_objectSpread2(_objectSpread2({}, renderOptions), {}, { + insights: insights + }), isFirstRender); + } + + return renderFn(renderOptions, isFirstRender); + }, unmountFn); + }; } - function getOperatorSymbol(operator) { - switch (operator) { - case '>=': - return '≥'; + var n,l,u,t,o,r,f={},e=[],c=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i;function s(n,l){for(var u in l)n[u]=l[u];return n}function a(n){var l=n.parentNode;l&&l.removeChild(n);}function h(l,u,i){var t,o,r,f={};for(r in u)"key"==r?t=u[r]:"ref"==r?o=u[r]:f[r]=u[r];if(arguments.length>2&&(f.children=arguments.length>3?n.call(arguments,2):i),"function"==typeof l&&null!=l.defaultProps)for(r in l.defaultProps)void 0===f[r]&&(f[r]=l.defaultProps[r]);return v(l,f,t,o,null)}function v(n,i,t,o,r){var f={type:n,props:i,key:t,ref:o,__k:null,__:null,__b:0,__e:null,__d:void 0,__c:null,__h:null,constructor:void 0,__v:null==r?++u:r};return null==r&&null!=l.vnode&&l.vnode(f),f}function y(){return {current:null}}function p(n){return n.children}function d(n,l){this.props=n,this.context=l;}function _(n,l){if(null==l)return n.__?_(n.__,n.__.__k.indexOf(n)+1):null;for(var u;l0?v(k.type,k.props,k.key,null,k.__v):k)){if(k.__=u,k.__b=u.__b+1,null===(d=x[h])||d&&k.key==d.key&&k.type===d.type)x[h]=void 0;else for(y=0;y 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$4()); + checkRendering(renderFn, withUsage$6()); return function (widgetParams) { var _ref = widgetParams || {}, - attributes = _ref.attributes, - _ref$separator = _ref.separator, - separator = _ref$separator === void 0 ? ' > ' : _ref$separator, - _ref$rootPath = _ref.rootPath, - rootPath = _ref$rootPath === void 0 ? null : _ref$rootPath, - _ref$showParentLevel = _ref.showParentLevel, - showParentLevel = _ref$showParentLevel === void 0 ? true : _ref$showParentLevel, - _ref$limit = _ref.limit, - limit = _ref$limit === void 0 ? 10 : _ref$limit, - _ref$showMore = _ref.showMore, - showMore = _ref$showMore === void 0 ? false : _ref$showMore, - _ref$showMoreLimit = _ref.showMoreLimit, - showMoreLimit = _ref$showMoreLimit === void 0 ? 20 : _ref$showMoreLimit, - _ref$sortBy = _ref.sortBy, - sortBy = _ref$sortBy === void 0 ? DEFAULT_SORT : _ref$sortBy, + userItems = _ref.items, _ref$transformItems = _ref.transformItems, transformItems = _ref$transformItems === void 0 ? function (items) { return items; } : _ref$transformItems; - if (!attributes || !Array.isArray(attributes) || attributes.length === 0) { - throw new Error(withUsage$4('The `attributes` option expects an array of strings.')); - } - - if (showMore === true && showMoreLimit <= limit) { - throw new Error(withUsage$4('The `showMoreLimit` option must be greater than `limit`.')); - } - - // we need to provide a hierarchicalFacet name for the search state - // so that we can always map $hierarchicalFacetName => real attributes - // we use the first attribute name - var _attributes = _slicedToArray(attributes, 1), - hierarchicalFacetName = _attributes[0]; - - var sendEvent; // Provide the same function to the `renderFn` so that way the user - // has to only bind it once when `isFirstRendering` for instance - - var toggleShowMore = function toggleShowMore() {}; - - function cachedToggleShowMore() { - toggleShowMore(); + if (!Array.isArray(userItems)) { + throw new Error(withUsage$6('The `items` option expects an array of objects.')); } - var _refine; - - var isShowingMore = false; - - function createToggleShowMore(renderOptions, widget) { - return function () { - isShowingMore = !isShowingMore; - widget.render(renderOptions); - }; + var items = userItems; + var defaultItems = items.filter(function (item) { + return item.default === true; + }); + + if (defaultItems.length === 0) { + throw new Error(withUsage$6("A default value must be specified in `items`.")); } - function getLimit() { - return isShowingMore ? showMoreLimit : limit; + if (defaultItems.length > 1) { + throw new Error(withUsage$6('More than one default value is specified in `items`.')); } - function _prepareFacetValues(facetValues) { - return facetValues.slice(0, getLimit()).map(function (_ref2) { - var label = _ref2.name, - value = _ref2.escapedValue, - data = _ref2.data, - path = _ref2.path, - subValue = _objectWithoutProperties(_ref2, ["name", "escapedValue", "data", "path"]); + var defaultItem = defaultItems[0]; - var item = _objectSpread2(_objectSpread2({}, subValue), {}, { - value: value, - label: label, - data: null + var normalizeItems = function normalizeItems(_ref2) { + var hitsPerPage = _ref2.hitsPerPage; + return items.map(function (item) { + return _objectSpread2(_objectSpread2({}, item), {}, { + isRefined: Number(item.value) === Number(hitsPerPage) }); - - if (Array.isArray(data)) { - item.data = _prepareFacetValues(data); - } - - return item; }); - } + }; + var connectorState = { + getRefine: function getRefine(helper) { + return function (value) { + return !value && value !== 0 ? helper.setQueryParameter('hitsPerPage', undefined).search() : helper.setQueryParameter('hitsPerPage', value).search(); + }; + }, + createURLFactory: function createURLFactory(_ref3) { + var state = _ref3.state, + createURL = _ref3.createURL; + return function (value) { + return createURL(state.resetPage().setQueryParameter('hitsPerPage', !value && value !== 0 ? undefined : value)); + }; + } + }; return { - $$type: 'ais.hierarchicalMenu', + $$type: 'ais.hitsPerPage', init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; + var state = initOptions.state, + instantSearchInstance = initOptions.instantSearchInstance; + var isCurrentInOptions = items.some(function (item) { + return Number(state.hitsPerPage) === Number(item.value); + }); + + if (!isCurrentInOptions) { + _warning(state.hitsPerPage !== undefined, "\n`hitsPerPage` is not defined.\nThe option `hitsPerPage` needs to be set using the `configure` widget.\n\nLearn more: https://www.algolia.com/doc/api-reference/widgets/hits-per-page/js/\n ") ; + _warning(false, "\nThe `items` option of `hitsPerPage` does not contain the \"hits per page\" value coming from the state: ".concat(state.hitsPerPage, ".\n\nYou may want to add another entry to the `items` option with this value.")) ; + items = [// The helper will convert the empty string to `undefined`. + { + value: '', + label: '' + }].concat(_toConsumableArray(items)); + } + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { instantSearchInstance: instantSearchInstance }), true); }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - toggleShowMore = createToggleShowMore(renderOptions, this); - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + render: function render(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { instantSearchInstance: instantSearchInstance }), false); }, - dispose: function dispose(_ref3) { - var state = _ref3.state; + dispose: function dispose(_ref4) { + var state = _ref4.state; unmountFn(); - return state.removeHierarchicalFacet(hierarchicalFacetName).setQueryParameter('maxValuesPerFacet', undefined); + return state.setQueryParameter('hitsPerPage', undefined); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - hierarchicalMenu: _objectSpread2(_objectSpread2({}, renderState.hierarchicalMenu), {}, _defineProperty({}, hierarchicalFacetName, this.getWidgetRenderState(renderOptions))) + hitsPerPage: this.getWidgetRenderState(renderOptions) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref4) { - var results = _ref4.results, - state = _ref4.state, - createURL = _ref4.createURL, - instantSearchInstance = _ref4.instantSearchInstance, - helper = _ref4.helper; - var items = []; - var canToggleShowMore = false; // Bind createURL to this specific attribute - - function _createURL(facetValue) { - return createURL(state.resetPage().toggleFacetRefinement(hierarchicalFacetName, facetValue)); - } - - if (!sendEvent) { - sendEvent = createSendEventForFacet({ - instantSearchInstance: instantSearchInstance, - helper: helper, - attribute: hierarchicalFacetName, - widgetType: this.$$type - }); - } - - if (!_refine) { - _refine = function _refine(facetValue) { - sendEvent('click', facetValue); - helper.toggleFacetRefinement(hierarchicalFacetName, facetValue).search(); - }; - } - - if (results) { - var facetValues = results.getFacetValues(hierarchicalFacetName, { - sortBy: sortBy, - facetOrdering: sortBy === DEFAULT_SORT - }); - var facetItems = facetValues && !Array.isArray(facetValues) && facetValues.data ? facetValues.data : []; // If the limit is the max number of facet retrieved it is impossible to know - // if the facets are exhaustive. The only moment we are sure it is exhaustive - // is when it is strictly under the number requested unless we know that another - // widget has requested more values (maxValuesPerFacet > getLimit()). - // Because this is used for making the search of facets unable or not, it is important - // to be conservative here. - - var hasExhaustiveItems = (state.maxValuesPerFacet || 0) > getLimit() ? facetItems.length <= getLimit() : facetItems.length < getLimit(); - canToggleShowMore = showMore && (isShowingMore || !hasExhaustiveItems); - items = transformItems(_prepareFacetValues(facetItems), { - results: results - }); - } - + getWidgetRenderState: function getWidgetRenderState(_ref5) { + var state = _ref5.state, + results = _ref5.results, + createURL = _ref5.createURL, + helper = _ref5.helper; + var canRefine = results ? results.nbHits > 0 : false; return { - items: items, - refine: _refine, - canRefine: items.length > 0, - createURL: _createURL, - sendEvent: sendEvent, - widgetParams: widgetParams, - isShowingMore: isShowingMore, - toggleShowMore: cachedToggleShowMore, - canToggleShowMore: canToggleShowMore + items: transformItems(normalizeItems(state), { + results: results + }), + refine: connectorState.getRefine(helper), + createURL: connectorState.createURLFactory({ + state: state, + createURL: createURL + }), + hasNoResults: !canRefine, + canRefine: canRefine, + widgetParams: widgetParams }; }, - getWidgetUiState: function getWidgetUiState(uiState, _ref5) { - var searchParameters = _ref5.searchParameters; - var path = searchParameters.getHierarchicalFacetBreadcrumb(hierarchicalFacetName); + getWidgetUiState: function getWidgetUiState(uiState, _ref6) { + var searchParameters = _ref6.searchParameters; + var hitsPerPage = searchParameters.hitsPerPage; - if (!path.length) { + if (hitsPerPage === undefined || hitsPerPage === defaultItem.value) { return uiState; } return _objectSpread2(_objectSpread2({}, uiState), {}, { - hierarchicalMenu: _objectSpread2(_objectSpread2({}, uiState.hierarchicalMenu), {}, _defineProperty({}, hierarchicalFacetName, path)) + hitsPerPage: hitsPerPage }); }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { - var uiState = _ref6.uiState; - var values = uiState.hierarchicalMenu && uiState.hierarchicalMenu[hierarchicalFacetName]; + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref7) { + var uiState = _ref7.uiState; + return searchParameters.setQueryParameters({ + hitsPerPage: uiState.hitsPerPage || defaultItem.value + }); + } + }; + }; + }; - if (searchParameters.isHierarchicalFacet(hierarchicalFacetName)) { - var facet = searchParameters.getHierarchicalFacetByName(hierarchicalFacetName); - _warning(isEqual(facet.attributes, attributes) && facet.separator === separator && facet.rootPath === rootPath, 'Using Breadcrumb and HierarchicalMenu on the same facet with different options overrides the configuration of the HierarchicalMenu.') ; - } + var withUsage$7 = createDocumentationMessageGenerator({ + name: 'infinite-hits', + connector: true + }); - var withFacetConfiguration = searchParameters.removeHierarchicalFacet(hierarchicalFacetName).addHierarchicalFacet({ - name: hierarchicalFacetName, - attributes: attributes, - separator: separator, - rootPath: rootPath, - showParentLevel: showParentLevel - }); - var currentMaxValuesPerFacet = withFacetConfiguration.maxValuesPerFacet || 0; - var nextMaxValuesPerFacet = Math.max(currentMaxValuesPerFacet, showMore ? showMoreLimit : limit); - var withMaxValuesPerFacet = withFacetConfiguration.setQueryParameter('maxValuesPerFacet', nextMaxValuesPerFacet); + function getStateWithoutPage(state) { + var _ref = state || {}, + page = _ref.page, + rest = _objectWithoutProperties(_ref, ["page"]); + + return rest; + } + + function getInMemoryCache() { + var cachedHits = null; + var cachedState = null; + return { + read: function read(_ref2) { + var state = _ref2.state; + return isEqual(cachedState, getStateWithoutPage(state)) ? cachedHits : null; + }, + write: function write(_ref3) { + var state = _ref3.state, + hits = _ref3.hits; + cachedState = getStateWithoutPage(state); + cachedHits = hits; + } + }; + } + + function extractHitsFromCachedHits(cachedHits) { + return Object.keys(cachedHits).map(Number).sort(function (a, b) { + return a - b; + }).reduce(function (acc, page) { + return acc.concat(cachedHits[page]); + }, []); + } + + var connectInfiniteHits = function connectInfiniteHits(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$7()); // @TODO: this should be a generic, but a Connector can not yet be generic itself + + return function (widgetParams) { + var _ref4 = widgetParams || {}, + _ref4$escapeHTML = _ref4.escapeHTML, + escapeHTML = _ref4$escapeHTML === void 0 ? true : _ref4$escapeHTML, + _ref4$transformItems = _ref4.transformItems, + transformItems = _ref4$transformItems === void 0 ? function (items) { + return items; + } : _ref4$transformItems, + _ref4$cache = _ref4.cache, + cache = _ref4$cache === void 0 ? getInMemoryCache() : _ref4$cache; + + var showPrevious; + var showMore; + var sendEvent; + var bindEvent; + + var getFirstReceivedPage = function getFirstReceivedPage(state, cachedHits) { + var _state$page = state.page, + page = _state$page === void 0 ? 0 : _state$page; + var pages = Object.keys(cachedHits).map(Number); + + if (pages.length === 0) { + return page; + } else { + return Math.min.apply(Math, [page].concat(_toConsumableArray(pages))); + } + }; - if (!values) { - return withMaxValuesPerFacet.setQueryParameters({ - hierarchicalFacetsRefinements: _objectSpread2(_objectSpread2({}, withMaxValuesPerFacet.hierarchicalFacetsRefinements), {}, _defineProperty({}, hierarchicalFacetName, [])) - }); - } + var getLastReceivedPage = function getLastReceivedPage(state, cachedHits) { + var _state$page2 = state.page, + page = _state$page2 === void 0 ? 0 : _state$page2; + var pages = Object.keys(cachedHits).map(Number); - return withMaxValuesPerFacet.addHierarchicalFacetRefinement(hierarchicalFacetName, values.join(separator)); + if (pages.length === 0) { + return page; + } else { + return Math.max.apply(Math, [page].concat(_toConsumableArray(pages))); } }; - }; - }; - var withUsage$5 = createDocumentationMessageGenerator({ - name: 'hits', - connector: true - }); + var getShowPrevious = function getShowPrevious(helper) { + return function () { + // Using the helper's `overrideStateWithoutTriggeringChangeEvent` method + // avoid updating the browser URL when the user displays the previous page. + helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread2(_objectSpread2({}, helper.state), {}, { + page: getFirstReceivedPage(helper.state, cache.read({ + state: helper.state + }) || {}) - 1 + })).searchWithoutTriggeringOnStateChange(); + }; + }; - var connectHits = function connectHits(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$5()); - return function (widgetParams) { - var _ref = widgetParams || {}, - _ref$escapeHTML = _ref.escapeHTML, - escapeHTML = _ref$escapeHTML === void 0 ? true : _ref$escapeHTML, - _ref$transformItems = _ref.transformItems, - transformItems = _ref$transformItems === void 0 ? function (items) { - return items; - } : _ref$transformItems; + var getShowMore = function getShowMore(helper) { + return function () { + helper.setPage(getLastReceivedPage(helper.state, cache.read({ + state: helper.state + }) || {}) + 1).search(); + }; + }; - var sendEvent; - var bindEvent; return { - $$type: 'ais.hits', + $$type: 'ais.infiniteHits', init: function init(initOptions) { renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { instantSearchInstance: initOptions.instantSearchInstance }), true); }, render: function render(renderOptions) { - var renderState = this.getWidgetRenderState(renderOptions); - renderFn(_objectSpread2(_objectSpread2({}, renderState), {}, { - instantSearchInstance: renderOptions.instantSearchInstance + var instantSearchInstance = renderOptions.instantSearchInstance; + var widgetRenderState = this.getWidgetRenderState(renderOptions); + renderFn(_objectSpread2(_objectSpread2({}, widgetRenderState), {}, { + instantSearchInstance: instantSearchInstance }), false); - renderState.sendEvent('view', renderState.hits); + sendEvent('view', widgetRenderState.currentPageHits); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - hits: this.getWidgetRenderState(renderOptions) + infiniteHits: this.getWidgetRenderState(renderOptions) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref2) { - var results = _ref2.results, - helper = _ref2.helper, - instantSearchInstance = _ref2.instantSearchInstance; + getWidgetRenderState: function getWidgetRenderState(_ref5) { + var results = _ref5.results, + helper = _ref5.helper, + state = _ref5.state, + instantSearchInstance = _ref5.instantSearchInstance; + var isFirstPage; + var currentPageHits = []; + var cachedHits = cache.read({ + state: state + }) || {}; - if (!sendEvent) { + if (!results) { + showPrevious = getShowPrevious(helper); + showMore = getShowMore(helper); sendEvent = createSendEventForHits({ instantSearchInstance: instantSearchInstance, index: helper.getIndex(), widgetType: this.$$type }); - } - - if (!bindEvent) { bindEvent = createBindEventForHits({ index: helper.getIndex(), widgetType: this.$$type }); - } + isFirstPage = state.page === undefined || getFirstReceivedPage(state, cachedHits) === 0; + } else { + var _state$page3 = state.page, + _page = _state$page3 === void 0 ? 0 : _state$page3; - if (!results) { - return { - hits: [], - results: undefined, - sendEvent: sendEvent, - bindEvent: bindEvent, - widgetParams: widgetParams - }; - } + if (escapeHTML && results.hits.length > 0) { + results.hits = escapeHits(results.hits); + } - if (escapeHTML && results.hits.length > 0) { - results.hits = escapeHits(results.hits); + var hitsWithAbsolutePosition = addAbsolutePosition(results.hits, results.page, results.hitsPerPage); + var hitsWithAbsolutePositionAndQueryID = addQueryID(hitsWithAbsolutePosition, results.queryID); + var transformedHits = transformItems(hitsWithAbsolutePositionAndQueryID, { + results: results + }); + + if (cachedHits[_page] === undefined && !results.__isArtificial) { + cachedHits[_page] = transformedHits; + cache.write({ + state: state, + hits: cachedHits + }); + } + + currentPageHits = transformedHits; + isFirstPage = getFirstReceivedPage(state, cachedHits) === 0; } - var hitsWithAbsolutePosition = addAbsolutePosition(results.hits, results.page, results.hitsPerPage); - var hitsWithAbsolutePositionAndQueryID = addQueryID(hitsWithAbsolutePosition, results.queryID); - var transformedHits = transformItems(hitsWithAbsolutePositionAndQueryID, { - results: results - }); + var hits = extractHitsFromCachedHits(cachedHits); + var isLastPage = results ? results.nbPages <= getLastReceivedPage(state, cachedHits) + 1 : true; return { - hits: transformedHits, - results: results, + hits: hits, + currentPageHits: currentPageHits, sendEvent: sendEvent, bindEvent: bindEvent, + results: results, + showPrevious: showPrevious, + showMore: showMore, + isFirstPage: isFirstPage, + isLastPage: isLastPage, widgetParams: widgetParams }; }, - dispose: function dispose(_ref3) { - var state = _ref3.state; + dispose: function dispose(_ref6) { + var state = _ref6.state; unmountFn(); + var stateWithoutPage = state.setQueryParameter('page', undefined); if (!escapeHTML) { - return state; + return stateWithoutPage; } - return state.setQueryParameters(Object.keys(TAG_PLACEHOLDER).reduce(function (acc, key) { + return stateWithoutPage.setQueryParameters(Object.keys(TAG_PLACEHOLDER).reduce(function (acc, key) { return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, key, undefined)); }, {})); }, - getWidgetSearchParameters: function getWidgetSearchParameters(state) { - if (!escapeHTML) { - return state; - } - - return state.setQueryParameters(TAG_PLACEHOLDER); - } - }; - }; - }; - - var getSelectedHits = function getSelectedHits(hits, selectedObjectIDs) { - return selectedObjectIDs.map(function (objectID) { - var hit = find$1(hits, function (h) { - return h.objectID === objectID; - }); - - if (typeof hit === 'undefined') { - throw new Error("Could not find objectID \"".concat(objectID, "\" passed to `clickedObjectIDsAfterSearch` in the returned hits. This is necessary to infer the absolute position and the query ID.")); - } - - return hit; - }); - }; - - var getQueryID = function getQueryID(selectedHits) { - var queryIDs = uniq(selectedHits.map(function (hit) { - return hit.__queryID; - })); - - if (queryIDs.length > 1) { - throw new Error('Insights currently allows a single `queryID`. The `objectIDs` provided map to multiple `queryID`s.'); - } - - var queryID = queryIDs[0]; + getWidgetUiState: function getWidgetUiState(uiState, _ref7) { + var searchParameters = _ref7.searchParameters; + var page = searchParameters.page || 0; - if (typeof queryID !== 'string') { - throw new Error("Could not infer `queryID`. Ensure InstantSearch `clickAnalytics: true` was added with the Configure widget.\n\nSee: https://alg.li/lNiZZ7"); - } + if (!page) { + // return without adding `page` to uiState + // because we don't want `page=1` in the URL + return uiState; + } - return queryID; - }; + return _objectSpread2(_objectSpread2({}, uiState), {}, { + // The page in the UI state is incremented by one + // to expose the user value (not `0`). + page: page + 1 + }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref8) { + var uiState = _ref8.uiState; + var widgetSearchParameters = searchParameters; - var getPositions = function getPositions(selectedHits) { - return selectedHits.map(function (hit) { - return hit.__position; - }); - }; + if (escapeHTML) { + widgetSearchParameters = searchParameters.setQueryParameters(TAG_PLACEHOLDER); + } // The page in the search parameters is decremented by one + // to get to the actual parameter value from the UI state. - var inferPayload = function inferPayload(_ref) { - var method = _ref.method, - results = _ref.results, - hits = _ref.hits, - objectIDs = _ref.objectIDs; - var index = results.index; - var selectedHits = getSelectedHits(hits, objectIDs); - var queryID = getQueryID(selectedHits); - switch (method) { - case 'clickedObjectIDsAfterSearch': - { - var positions = getPositions(selectedHits); - return { - index: index, - queryID: queryID, - objectIDs: objectIDs, - positions: positions - }; + var page = uiState.page ? uiState.page - 1 : 0; + return widgetSearchParameters.setQueryParameter('page', page); } - - case 'convertedObjectIDsAfterSearch': - return { - index: index, - queryID: queryID, - objectIDs: objectIDs - }; - - default: - throw new Error("Unsupported method passed to insights: \"".concat(method, "\".")); - } + }; + }; }; - var wrapInsightsClient = function wrapInsightsClient(aa, results, hits) { - return function (method) { - for (var _len = arguments.length, payloads = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { - payloads[_key - 1] = arguments[_key]; - } + var connectInfiniteHitsWithInsights = withInsights(connectInfiniteHits); + + var withUsage$8 = createDocumentationMessageGenerator({ + name: 'menu', + connector: true + }); + var DEFAULT_SORT$1 = ['isRefined', 'name:asc']; - var payload = payloads[0]; - _warning(false, "`insights` function has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the `insights` middleware.\n\nFor more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/") ; + /** + * **Menu** connector provides the logic to build a widget that will give the user the ability to choose a single value for a specific facet. The typical usage of menu is for navigation in categories. + * + * This connector provides a `toggleShowMore()` function to display more or less items and a `refine()` + * function to select an item. While selecting a new element, the `refine` will also unselect the + * one that is currently selected. + * + * **Requirement:** the attribute passed as `attribute` must be present in "attributes for faceting" on the Algolia dashboard or configured as attributesForFaceting via a set settings call to the Algolia API. + */ + var connectMenu = function connectMenu(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$8()); + return function (widgetParams) { + var _ref = widgetParams || {}, + attribute = _ref.attribute, + _ref$limit = _ref.limit, + limit = _ref$limit === void 0 ? 10 : _ref$limit, + _ref$showMore = _ref.showMore, + showMore = _ref$showMore === void 0 ? false : _ref$showMore, + _ref$showMoreLimit = _ref.showMoreLimit, + showMoreLimit = _ref$showMoreLimit === void 0 ? 20 : _ref$showMoreLimit, + _ref$sortBy = _ref.sortBy, + sortBy = _ref$sortBy === void 0 ? DEFAULT_SORT$1 : _ref$sortBy, + _ref$transformItems = _ref.transformItems, + transformItems = _ref$transformItems === void 0 ? function (items) { + return items; + } : _ref$transformItems; - if (!aa) { - var withInstantSearchUsage = createDocumentationMessageGenerator({ - name: 'instantsearch' - }); - throw new Error(withInstantSearchUsage('The `insightsClient` option has not been provided to `instantsearch`.')); + if (!attribute) { + throw new Error(withUsage$8('The `attribute` option is required.')); } - if (!Array.isArray(payload.objectIDs)) { - throw new TypeError('Expected `objectIDs` to be an array.'); + if (showMore === true && showMoreLimit <= limit) { + throw new Error(withUsage$8('The `showMoreLimit` option must be greater than `limit`.')); } - var inferredPayload = inferPayload({ - method: method, - results: results, - hits: hits, - objectIDs: payload.objectIDs - }); - aa(method, _objectSpread2(_objectSpread2({}, inferredPayload), payload)); - }; - }; - /** - * @deprecated This function will be still supported in 4.x releases, but not further. It is replaced by the `insights` middleware. For more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/ - * It passes `insights` to `HitsWithInsightsListener` and `InfiniteHitsWithInsightsListener`. - */ + var sendEvent; + var _createURL; - function withInsights(connector) { - return function (renderFn, unmountFn) { - return connector(function (renderOptions, isFirstRender) { - var results = renderOptions.results, - hits = renderOptions.hits, - instantSearchInstance = renderOptions.instantSearchInstance; + var _refine; // Provide the same function to the `renderFn` so that way the user + // has to only bind it once when `isFirstRendering` for instance - if (results && hits && instantSearchInstance) { - var insights = wrapInsightsClient(instantSearchInstance.insightsClient, results, hits); - return renderFn(_objectSpread2(_objectSpread2({}, renderOptions), {}, { - insights: insights - }), isFirstRender); - } - return renderFn(renderOptions, isFirstRender); - }, unmountFn); - }; - } + var isShowingMore = false; - var n,l,u,t,r,o,f,e={},c=[],s=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i;function a(n,l){for(var u in l)n[u]=l[u];return n}function h(n){var l=n.parentNode;l&&l.removeChild(n);}function v(l,u,i){var t,r,o,f={};for(o in u)"key"==o?t=u[o]:"ref"==o?r=u[o]:f[o]=u[o];if(arguments.length>2&&(f.children=arguments.length>3?n.call(arguments,2):i),"function"==typeof l&&null!=l.defaultProps)for(o in l.defaultProps)void 0===f[o]&&(f[o]=l.defaultProps[o]);return y(l,f,t,r,null)}function y(n,i,t,r,o){var f={type:n,props:i,key:t,ref:r,__k:null,__:null,__b:0,__e:null,__d:void 0,__c:null,__h:null,constructor:void 0,__v:null==o?++u:o};return null==o&&null!=l.vnode&&l.vnode(f),f}function p(){return {current:null}}function d(n){return n.children}function _(n,l){this.props=n,this.context=l;}function k(n,l){if(null==l)return n.__?k(n.__,n.__.__k.indexOf(n)+1):null;for(var u;l0?y(_.type,_.props,_.key,null,_.__v):_)){if(_.__=u,_.__b=u.__b+1,null===(p=w[h])||p&&_.key==p.key&&_.type===p.type)w[h]=void 0;else for(v=0;v getLimit()); + items = transformItems(facetItems.slice(0, getLimit()).map(function (_ref3) { + var label = _ref3.name, + value = _ref3.escapedValue, + path = _ref3.path, + item = _objectWithoutProperties(_ref3, ["name", "escapedValue", "path"]); + + return _objectSpread2(_objectSpread2({}, item), {}, { + label: label, + value: value + }); + }), { + results: results }); } - } // old way, e.g. instantsearch.insights("clickedObjectIDsAfterSearch", { .. }) + return { + items: items, + createURL: _createURL, + refine: _refine, + sendEvent: sendEvent, + canRefine: items.length > 0, + widgetParams: widgetParams, + isShowingMore: isShowingMore, + toggleShowMore: cachedToggleShowMore, + canToggleShowMore: canToggleShowMore + }; + }, + getWidgetUiState: function getWidgetUiState(uiState, _ref4) { + var searchParameters = _ref4.searchParameters; - var insightsTarget = findInsightsTarget(event.target, event.currentTarget, function (element) { - return hasDataAttributes(element); - }); + var _searchParameters$get = searchParameters.getHierarchicalFacetBreadcrumb(attribute), + _searchParameters$get2 = _slicedToArray(_searchParameters$get, 1), + value = _searchParameters$get2[0]; - if (insightsTarget) { - var _readDataAttributes = readDataAttributes(insightsTarget), - method = _readDataAttributes.method, - _payload = _readDataAttributes.payload; + if (!value) { + return uiState; + } - props.insights(method, _payload); - } - }; + return _objectSpread2(_objectSpread2({}, uiState), {}, { + menu: _objectSpread2(_objectSpread2({}, uiState.menu), {}, _defineProperty({}, attribute, value)) + }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { + var uiState = _ref5.uiState; + var value = uiState.menu && uiState.menu[attribute]; + var withFacetConfiguration = searchParameters.removeHierarchicalFacet(attribute).addHierarchicalFacet({ + name: attribute, + attributes: [attribute] + }); + var currentMaxValuesPerFacet = withFacetConfiguration.maxValuesPerFacet || 0; + var nextMaxValuesPerFacet = Math.max(currentMaxValuesPerFacet, showMore ? showMoreLimit : limit); + var withMaxValuesPerFacet = withFacetConfiguration.setQueryParameter('maxValuesPerFacet', nextMaxValuesPerFacet); - return v("div", { - onClick: handleClick - }, v(BaseComponent, props)); - } + if (!value) { + return withMaxValuesPerFacet.setQueryParameters({ + hierarchicalFacetsRefinements: _objectSpread2(_objectSpread2({}, withMaxValuesPerFacet.hierarchicalFacetsRefinements), {}, _defineProperty({}, attribute, [])) + }); + } - return WithInsightsListener; + return withMaxValuesPerFacet.addHierarchicalFacetRefinement(attribute, value); + } + }; + }; }; - var connectHitsWithInsights = withInsights(connectHits); - - var withUsage$6 = createDocumentationMessageGenerator({ - name: 'hits-per-page', + var withUsage$9 = createDocumentationMessageGenerator({ + name: 'numeric-menu', connector: true }); + var $$type = 'ais.numericMenu'; - var connectHitsPerPage = function connectHitsPerPage(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$6()); - return function (widgetParams) { - var _ref = widgetParams || {}, - userItems = _ref.items, - _ref$transformItems = _ref.transformItems, - transformItems = _ref$transformItems === void 0 ? function (items) { - return items; - } : _ref$transformItems; - - if (!Array.isArray(userItems)) { - throw new Error(withUsage$6('The `items` option expects an array of objects.')); + var createSendEvent = function createSendEvent(_ref) { + var instantSearchInstance = _ref.instantSearchInstance; + return function () { + if (arguments.length === 1) { + instantSearchInstance.sendEventToInsights(arguments.length <= 0 ? undefined : arguments[0]); + return; } + }; + }; - var items = userItems; - var defaultItems = items.filter(function (item) { - return item.default === true; - }); + var connectNumericMenu = function connectNumericMenu(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$9()); + return function (widgetParams) { + var _ref2 = widgetParams || {}, + _ref2$attribute = _ref2.attribute, + attribute = _ref2$attribute === void 0 ? '' : _ref2$attribute, + _ref2$items = _ref2.items, + items = _ref2$items === void 0 ? [] : _ref2$items, + _ref2$transformItems = _ref2.transformItems, + transformItems = _ref2$transformItems === void 0 ? function (item) { + return item; + } : _ref2$transformItems; - if (defaultItems.length === 0) { - throw new Error(withUsage$6("A default value must be specified in `items`.")); + if (attribute === '') { + throw new Error(withUsage$9('The `attribute` option is required.')); } - if (defaultItems.length > 1) { - throw new Error(withUsage$6('More than one default value is specified in `items`.')); + if (!items || items.length === 0) { + throw new Error(withUsage$9('The `items` option expects an array of objects.')); } - var defaultItem = defaultItems[0]; - - var normalizeItems = function normalizeItems(_ref2) { - var hitsPerPage = _ref2.hitsPerPage; - return items.map(function (item) { - return _objectSpread2(_objectSpread2({}, item), {}, { - isRefined: Number(item.value) === Number(hitsPerPage) - }); + var prepareItems = function prepareItems(state) { + return items.map(function (_ref3) { + var start = _ref3.start, + end = _ref3.end, + label = _ref3.label; + return { + label: label, + value: encodeURI(JSON.stringify({ + start: start, + end: end + })), + isRefined: isRefined(state, attribute, { + start: start, + end: end, + label: label + }) + }; }); }; - var connectorState = { - getRefine: function getRefine(helper) { - return function (value) { - return !value && value !== 0 ? helper.setQueryParameter('hitsPerPage', undefined).search() : helper.setQueryParameter('hitsPerPage', value).search(); - }; - }, - createURLFactory: function createURLFactory(_ref3) { - var state = _ref3.state, - createURL = _ref3.createURL; - return function (value) { - return createURL(state.resetPage().setQueryParameter('hitsPerPage', !value && value !== 0 ? undefined : value)); - }; - } - }; + var connectorState = {}; return { - $$type: 'ais.hitsPerPage', + $$type: $$type, init: function init(initOptions) { - var state = initOptions.state, - instantSearchInstance = initOptions.instantSearchInstance; - var isCurrentInOptions = items.some(function (item) { - return Number(state.hitsPerPage) === Number(item.value); - }); - - if (!isCurrentInOptions) { - _warning(state.hitsPerPage !== undefined, "\n`hitsPerPage` is not defined.\nThe option `hitsPerPage` needs to be set using the `configure` widget.\n\nLearn more: https://www.algolia.com/doc/api-reference/widgets/hits-per-page/js/\n ") ; - _warning(false, "\nThe `items` option of `hitsPerPage` does not contain the \"hits per page\" value coming from the state: ".concat(state.hitsPerPage, ".\n\nYou may want to add another entry to the `items` option with this value.")) ; - items = [// The helper will convert the empty string to `undefined`. - { - value: '', - label: '' - }].concat(_toConsumableArray(items)); - } - + var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { instantSearchInstance: instantSearchInstance }), true); }, - render: function render(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { instantSearchInstance: instantSearchInstance }), false); }, dispose: function dispose(_ref4) { var state = _ref4.state; unmountFn(); - return state.setQueryParameter('hitsPerPage', undefined); + return state.clearRefinements(attribute); + }, + getWidgetUiState: function getWidgetUiState(uiState, _ref5) { + var searchParameters = _ref5.searchParameters; + var values = searchParameters.getNumericRefinements(attribute); + var equal = values['='] && values['='][0]; + + if (equal || equal === 0) { + return _objectSpread2(_objectSpread2({}, uiState), {}, { + numericMenu: _objectSpread2(_objectSpread2({}, uiState.numericMenu), {}, _defineProperty({}, attribute, "".concat(values['=']))) + }); + } + + var min = values['>='] && values['>='][0] || ''; + var max = values['<='] && values['<='][0] || ''; + + if (min === '' && max === '') { + return uiState; + } + + return _objectSpread2(_objectSpread2({}, uiState), {}, { + numericMenu: _objectSpread2(_objectSpread2({}, uiState.numericMenu), {}, _defineProperty({}, attribute, "".concat(min, ":").concat(max))) + }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { + var uiState = _ref6.uiState; + var value = uiState.numericMenu && uiState.numericMenu[attribute]; + var withoutRefinements = searchParameters.clearRefinements(attribute); + + if (!value) { + return withoutRefinements.setQueryParameters({ + numericRefinements: _objectSpread2(_objectSpread2({}, withoutRefinements.numericRefinements), {}, _defineProperty({}, attribute, {})) + }); + } + + var isExact = value.indexOf(':') === -1; + + if (isExact) { + return withoutRefinements.addNumericRefinement(attribute, '=', Number(value)); + } + + var _value$split$map = value.split(':').map(parseFloat), + _value$split$map2 = _slicedToArray(_value$split$map, 2), + min = _value$split$map2[0], + max = _value$split$map2[1]; + + var withMinRefinement = isFiniteNumber(min) ? withoutRefinements.addNumericRefinement(attribute, '>=', min) : withoutRefinements; + var withMaxRefinement = isFiniteNumber(max) ? withMinRefinement.addNumericRefinement(attribute, '<=', max) : withMinRefinement; + return withMaxRefinement; }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - hitsPerPage: this.getWidgetRenderState(renderOptions) + numericMenu: _objectSpread2(_objectSpread2({}, renderState.numericMenu), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref5) { - var state = _ref5.state, - results = _ref5.results, - createURL = _ref5.createURL, - helper = _ref5.helper; + getWidgetRenderState: function getWidgetRenderState(_ref7) { + var results = _ref7.results, + state = _ref7.state, + instantSearchInstance = _ref7.instantSearchInstance, + helper = _ref7.helper, + createURL = _ref7.createURL; + + if (!connectorState.refine) { + connectorState.refine = function (facetValue) { + var refinedState = getRefinedState(helper.state, attribute, facetValue); + connectorState.sendEvent('click', facetValue); + helper.setState(refinedState).search(); + }; + } + + if (!connectorState.createURL) { + connectorState.createURL = function (newState) { + return function (facetValue) { + return createURL(getRefinedState(newState, attribute, facetValue)); + }; + }; + } + + if (!connectorState.sendEvent) { + connectorState.sendEvent = createSendEvent({ + instantSearchInstance: instantSearchInstance + }); + } + + var hasNoResults = results ? results.nbHits === 0 : true; + var preparedItems = prepareItems(state); + var allIsSelected = true; + + var _iterator = _createForOfIteratorHelper(preparedItems), + _step; + + try { + for (_iterator.s(); !(_step = _iterator.n()).done;) { + var item = _step.value; + + if (item.isRefined && decodeURI(item.value) !== '{}') { + allIsSelected = false; + break; + } + } + } catch (err) { + _iterator.e(err); + } finally { + _iterator.f(); + } + return { - items: transformItems(normalizeItems(state), { + createURL: connectorState.createURL(state), + items: transformItems(preparedItems, { results: results }), - refine: connectorState.getRefine(helper), - createURL: connectorState.createURLFactory({ - state: state, - createURL: createURL - }), - hasNoResults: results ? results.nbHits === 0 : true, + hasNoResults: hasNoResults, + canRefine: !(hasNoResults && allIsSelected), + refine: connectorState.refine, + sendEvent: connectorState.sendEvent, widgetParams: widgetParams }; - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref6) { - var searchParameters = _ref6.searchParameters; - var hitsPerPage = searchParameters.hitsPerPage; - - if (hitsPerPage === undefined || hitsPerPage === defaultItem.value) { - return uiState; - } - - return _objectSpread2(_objectSpread2({}, uiState), {}, { - hitsPerPage: hitsPerPage - }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref7) { - var uiState = _ref7.uiState; - return searchParameters.setQueryParameters({ - hitsPerPage: uiState.hitsPerPage || defaultItem.value - }); } }; }; }; - var withUsage$7 = createDocumentationMessageGenerator({ - name: 'infinite-hits', - connector: true - }); - - function getStateWithoutPage(state) { - var _ref = state || {}, - page = _ref.page, - rest = _objectWithoutProperties(_ref, ["page"]); - - return rest; - } - - function getInMemoryCache() { - var cachedHits = null; - var cachedState = null; - return { - read: function read(_ref2) { - var state = _ref2.state; - return isEqual(cachedState, getStateWithoutPage(state)) ? cachedHits : null; - }, - write: function write(_ref3) { - var state = _ref3.state, - hits = _ref3.hits; - cachedState = getStateWithoutPage(state); - cachedHits = hits; + function isRefined(state, attribute, option) { + // @TODO: same as another spot, why is this mixing arrays & elements? + var currentRefinements = state.getNumericRefinements(attribute); + + if (option.start !== undefined && option.end !== undefined) { + if (option.start === option.end) { + return hasNumericRefinement(currentRefinements, '=', option.start); + } else { + return hasNumericRefinement(currentRefinements, '>=', option.start) && hasNumericRefinement(currentRefinements, '<=', option.end); } - }; - } + } - function extractHitsFromCachedHits(cachedHits) { - return Object.keys(cachedHits).map(Number).sort(function (a, b) { - return a - b; - }).reduce(function (acc, page) { - return acc.concat(cachedHits[page]); - }, []); - } + if (option.start !== undefined) { + return hasNumericRefinement(currentRefinements, '>=', option.start); + } - var connectInfiniteHits = function connectInfiniteHits(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$7()); // @TODO: this should be a generic, but a Connector can not yet be generic itself + if (option.end !== undefined) { + return hasNumericRefinement(currentRefinements, '<=', option.end); + } - return function (widgetParams) { - var _ref4 = widgetParams || {}, - _ref4$escapeHTML = _ref4.escapeHTML, - escapeHTML = _ref4$escapeHTML === void 0 ? true : _ref4$escapeHTML, - _ref4$transformItems = _ref4.transformItems, - transformItems = _ref4$transformItems === void 0 ? function (items) { - return items; - } : _ref4$transformItems, - _ref4$cache = _ref4.cache, - cache = _ref4$cache === void 0 ? getInMemoryCache() : _ref4$cache; + if (option.start === undefined && option.end === undefined) { + return Object.keys(currentRefinements).every(function (operator) { + return (currentRefinements[operator] || []).length === 0; + }); + } - var showPrevious; - var showMore; - var sendEvent; - var bindEvent; + return false; + } - var getFirstReceivedPage = function getFirstReceivedPage(state, cachedHits) { - var _state$page = state.page, - page = _state$page === void 0 ? 0 : _state$page; - var pages = Object.keys(cachedHits).map(Number); + function getRefinedState(state, attribute, facetValue) { + var resolvedState = state; + var refinedOption = JSON.parse(decodeURI(facetValue)); // @TODO: why is array / element mixed here & hasRefinements; seems wrong? - if (pages.length === 0) { - return page; - } else { - return Math.min.apply(Math, [page].concat(_toConsumableArray(pages))); - } - }; + var currentRefinements = resolvedState.getNumericRefinements(attribute); - var getLastReceivedPage = function getLastReceivedPage(state, cachedHits) { - var _state$page2 = state.page, - page = _state$page2 === void 0 ? 0 : _state$page2; - var pages = Object.keys(cachedHits).map(Number); + if (refinedOption.start === undefined && refinedOption.end === undefined) { + return resolvedState.removeNumericRefinement(attribute); + } - if (pages.length === 0) { - return page; + if (!isRefined(resolvedState, attribute, refinedOption)) { + resolvedState = resolvedState.removeNumericRefinement(attribute); + } + + if (refinedOption.start !== undefined && refinedOption.end !== undefined) { + if (refinedOption.start > refinedOption.end) { + throw new Error('option.start should be > to option.end'); + } + + if (refinedOption.start === refinedOption.end) { + if (hasNumericRefinement(currentRefinements, '=', refinedOption.start)) { + resolvedState = resolvedState.removeNumericRefinement(attribute, '=', refinedOption.start); } else { - return Math.max.apply(Math, [page].concat(_toConsumableArray(pages))); + resolvedState = resolvedState.addNumericRefinement(attribute, '=', refinedOption.start); } - }; - var getShowPrevious = function getShowPrevious(helper) { - return function () { - // Using the helper's `overrideStateWithoutTriggeringChangeEvent` method - // avoid updating the browser URL when the user displays the previous page. - helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread2(_objectSpread2({}, helper.state), {}, { - page: getFirstReceivedPage(helper.state, cache.read({ - state: helper.state - }) || {}) - 1 - })).searchWithoutTriggeringOnStateChange(); - }; - }; + return resolvedState; + } + } - var getShowMore = function getShowMore(helper) { - return function () { - helper.setPage(getLastReceivedPage(helper.state, cache.read({ - state: helper.state - }) || {}) + 1).search(); - }; - }; + if (refinedOption.start !== undefined) { + if (hasNumericRefinement(currentRefinements, '>=', refinedOption.start)) { + resolvedState = resolvedState.removeNumericRefinement(attribute, '>=', refinedOption.start); + } - return { - $$type: 'ais.infiniteHits', - init: function init(initOptions) { - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: initOptions.instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - var widgetRenderState = this.getWidgetRenderState(renderOptions); - renderFn(_objectSpread2(_objectSpread2({}, widgetRenderState), {}, { - instantSearchInstance: instantSearchInstance - }), false); - sendEvent('view', widgetRenderState.currentPageHits); - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - infiniteHits: this.getWidgetRenderState(renderOptions) - }); - }, - getWidgetRenderState: function getWidgetRenderState(_ref5) { - var results = _ref5.results, - helper = _ref5.helper, - state = _ref5.state, - instantSearchInstance = _ref5.instantSearchInstance; - var isFirstPage; - var currentPageHits = []; - var cachedHits = cache.read({ - state: state - }) || {}; + resolvedState = resolvedState.addNumericRefinement(attribute, '>=', refinedOption.start); + } - if (!results) { - showPrevious = getShowPrevious(helper); - showMore = getShowMore(helper); - sendEvent = createSendEventForHits({ - instantSearchInstance: instantSearchInstance, - index: helper.getIndex(), - widgetType: this.$$type - }); - bindEvent = createBindEventForHits({ - index: helper.getIndex(), - widgetType: this.$$type - }); - isFirstPage = state.page === undefined || getFirstReceivedPage(state, cachedHits) === 0; - } else { - var _state$page3 = state.page, - _page = _state$page3 === void 0 ? 0 : _state$page3; + if (refinedOption.end !== undefined) { + if (hasNumericRefinement(currentRefinements, '<=', refinedOption.end)) { + resolvedState = resolvedState.removeNumericRefinement(attribute, '<=', refinedOption.end); + } - if (escapeHTML && results.hits.length > 0) { - results.hits = escapeHits(results.hits); - } + resolvedState = resolvedState.addNumericRefinement(attribute, '<=', refinedOption.end); + } - var hitsWithAbsolutePosition = addAbsolutePosition(results.hits, results.page, results.hitsPerPage); - var hitsWithAbsolutePositionAndQueryID = addQueryID(hitsWithAbsolutePosition, results.queryID); - var transformedHits = transformItems(hitsWithAbsolutePositionAndQueryID, { - results: results - }); + if (typeof resolvedState.page === 'number') { + resolvedState.page = 0; + } - if (cachedHits[_page] === undefined && !results.__isArtificial) { - cachedHits[_page] = transformedHits; - cache.write({ - state: state, - hits: cachedHits - }); - } + return resolvedState; + } - currentPageHits = transformedHits; - isFirstPage = getFirstReceivedPage(state, cachedHits) === 0; - } + function hasNumericRefinement(currentRefinements, operator, value) { + return currentRefinements[operator] !== undefined && currentRefinements[operator].includes(value); + } - var hits = extractHitsFromCachedHits(cachedHits); - var isLastPage = results ? results.nbPages <= getLastReceivedPage(state, cachedHits) + 1 : true; - return { - hits: hits, - currentPageHits: currentPageHits, - sendEvent: sendEvent, - bindEvent: bindEvent, - results: results, - showPrevious: showPrevious, - showMore: showMore, - isFirstPage: isFirstPage, - isLastPage: isLastPage, - widgetParams: widgetParams - }; - }, - dispose: function dispose(_ref6) { - var state = _ref6.state; - unmountFn(); - var stateWithoutPage = state.setQueryParameter('page', undefined); + var Paginator = /*#__PURE__*/function () { + function Paginator(params) { + _classCallCheck(this, Paginator); - if (!escapeHTML) { - return stateWithoutPage; - } + _defineProperty(this, "currentPage", void 0); - return stateWithoutPage.setQueryParameters(Object.keys(TAG_PLACEHOLDER).reduce(function (acc, key) { - return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, key, undefined)); - }, {})); - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref7) { - var searchParameters = _ref7.searchParameters; - var page = searchParameters.page || 0; + _defineProperty(this, "total", void 0); - if (!page) { - // return without adding `page` to uiState - // because we don't want `page=1` in the URL - return uiState; - } + _defineProperty(this, "padding", void 0); - return _objectSpread2(_objectSpread2({}, uiState), {}, { - // The page in the UI state is incremented by one - // to expose the user value (not `0`). - page: page + 1 - }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref8) { - var uiState = _ref8.uiState; - var widgetSearchParameters = searchParameters; + this.currentPage = params.currentPage; + this.total = params.total; + this.padding = params.padding; + } - if (escapeHTML) { - widgetSearchParameters = searchParameters.setQueryParameters(TAG_PLACEHOLDER); - } // The page in the search parameters is decremented by one - // to get to the actual parameter value from the UI state. + _createClass(Paginator, [{ + key: "pages", + value: function pages() { + var total = this.total, + currentPage = this.currentPage, + padding = this.padding; + if (total === 0) return [0]; + var totalDisplayedPages = this.nbPagesDisplayed(padding, total); + if (totalDisplayedPages === total) { + return range({ + end: total + }); + } - var page = uiState.page ? uiState.page - 1 : 0; - return widgetSearchParameters.setQueryParameter('page', page); + var paddingLeft = this.calculatePaddingLeft(currentPage, padding, total, totalDisplayedPages); + var paddingRight = totalDisplayedPages - paddingLeft; + var first = currentPage - paddingLeft; + var last = currentPage + paddingRight; + return range({ + start: first, + end: last + }); + } + }, { + key: "nbPagesDisplayed", + value: function nbPagesDisplayed(padding, total) { + return Math.min(2 * padding + 1, total); + } + }, { + key: "calculatePaddingLeft", + value: function calculatePaddingLeft(current, padding, total, totalDisplayedPages) { + if (current <= padding) { + return current; } - }; - }; - }; - var connectInfiniteHitsWithInsights = withInsights(connectInfiniteHits); + if (current >= total - padding) { + return totalDisplayedPages - (total - current); + } - var withUsage$8 = createDocumentationMessageGenerator({ - name: 'menu', + return padding; + } + }, { + key: "isLastPage", + value: function isLastPage() { + return this.currentPage === this.total - 1 || this.total === 0; + } + }, { + key: "isFirstPage", + value: function isFirstPage() { + return this.currentPage === 0; + } + }]); + + return Paginator; + }(); + + var withUsage$a = createDocumentationMessageGenerator({ + name: 'pagination', connector: true }); - var DEFAULT_SORT$1 = ['isRefined', 'name:asc']; /** - * **Menu** connector provides the logic to build a widget that will give the user the ability to choose a single value for a specific facet. The typical usage of menu is for navigation in categories. - * - * This connector provides a `toggleShowMore()` function to display more or less items and a `refine()` - * function to select an item. While selecting a new element, the `refine` will also unselect the - * one that is currently selected. + * **Pagination** connector provides the logic to build a widget that will let the user + * choose the current page of the results. * - * **Requirement:** the attribute passed as `attribute` must be present in "attributes for faceting" on the Algolia dashboard or configured as attributesForFaceting via a set settings call to the Algolia API. + * When using the pagination with Algolia, you should be aware that the engine won't provide you pages + * beyond the 1000th hits by default. You can find more information on the [Algolia documentation](https://www.algolia.com/doc/guides/searching/pagination/#pagination-limitations). */ - var connectMenu = function connectMenu(renderFn) { + var connectPagination = function connectPagination(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$8()); + checkRendering(renderFn, withUsage$a()); return function (widgetParams) { var _ref = widgetParams || {}, - attribute = _ref.attribute, - _ref$limit = _ref.limit, - limit = _ref$limit === void 0 ? 10 : _ref$limit, - _ref$showMore = _ref.showMore, - showMore = _ref$showMore === void 0 ? false : _ref$showMore, - _ref$showMoreLimit = _ref.showMoreLimit, - showMoreLimit = _ref$showMoreLimit === void 0 ? 20 : _ref$showMoreLimit, - _ref$sortBy = _ref.sortBy, - sortBy = _ref$sortBy === void 0 ? DEFAULT_SORT$1 : _ref$sortBy, - _ref$transformItems = _ref.transformItems, - transformItems = _ref$transformItems === void 0 ? function (items) { - return items; - } : _ref$transformItems; - - if (!attribute) { - throw new Error(withUsage$8('The `attribute` option is required.')); - } - - if (showMore === true && showMoreLimit <= limit) { - throw new Error(withUsage$8('The `showMoreLimit` option must be greater than `limit`.')); - } - - var sendEvent; - - var _createURL; - - var _refine; // Provide the same function to the `renderFn` so that way the user - // has to only bind it once when `isFirstRendering` for instance - - - var isShowingMore = false; - - var toggleShowMore = function toggleShowMore() {}; - - function createToggleShowMore(renderOptions, widget) { - return function () { - isShowingMore = !isShowingMore; - widget.render(renderOptions); - }; - } + totalPages = _ref.totalPages, + _ref$padding = _ref.padding, + padding = _ref$padding === void 0 ? 3 : _ref$padding; - function cachedToggleShowMore() { - toggleShowMore(); - } + var pager = new Paginator({ + currentPage: 0, + total: 0, + padding: padding + }); + var connectorState = {}; - function getLimit() { - return isShowingMore ? showMoreLimit : limit; + function getMaxPage(_ref2) { + var nbPages = _ref2.nbPages; + return totalPages !== undefined ? Math.min(totalPages, nbPages) : nbPages; } return { - $$type: 'ais.menu', + $$type: 'ais.pagination', init: function init(initOptions) { var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { @@ -11906,531 +11745,708 @@ instantSearchInstance: instantSearchInstance }), false); }, - dispose: function dispose(_ref2) { - var state = _ref2.state; + dispose: function dispose(_ref3) { + var state = _ref3.state; unmountFn(); - return state.removeHierarchicalFacet(attribute).setQueryParameter('maxValuesPerFacet', undefined); + return state.setQueryParameter('page', undefined); }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - menu: _objectSpread2(_objectSpread2({}, renderState.menu), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) + getWidgetUiState: function getWidgetUiState(uiState, _ref4) { + var searchParameters = _ref4.searchParameters; + var page = searchParameters.page || 0; + + if (!page) { + return uiState; + } + + return _objectSpread2(_objectSpread2({}, uiState), {}, { + page: page + 1 }); }, - getWidgetRenderState: function getWidgetRenderState(renderOptions) { - var results = renderOptions.results, - createURL = renderOptions.createURL, - instantSearchInstance = renderOptions.instantSearchInstance, - helper = renderOptions.helper; - var items = []; - var canToggleShowMore = false; + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { + var uiState = _ref5.uiState; + var page = uiState.page ? uiState.page - 1 : 0; + return searchParameters.setQueryParameter('page', page); + }, + getWidgetRenderState: function getWidgetRenderState(_ref6) { + var results = _ref6.results, + helper = _ref6.helper, + state = _ref6.state, + createURL = _ref6.createURL; - if (!sendEvent) { - sendEvent = createSendEventForFacet({ - instantSearchInstance: instantSearchInstance, - helper: helper, - attribute: attribute, - widgetType: this.$$type - }); + if (!connectorState.refine) { + connectorState.refine = function (page) { + helper.setPage(page); + helper.search(); + }; } - if (!_createURL) { - _createURL = function _createURL(facetValue) { - return createURL(helper.state.resetPage().toggleFacetRefinement(attribute, facetValue)); - }; - } + if (!connectorState.createURL) { + connectorState.createURL = function (helperState) { + return function (page) { + return createURL(helperState.setPage(page)); + }; + }; + } + + var page = state.page || 0; + var nbPages = getMaxPage(results || { + nbPages: 0 + }); + pager.currentPage = page; + pager.total = nbPages; + return { + createURL: connectorState.createURL(state), + refine: connectorState.refine, + canRefine: nbPages > 1, + currentRefinement: page, + nbHits: (results === null || results === void 0 ? void 0 : results.nbHits) || 0, + nbPages: nbPages, + pages: results ? pager.pages() : [], + isFirstPage: pager.isFirstPage(), + isLastPage: pager.isLastPage(), + widgetParams: widgetParams + }; + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + pagination: this.getWidgetRenderState(renderOptions) + }); + } + }; + }; + }; + + var withUsage$b = createDocumentationMessageGenerator({ + name: 'range-input', + connector: true + }, { + name: 'range-slider', + connector: true + }); + var $$type$1 = 'ais.range'; + + function toPrecision(_ref) { + var min = _ref.min, + max = _ref.max, + precision = _ref.precision; + var pow = Math.pow(10, precision); + return { + min: min ? Math.floor(min * pow) / pow : min, + max: max ? Math.ceil(max * pow) / pow : max + }; + } + /** + * **Range** connector provides the logic to create custom widget that will let + * the user refine results using a numeric range. + * + * This connectors provides a `refine()` function that accepts bounds. It will also provide + * information about the min and max bounds for the current result set. + */ + + + var connectRange = function connectRange(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$b()); + return function (widgetParams) { + var _ref2 = widgetParams || {}, + _ref2$attribute = _ref2.attribute, + attribute = _ref2$attribute === void 0 ? '' : _ref2$attribute, + minBound = _ref2.min, + maxBound = _ref2.max, + _ref2$precision = _ref2.precision, + precision = _ref2$precision === void 0 ? 0 : _ref2$precision; + + if (!attribute) { + throw new Error(withUsage$b('The `attribute` option is required.')); + } + + if (isFiniteNumber(minBound) && isFiniteNumber(maxBound) && minBound > maxBound) { + throw new Error(withUsage$b("The `max` option can't be lower than `min`.")); + } - if (!_refine) { - _refine = function _refine(facetValue) { - var _helper$getHierarchic = helper.getHierarchicalFacetBreadcrumb(attribute), - _helper$getHierarchic2 = _slicedToArray(_helper$getHierarchic, 1), - refinedItem = _helper$getHierarchic2[0]; + var formatToNumber = function formatToNumber(v) { + return Number(Number(v).toFixed(precision)); + }; - sendEvent('click', facetValue ? facetValue : refinedItem); - helper.toggleFacetRefinement(attribute, facetValue ? facetValue : refinedItem).search(); - }; - } + var rangeFormatter = { + from: function from(v) { + return v.toLocaleString(); + }, + to: function to(v) { + return formatToNumber(v).toLocaleString(); + } + }; // eslint-disable-next-line complexity - if (renderOptions.results) { - toggleShowMore = createToggleShowMore(renderOptions, this); - } + var getRefinedState = function getRefinedState(helper, currentRange, nextMin, nextMax) { + var resolvedState = helper.state; + var currentRangeMin = currentRange.min, + currentRangeMax = currentRange.max; - if (results) { - var facetValues = results.getFacetValues(attribute, { - sortBy: sortBy, - facetOrdering: sortBy === DEFAULT_SORT$1 - }); - var facetItems = facetValues && !Array.isArray(facetValues) && facetValues.data ? facetValues.data : []; - canToggleShowMore = showMore && (isShowingMore || facetItems.length > getLimit()); - items = transformItems(facetItems.slice(0, getLimit()).map(function (_ref3) { - var label = _ref3.name, - value = _ref3.escapedValue, - path = _ref3.path, - item = _objectWithoutProperties(_ref3, ["name", "escapedValue", "path"]); + var _ref3 = resolvedState.getNumericRefinement(attribute, '>=') || [], + _ref4 = _slicedToArray(_ref3, 1), + min = _ref4[0]; - return _objectSpread2(_objectSpread2({}, item), {}, { - label: label, - value: value - }); - }), { - results: results - }); - } + var _ref5 = resolvedState.getNumericRefinement(attribute, '<=') || [], + _ref6 = _slicedToArray(_ref5, 1), + max = _ref6[0]; - return { - items: items, - createURL: _createURL, - refine: _refine, - sendEvent: sendEvent, - canRefine: items.length > 0, - widgetParams: widgetParams, - isShowingMore: isShowingMore, - toggleShowMore: cachedToggleShowMore, - canToggleShowMore: canToggleShowMore - }; - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref4) { - var searchParameters = _ref4.searchParameters; + var isResetMin = nextMin === undefined || nextMin === ''; + var isResetMax = nextMax === undefined || nextMax === ''; - var _searchParameters$get = searchParameters.getHierarchicalFacetBreadcrumb(attribute), - _searchParameters$get2 = _slicedToArray(_searchParameters$get, 1), - value = _searchParameters$get2[0]; + var _toPrecision = toPrecision({ + min: !isResetMin ? parseFloat(nextMin) : undefined, + max: !isResetMax ? parseFloat(nextMax) : undefined, + precision: precision + }), + nextMinAsNumber = _toPrecision.min, + nextMaxAsNumber = _toPrecision.max; - if (!value) { - return uiState; - } + var newNextMin; - return _objectSpread2(_objectSpread2({}, uiState), {}, { - menu: _objectSpread2(_objectSpread2({}, uiState.menu), {}, _defineProperty({}, attribute, value)) - }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { - var uiState = _ref5.uiState; - var value = uiState.menu && uiState.menu[attribute]; - var withFacetConfiguration = searchParameters.removeHierarchicalFacet(attribute).addHierarchicalFacet({ - name: attribute, - attributes: [attribute] - }); - var currentMaxValuesPerFacet = withFacetConfiguration.maxValuesPerFacet || 0; - var nextMaxValuesPerFacet = Math.max(currentMaxValuesPerFacet, showMore ? showMoreLimit : limit); - var withMaxValuesPerFacet = withFacetConfiguration.setQueryParameter('maxValuesPerFacet', nextMaxValuesPerFacet); + if (!isFiniteNumber(minBound) && currentRangeMin === nextMinAsNumber) { + newNextMin = undefined; + } else if (isFiniteNumber(minBound) && isResetMin) { + newNextMin = minBound; + } else { + newNextMin = nextMinAsNumber; + } - if (!value) { - return withMaxValuesPerFacet.setQueryParameters({ - hierarchicalFacetsRefinements: _objectSpread2(_objectSpread2({}, withMaxValuesPerFacet.hierarchicalFacetsRefinements), {}, _defineProperty({}, attribute, [])) - }); - } + var newNextMax; - return withMaxValuesPerFacet.addHierarchicalFacetRefinement(attribute, value); + if (!isFiniteNumber(maxBound) && currentRangeMax === nextMaxAsNumber) { + newNextMax = undefined; + } else if (isFiniteNumber(maxBound) && isResetMax) { + newNextMax = maxBound; + } else { + newNextMax = nextMaxAsNumber; } - }; - }; - }; - var withUsage$9 = createDocumentationMessageGenerator({ - name: 'numeric-menu', - connector: true - }); - var $$type = 'ais.numericMenu'; + var isResetNewNextMin = newNextMin === undefined; + var isGreaterThanCurrentRange = isFiniteNumber(currentRangeMin) && currentRangeMin <= newNextMin; + var isMinValid = isResetNewNextMin || isFiniteNumber(newNextMin) && (!isFiniteNumber(currentRangeMin) || isGreaterThanCurrentRange); + var isResetNewNextMax = newNextMax === undefined; + var isLowerThanRange = isFiniteNumber(newNextMax) && currentRangeMax >= newNextMax; + var isMaxValid = isResetNewNextMax || isFiniteNumber(newNextMax) && (!isFiniteNumber(currentRangeMax) || isLowerThanRange); + var hasMinChange = min !== newNextMin; + var hasMaxChange = max !== newNextMax; - var createSendEvent = function createSendEvent(_ref) { - var instantSearchInstance = _ref.instantSearchInstance, - helper = _ref.helper, - attribute = _ref.attribute; - return function () { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } + if ((hasMinChange || hasMaxChange) && isMinValid && isMaxValid) { + resolvedState = resolvedState.removeNumericRefinement(attribute); - if (args.length === 1) { - instantSearchInstance.sendEventToInsights(args[0]); - return; - } + if (isFiniteNumber(newNextMin)) { + resolvedState = resolvedState.addNumericRefinement(attribute, '>=', newNextMin); + } - var eventType = args[0], - facetValue = args[1], - _args$ = args[2], - eventName = _args$ === void 0 ? 'Filter Applied' : _args$; + if (isFiniteNumber(newNextMax)) { + resolvedState = resolvedState.addNumericRefinement(attribute, '<=', newNextMax); + } - if (eventType !== 'click') { - return; - } // facetValue === "%7B%22start%22:5,%22end%22:10%7D" + return resolvedState.resetPage(); + } + return null; + }; - var filters = convertNumericRefinementsToFilters(getRefinedState(helper.state, attribute, facetValue), attribute); + var createSendEvent = function createSendEvent(instantSearchInstance) { + return function () { + if (arguments.length === 1) { + instantSearchInstance.sendEventToInsights(arguments.length <= 0 ? undefined : arguments[0]); + return; + } + }; + }; - if (filters && filters.length > 0) { - /* - filters === ["price<=10", "price>=5"] - */ - instantSearchInstance.sendEventToInsights({ - insightsMethod: 'clickedFilters', - widgetType: $$type, - eventType: eventType, - payload: { - eventName: eventName, - index: helper.getIndex(), - filters: filters - }, - attribute: attribute - }); - } - }; - }; + function _getCurrentRange(stats) { + var min; - var connectNumericMenu = function connectNumericMenu(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$9()); - return function (widgetParams) { - var _ref2 = widgetParams || {}, - _ref2$attribute = _ref2.attribute, - attribute = _ref2$attribute === void 0 ? '' : _ref2$attribute, - _ref2$items = _ref2.items, - items = _ref2$items === void 0 ? [] : _ref2$items, - _ref2$transformItems = _ref2.transformItems, - transformItems = _ref2$transformItems === void 0 ? function (item) { - return item; - } : _ref2$transformItems; + if (isFiniteNumber(minBound)) { + min = minBound; + } else if (isFiniteNumber(stats.min)) { + min = stats.min; + } else { + min = 0; + } - if (attribute === '') { - throw new Error(withUsage$9('The `attribute` option is required.')); - } + var max; - if (!items || items.length === 0) { - throw new Error(withUsage$9('The `items` option expects an array of objects.')); - } + if (isFiniteNumber(maxBound)) { + max = maxBound; + } else if (isFiniteNumber(stats.max)) { + max = stats.max; + } else { + max = 0; + } - var prepareItems = function prepareItems(state) { - return items.map(function (_ref3) { - var start = _ref3.start, - end = _ref3.end, - label = _ref3.label; - return { - label: label, - value: encodeURI(JSON.stringify({ - start: start, - end: end - })), - isRefined: isRefined(state, attribute, { - start: start, - end: end, - label: label - }) - }; + return toPrecision({ + min: min, + max: max, + precision: precision }); - }; + } + + function _getCurrentRefinement(helper) { + var _ref7 = helper.getNumericRefinement(attribute, '>=') || [], + _ref8 = _slicedToArray(_ref7, 1), + minValue = _ref8[0]; + + var _ref9 = helper.getNumericRefinement(attribute, '<=') || [], + _ref10 = _slicedToArray(_ref9, 1), + maxValue = _ref10[0]; + + var min = isFiniteNumber(minValue) ? minValue : -Infinity; + var max = isFiniteNumber(maxValue) ? maxValue : Infinity; + return [min, max]; + } + + function _refine(helper, currentRange) { + return function () { + var _ref11 = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [undefined, undefined], + _ref12 = _slicedToArray(_ref11, 2), + nextMin = _ref12[0], + nextMax = _ref12[1]; + + var refinedState = getRefinedState(helper, currentRange, nextMin, nextMax); + + if (refinedState) { + helper.setState(refinedState).search(); + } + }; + } - var connectorState = {}; return { - $$type: $$type, + $$type: $$type$1, init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance + instantSearchInstance: initOptions.instantSearchInstance }), true); }, render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance + instantSearchInstance: renderOptions.instantSearchInstance }), false); }, - dispose: function dispose(_ref4) { - var state = _ref4.state; - unmountFn(); - return state.clearRefinements(attribute); + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + range: _objectSpread2(_objectSpread2({}, renderState.range), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) + }); }, - getWidgetUiState: function getWidgetUiState(uiState, _ref5) { - var searchParameters = _ref5.searchParameters; - var values = searchParameters.getNumericRefinements(attribute); - var equal = values['='] && values['='][0]; + getWidgetRenderState: function getWidgetRenderState(_ref13) { + var results = _ref13.results, + helper = _ref13.helper, + instantSearchInstance = _ref13.instantSearchInstance; + var facetsFromResults = results && results.disjunctiveFacets || []; + var facet = find$1(facetsFromResults, function (facetResult) { + return facetResult.name === attribute; + }); + var stats = facet && facet.stats || { + min: undefined, + max: undefined + }; - if (equal || equal === 0) { - return _objectSpread2(_objectSpread2({}, uiState), {}, { - numericMenu: _objectSpread2(_objectSpread2({}, uiState.numericMenu), {}, _defineProperty({}, attribute, "".concat(values['=']))) + var currentRange = _getCurrentRange(stats); + + var start = _getCurrentRefinement(helper); + + var refine; + + if (!results) { + // On first render pass an empty range + // to be able to bypass the validation + // related to it + refine = _refine(helper, { + min: undefined, + max: undefined }); + } else { + refine = _refine(helper, currentRange); } - var min = values['>='] && values['>='][0] || ''; - var max = values['<='] && values['<='][0] || ''; + return { + refine: refine, + canRefine: currentRange.min !== currentRange.max, + format: rangeFormatter, + range: currentRange, + sendEvent: createSendEvent(instantSearchInstance), + widgetParams: _objectSpread2(_objectSpread2({}, widgetParams), {}, { + precision: precision + }), + start: start + }; + }, + dispose: function dispose(_ref14) { + var state = _ref14.state; + unmountFn(); + return state.removeDisjunctiveFacet(attribute).removeNumericRefinement(attribute); + }, + getWidgetUiState: function getWidgetUiState(uiState, _ref15) { + var searchParameters = _ref15.searchParameters; - if (min === '' && max === '') { + var _searchParameters$get = searchParameters.getNumericRefinements(attribute), + _searchParameters$get2 = _searchParameters$get['>='], + min = _searchParameters$get2 === void 0 ? [] : _searchParameters$get2, + _searchParameters$get3 = _searchParameters$get['<='], + max = _searchParameters$get3 === void 0 ? [] : _searchParameters$get3; + + if (min.length === 0 && max.length === 0) { return uiState; } return _objectSpread2(_objectSpread2({}, uiState), {}, { - numericMenu: _objectSpread2(_objectSpread2({}, uiState.numericMenu), {}, _defineProperty({}, attribute, "".concat(min, ":").concat(max))) + range: _objectSpread2(_objectSpread2({}, uiState.range), {}, _defineProperty({}, attribute, "".concat(min, ":").concat(max))) }); }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { - var uiState = _ref6.uiState; - var value = uiState.numericMenu && uiState.numericMenu[attribute]; - var withoutRefinements = searchParameters.clearRefinements(attribute); + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref16) { + var uiState = _ref16.uiState; + var widgetSearchParameters = searchParameters.addDisjunctiveFacet(attribute).setQueryParameters({ + numericRefinements: _objectSpread2(_objectSpread2({}, searchParameters.numericRefinements), {}, _defineProperty({}, attribute, {})) + }); - if (!value) { - return withoutRefinements.setQueryParameters({ - numericRefinements: _objectSpread2(_objectSpread2({}, withoutRefinements.numericRefinements), {}, _defineProperty({}, attribute, {})) - }); + if (isFiniteNumber(minBound)) { + widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '>=', minBound); } - var isExact = value.indexOf(':') === -1; + if (isFiniteNumber(maxBound)) { + widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '<=', maxBound); + } - if (isExact) { - return withoutRefinements.addNumericRefinement(attribute, '=', Number(value)); + var value = uiState.range && uiState.range[attribute]; + + if (!value || value.indexOf(':') === -1) { + return widgetSearchParameters; } var _value$split$map = value.split(':').map(parseFloat), _value$split$map2 = _slicedToArray(_value$split$map, 2), - min = _value$split$map2[0], - max = _value$split$map2[1]; - - var withMinRefinement = isFiniteNumber(min) ? withoutRefinements.addNumericRefinement(attribute, '>=', min) : withoutRefinements; - var withMaxRefinement = isFiniteNumber(max) ? withMinRefinement.addNumericRefinement(attribute, '<=', max) : withMinRefinement; - return withMaxRefinement; - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - numericMenu: _objectSpread2(_objectSpread2({}, renderState.numericMenu), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) - }); - }, - getWidgetRenderState: function getWidgetRenderState(_ref7) { - var results = _ref7.results, - state = _ref7.state, - instantSearchInstance = _ref7.instantSearchInstance, - helper = _ref7.helper, - createURL = _ref7.createURL; - - if (!connectorState.refine) { - connectorState.refine = function (facetValue) { - var refinedState = getRefinedState(helper.state, attribute, facetValue); - connectorState.sendEvent('click', facetValue); - helper.setState(refinedState).search(); - }; - } + lowerBound = _value$split$map2[0], + upperBound = _value$split$map2[1]; - if (!connectorState.createURL) { - connectorState.createURL = function (newState) { - return function (facetValue) { - return createURL(getRefinedState(newState, attribute, facetValue)); - }; - }; + if (isFiniteNumber(lowerBound) && (!isFiniteNumber(minBound) || minBound < lowerBound)) { + widgetSearchParameters = widgetSearchParameters.removeNumericRefinement(attribute, '>='); + widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '>=', lowerBound); } - if (!connectorState.sendEvent) { - connectorState.sendEvent = createSendEvent({ - instantSearchInstance: instantSearchInstance, - helper: helper, - attribute: attribute - }); + if (isFiniteNumber(upperBound) && (!isFiniteNumber(maxBound) || upperBound < maxBound)) { + widgetSearchParameters = widgetSearchParameters.removeNumericRefinement(attribute, '<='); + widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '<=', upperBound); } - return { - createURL: connectorState.createURL(state), - items: transformItems(prepareItems(state), { - results: results - }), - hasNoResults: results ? results.nbHits === 0 : true, - refine: connectorState.refine, - sendEvent: connectorState.sendEvent, - widgetParams: widgetParams - }; + return widgetSearchParameters; } }; }; }; - function isRefined(state, attribute, option) { - // @TODO: same as another spot, why is this mixing arrays & elements? - var currentRefinements = state.getNumericRefinements(attribute); - - if (option.start !== undefined && option.end !== undefined) { - if (option.start === option.end) { - return hasNumericRefinement(currentRefinements, '=', option.start); - } else { - return hasNumericRefinement(currentRefinements, '>=', option.start) && hasNumericRefinement(currentRefinements, '<=', option.end); - } - } + var withUsage$c = createDocumentationMessageGenerator({ + name: 'refinement-list', + connector: true + }); + var DEFAULT_SORT$2 = ['isRefined', 'count:desc', 'name:asc']; - if (option.start !== undefined) { - return hasNumericRefinement(currentRefinements, '>=', option.start); - } + /** + * **RefinementList** connector provides the logic to build a custom widget that + * will let the user filter the results based on the values of a specific facet. + * + * **Requirement:** the attribute passed as `attribute` must be present in + * attributesForFaceting of the searched index. + * + * This connector provides: + * - a `refine()` function to select an item. + * - a `toggleShowMore()` function to display more or less items + * - a `searchForItems()` function to search within the items. + */ + var connectRefinementList = function connectRefinementList(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$c()); + return function (widgetParams) { + var _ref = widgetParams || {}, + attribute = _ref.attribute, + _ref$operator = _ref.operator, + operator = _ref$operator === void 0 ? 'or' : _ref$operator, + _ref$limit = _ref.limit, + limit = _ref$limit === void 0 ? 10 : _ref$limit, + _ref$showMore = _ref.showMore, + showMore = _ref$showMore === void 0 ? false : _ref$showMore, + _ref$showMoreLimit = _ref.showMoreLimit, + showMoreLimit = _ref$showMoreLimit === void 0 ? 20 : _ref$showMoreLimit, + _ref$sortBy = _ref.sortBy, + sortBy = _ref$sortBy === void 0 ? DEFAULT_SORT$2 : _ref$sortBy, + _ref$escapeFacetValue = _ref.escapeFacetValues, + escapeFacetValues = _ref$escapeFacetValue === void 0 ? true : _ref$escapeFacetValue, + _ref$transformItems = _ref.transformItems, + transformItems = _ref$transformItems === void 0 ? function (items) { + return items; + } : _ref$transformItems; - if (option.end !== undefined) { - return hasNumericRefinement(currentRefinements, '<=', option.end); - } + if (!attribute) { + throw new Error(withUsage$c('The `attribute` option is required.')); + } - if (option.start === undefined && option.end === undefined) { - return Object.keys(currentRefinements).every(function (operator) { - return (currentRefinements[operator] || []).length === 0; - }); - } + if (!/^(and|or)$/.test(operator)) { + throw new Error(withUsage$c("The `operator` must one of: `\"and\"`, `\"or\"` (got \"".concat(operator, "\")."))); + } - return false; - } + if (showMore === true && showMoreLimit <= limit) { + throw new Error(withUsage$c('`showMoreLimit` should be greater than `limit`.')); + } - function getRefinedState(state, attribute, facetValue) { - var resolvedState = state; - var refinedOption = JSON.parse(decodeURI(facetValue)); // @TODO: why is array / element mixed here & hasRefinements; seems wrong? + var formatItems = function formatItems(_ref2) { + var label = _ref2.name, + value = _ref2.escapedValue, + item = _objectWithoutProperties(_ref2, ["name", "escapedValue"]); - var currentRefinements = resolvedState.getNumericRefinements(attribute); + return _objectSpread2(_objectSpread2({}, item), {}, { + value: value, + label: label, + highlighted: label + }); + }; - if (refinedOption.start === undefined && refinedOption.end === undefined) { - return resolvedState.removeNumericRefinement(attribute); - } + var lastResultsFromMainSearch; + var lastItemsFromMainSearch = []; + var hasExhaustiveItems = true; + var triggerRefine; + var sendEvent; + var isShowingMore = false; // Provide the same function to the `renderFn` so that way the user + // has to only bind it once when `isFirstRendering` for instance - if (!isRefined(resolvedState, attribute, refinedOption)) { - resolvedState = resolvedState.removeNumericRefinement(attribute); - } + var toggleShowMore = function toggleShowMore() {}; - if (refinedOption.start !== undefined && refinedOption.end !== undefined) { - if (refinedOption.start > refinedOption.end) { - throw new Error('option.start should be > to option.end'); + function cachedToggleShowMore() { + toggleShowMore(); } - if (refinedOption.start === refinedOption.end) { - if (hasNumericRefinement(currentRefinements, '=', refinedOption.start)) { - resolvedState = resolvedState.removeNumericRefinement(attribute, '=', refinedOption.start); - } else { - resolvedState = resolvedState.addNumericRefinement(attribute, '=', refinedOption.start); - } - - return resolvedState; + function createToggleShowMore(renderOptions, widget) { + return function () { + isShowingMore = !isShowingMore; + widget.render(renderOptions); + }; } - } - if (refinedOption.start !== undefined) { - if (hasNumericRefinement(currentRefinements, '>=', refinedOption.start)) { - resolvedState = resolvedState.removeNumericRefinement(attribute, '>=', refinedOption.start); + function getLimit() { + return isShowingMore ? showMoreLimit : limit; } - resolvedState = resolvedState.addNumericRefinement(attribute, '>=', refinedOption.start); - } + var searchForFacetValues = function searchForFacetValues() { + return function () {}; + }; - if (refinedOption.end !== undefined) { - if (hasNumericRefinement(currentRefinements, '<=', refinedOption.end)) { - resolvedState = resolvedState.removeNumericRefinement(attribute, '<=', refinedOption.end); - } + var createSearchForFacetValues = function createSearchForFacetValues(helper, widget) { + return function (renderOptions) { + return function (query) { + var instantSearchInstance = renderOptions.instantSearchInstance, + searchResults = renderOptions.results; - resolvedState = resolvedState.addNumericRefinement(attribute, '<=', refinedOption.end); - } + if (query === '' && lastItemsFromMainSearch) { + // render with previous data from the helper. + renderFn(_objectSpread2(_objectSpread2({}, widget.getWidgetRenderState(_objectSpread2(_objectSpread2({}, renderOptions), {}, { + results: lastResultsFromMainSearch + }))), {}, { + instantSearchInstance: instantSearchInstance + }), false); + } else { + var tags = { + highlightPreTag: escapeFacetValues ? TAG_PLACEHOLDER.highlightPreTag : TAG_REPLACEMENT.highlightPreTag, + highlightPostTag: escapeFacetValues ? TAG_PLACEHOLDER.highlightPostTag : TAG_REPLACEMENT.highlightPostTag + }; + helper.searchForFacetValues(attribute, query, // We cap the `maxFacetHits` value to 100 because the Algolia API + // doesn't support a greater number. + // See https://www.algolia.com/doc/api-reference/api-parameters/maxFacetHits/ + Math.min(getLimit(), 100), tags).then(function (results) { + var facetValues = escapeFacetValues ? escapeFacets(results.facetHits) : results.facetHits; + var normalizedFacetValues = transformItems(facetValues.map(function (_ref3) { + var escapedValue = _ref3.escapedValue, + value = _ref3.value, + item = _objectWithoutProperties(_ref3, ["escapedValue", "value"]); - if (typeof resolvedState.page === 'number') { - resolvedState.page = 0; - } + return _objectSpread2(_objectSpread2({}, item), {}, { + value: escapedValue, + label: value + }); + }), { + results: searchResults + }); + renderFn(_objectSpread2(_objectSpread2({}, widget.getWidgetRenderState(_objectSpread2(_objectSpread2({}, renderOptions), {}, { + results: lastResultsFromMainSearch + }))), {}, { + items: normalizedFacetValues, + canToggleShowMore: false, + canRefine: true, + isFromSearch: true, + instantSearchInstance: instantSearchInstance + }), false); + }); + } + }; + }; + }; - return resolvedState; - } + return { + $$type: 'ais.refinementList', + init: function init(initOptions) { + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: initOptions.instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: renderOptions.instantSearchInstance + }), false); + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + refinementList: _objectSpread2(_objectSpread2({}, renderState.refinementList), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) + }); + }, + getWidgetRenderState: function getWidgetRenderState(renderOptions) { + var results = renderOptions.results, + state = renderOptions.state, + _createURL = renderOptions.createURL, + instantSearchInstance = renderOptions.instantSearchInstance, + helper = renderOptions.helper; + var items = []; + var facetValues = []; - function hasNumericRefinement(currentRefinements, operator, value) { - return currentRefinements[operator] !== undefined && currentRefinements[operator].includes(value); - } + if (!sendEvent || !triggerRefine || !searchForFacetValues) { + sendEvent = createSendEventForFacet({ + instantSearchInstance: instantSearchInstance, + helper: helper, + attribute: attribute, + widgetType: this.$$type + }); - var Paginator = /*#__PURE__*/function () { - function Paginator(params) { - _classCallCheck(this, Paginator); + triggerRefine = function triggerRefine(facetValue) { + sendEvent('click', facetValue); + helper.toggleFacetRefinement(attribute, facetValue).search(); + }; - _defineProperty(this, "currentPage", void 0); + searchForFacetValues = createSearchForFacetValues(helper, this); + } - _defineProperty(this, "total", void 0); + if (results) { + var values = results.getFacetValues(attribute, { + sortBy: sortBy, + facetOrdering: sortBy === DEFAULT_SORT$2 + }); + facetValues = values && Array.isArray(values) ? values : []; + items = transformItems(facetValues.slice(0, getLimit()).map(formatItems), { + results: results + }); + var maxValuesPerFacetConfig = state.maxValuesPerFacet; + var currentLimit = getLimit(); // If the limit is the max number of facet retrieved it is impossible to know + // if the facets are exhaustive. The only moment we are sure it is exhaustive + // is when it is strictly under the number requested unless we know that another + // widget has requested more values (maxValuesPerFacet > getLimit()). + // Because this is used for making the search of facets unable or not, it is important + // to be conservative here. - _defineProperty(this, "padding", void 0); + hasExhaustiveItems = maxValuesPerFacetConfig > currentLimit ? facetValues.length <= currentLimit : facetValues.length < currentLimit; + lastResultsFromMainSearch = results; + lastItemsFromMainSearch = items; - this.currentPage = params.currentPage; - this.total = params.total; - this.padding = params.padding; - } + if (renderOptions.results) { + toggleShowMore = createToggleShowMore(renderOptions, this); + } + } // Do not mistake searchForFacetValues and searchFacetValues which is the actual search + // function - _createClass(Paginator, [{ - key: "pages", - value: function pages() { - var total = this.total, - currentPage = this.currentPage, - padding = this.padding; - if (total === 0) return [0]; - var totalDisplayedPages = this.nbPagesDisplayed(padding, total); - if (totalDisplayedPages === total) { - return range({ - end: total - }); - } + var searchFacetValues = searchForFacetValues && searchForFacetValues(renderOptions); + var canShowLess = isShowingMore && lastItemsFromMainSearch.length > limit; + var canShowMore = showMore && !hasExhaustiveItems; + var canToggleShowMore = canShowLess || canShowMore; + return { + createURL: function createURL(facetValue) { + return _createURL(state.resetPage().toggleFacetRefinement(attribute, facetValue)); + }, + items: items, + refine: triggerRefine, + searchForItems: searchFacetValues, + isFromSearch: false, + canRefine: items.length > 0, + widgetParams: widgetParams, + isShowingMore: isShowingMore, + canToggleShowMore: canToggleShowMore, + toggleShowMore: cachedToggleShowMore, + sendEvent: sendEvent, + hasExhaustiveItems: hasExhaustiveItems + }; + }, + dispose: function dispose(_ref4) { + var state = _ref4.state; + unmountFn(); + var withoutMaxValuesPerFacet = state.setQueryParameter('maxValuesPerFacet', undefined); - var paddingLeft = this.calculatePaddingLeft(currentPage, padding, total, totalDisplayedPages); - var paddingRight = totalDisplayedPages - paddingLeft; - var first = currentPage - paddingLeft; - var last = currentPage + paddingRight; - return range({ - start: first, - end: last - }); - } - }, { - key: "nbPagesDisplayed", - value: function nbPagesDisplayed(padding, total) { - return Math.min(2 * padding + 1, total); - } - }, { - key: "calculatePaddingLeft", - value: function calculatePaddingLeft(current, padding, total, totalDisplayedPages) { - if (current <= padding) { - return current; - } + if (operator === 'and') { + return withoutMaxValuesPerFacet.removeFacet(attribute); + } - if (current >= total - padding) { - return totalDisplayedPages - (total - current); - } + return withoutMaxValuesPerFacet.removeDisjunctiveFacet(attribute); + }, + getWidgetUiState: function getWidgetUiState(uiState, _ref5) { + var searchParameters = _ref5.searchParameters; + var values = operator === 'or' ? searchParameters.getDisjunctiveRefinements(attribute) : searchParameters.getConjunctiveRefinements(attribute); + + if (!values.length) { + return uiState; + } - return padding; - } - }, { - key: "isLastPage", - value: function isLastPage() { - return this.currentPage === this.total - 1 || this.total === 0; - } - }, { - key: "isFirstPage", - value: function isFirstPage() { - return this.currentPage === 0; - } - }]); + return _objectSpread2(_objectSpread2({}, uiState), {}, { + refinementList: _objectSpread2(_objectSpread2({}, uiState.refinementList), {}, _defineProperty({}, attribute, values)) + }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { + var uiState = _ref6.uiState; + var isDisjunctive = operator === 'or'; + var values = uiState.refinementList && uiState.refinementList[attribute]; + var withoutRefinements = searchParameters.clearRefinements(attribute); + var withFacetConfiguration = isDisjunctive ? withoutRefinements.addDisjunctiveFacet(attribute) : withoutRefinements.addFacet(attribute); + var currentMaxValuesPerFacet = withFacetConfiguration.maxValuesPerFacet || 0; + var nextMaxValuesPerFacet = Math.max(currentMaxValuesPerFacet, showMore ? showMoreLimit : limit); + var withMaxValuesPerFacet = withFacetConfiguration.setQueryParameter('maxValuesPerFacet', nextMaxValuesPerFacet); - return Paginator; - }(); + if (!values) { + var key = isDisjunctive ? 'disjunctiveFacetsRefinements' : 'facetsRefinements'; + return withMaxValuesPerFacet.setQueryParameters(_defineProperty({}, key, _objectSpread2(_objectSpread2({}, withMaxValuesPerFacet[key]), {}, _defineProperty({}, attribute, [])))); + } - var withUsage$a = createDocumentationMessageGenerator({ - name: 'pagination', + return values.reduce(function (parameters, value) { + return isDisjunctive ? parameters.addDisjunctiveFacetRefinement(attribute, value) : parameters.addFacetRefinement(attribute, value); + }, withMaxValuesPerFacet); + } + }; + }; + }; + + var withUsage$d = createDocumentationMessageGenerator({ + name: 'search-box', connector: true }); + var defaultQueryHook = function defaultQueryHook(query, hook) { + return hook(query); + }; /** - * **Pagination** connector provides the logic to build a widget that will let the user - * choose the current page of the results. + * **SearchBox** connector provides the logic to build a widget that will let the user search for a query. * - * When using the pagination with Algolia, you should be aware that the engine won't provide you pages - * beyond the 1000th hits by default. You can find more information on the [Algolia documentation](https://www.algolia.com/doc/guides/searching/pagination/#pagination-limitations). + * The connector provides to the rendering: `refine()` to set the query. The behaviour of this function + * may be impacted by the `queryHook` widget parameter. */ - var connectPagination = function connectPagination(renderFn) { + + + var connectSearchBox = function connectSearchBox(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$a()); + checkRendering(renderFn, withUsage$d()); return function (widgetParams) { var _ref = widgetParams || {}, - totalPages = _ref.totalPages, - _ref$padding = _ref.padding, - padding = _ref$padding === void 0 ? 3 : _ref$padding; + _ref$queryHook = _ref.queryHook, + queryHook = _ref$queryHook === void 0 ? defaultQueryHook : _ref$queryHook; - var pager = new Paginator({ - currentPage: 0, - total: 0, - padding: padding - }); - var connectorState = {}; + var _refine; - function getMaxPage(_ref2) { - var nbPages = _ref2.nbPages; - return totalPages !== undefined ? Math.min(totalPages, nbPages) : nbPages; - } + var _clear; return { - $$type: 'ais.pagination', + $$type: 'ais.searchBox', init: function init(initOptions) { var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { @@ -12443,1060 +12459,1226 @@ instantSearchInstance: instantSearchInstance }), false); }, - dispose: function dispose(_ref3) { - var state = _ref3.state; + dispose: function dispose(_ref2) { + var state = _ref2.state; unmountFn(); - return state.setQueryParameter('page', undefined); + return state.setQueryParameter('query', undefined); }, - getWidgetUiState: function getWidgetUiState(uiState, _ref4) { - var searchParameters = _ref4.searchParameters; - var page = searchParameters.page || 0; - - if (!page) { - return uiState; - } - - return _objectSpread2(_objectSpread2({}, uiState), {}, { - page: page + 1 + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + searchBox: this.getWidgetRenderState(renderOptions) }); }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { - var uiState = _ref5.uiState; - var page = uiState.page ? uiState.page - 1 : 0; - return searchParameters.setQueryParameter('page', page); - }, - getWidgetRenderState: function getWidgetRenderState(_ref6) { - var results = _ref6.results, - helper = _ref6.helper, - state = _ref6.state, - createURL = _ref6.createURL; + getWidgetRenderState: function getWidgetRenderState(_ref3) { + var helper = _ref3.helper, + searchMetadata = _ref3.searchMetadata, + state = _ref3.state; - if (!connectorState.refine) { - connectorState.refine = function (page) { - helper.setPage(page); - helper.search(); + if (!_refine) { + _refine = function _refine(query) { + queryHook(query, function (q) { + return helper.setQuery(q).search(); + }); }; - } - if (!connectorState.createURL) { - connectorState.createURL = function (helperState) { - return function (page) { - return createURL(helperState.setPage(page)); - }; + _clear = function _clear() { + helper.setQuery('').search(); }; } - var page = state.page || 0; - var nbPages = getMaxPage(results || { - nbPages: 0 - }); - pager.currentPage = page; - pager.total = nbPages; return { - createURL: connectorState.createURL(state), - refine: connectorState.refine, - canRefine: nbPages > 1, - currentRefinement: page, - nbHits: (results === null || results === void 0 ? void 0 : results.nbHits) || 0, - nbPages: nbPages, - pages: results ? pager.pages() : [], - isFirstPage: pager.isFirstPage(), - isLastPage: pager.isLastPage(), - widgetParams: widgetParams + query: state.query || '', + refine: _refine, + clear: _clear, + widgetParams: widgetParams, + isSearchStalled: searchMetadata.isSearchStalled }; }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - pagination: this.getWidgetRenderState(renderOptions) + getWidgetUiState: function getWidgetUiState(uiState, _ref4) { + var searchParameters = _ref4.searchParameters; + var query = searchParameters.query || ''; + + if (query === '' || uiState && uiState.query === query) { + return uiState; + } + + return _objectSpread2(_objectSpread2({}, uiState), {}, { + query: query }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { + var uiState = _ref5.uiState; + return searchParameters.setQueryParameter('query', uiState.query || ''); } }; }; }; - var withUsage$b = createDocumentationMessageGenerator({ - name: 'range-input', - connector: true - }, { - name: 'range-slider', + var withUsage$e = createDocumentationMessageGenerator({ + name: 'sort-by', connector: true }); - var $$type$1 = 'ais.range'; - - function toPrecision(_ref) { - var min = _ref.min, - max = _ref.max, - precision = _ref.precision; - var pow = Math.pow(10, precision); - return { - min: min ? Math.floor(min * pow) / pow : min, - max: max ? Math.ceil(max * pow) / pow : max - }; - } /** - * **Range** connector provides the logic to create custom widget that will let - * the user refine results using a numeric range. - * - * This connectors provides a `refine()` function that accepts bounds. It will also provide - * information about the min and max bounds for the current result set. + * The **SortBy** connector provides the logic to build a custom widget that will display a + * list of indices. With Algolia, this is most commonly used for changing ranking strategy. This allows + * a user to change how the hits are being sorted. */ - - var connectRange = function connectRange(renderFn) { + var connectSortBy = function connectSortBy(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$b()); + checkRendering(renderFn, withUsage$e()); + var connectorState = {}; return function (widgetParams) { - var _ref2 = widgetParams || {}, - _ref2$attribute = _ref2.attribute, - attribute = _ref2$attribute === void 0 ? '' : _ref2$attribute, - minBound = _ref2.min, - maxBound = _ref2.max, - _ref2$precision = _ref2.precision, - precision = _ref2$precision === void 0 ? 0 : _ref2$precision; + var _ref = widgetParams || {}, + items = _ref.items, + _ref$transformItems = _ref.transformItems, + transformItems = _ref$transformItems === void 0 ? function (x) { + return x; + } : _ref$transformItems; - if (!attribute) { - throw new Error(withUsage$b('The `attribute` option is required.')); + if (!Array.isArray(items)) { + throw new Error(withUsage$e('The `items` option expects an array of objects.')); } - if (isFiniteNumber(minBound) && isFiniteNumber(maxBound) && minBound > maxBound) { - throw new Error(withUsage$b("The `max` option can't be lower than `min`.")); - } + return { + $$type: 'ais.sortBy', + init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; + var widgetRenderState = this.getWidgetRenderState(initOptions); + var currentIndex = widgetRenderState.currentRefinement; + var isCurrentIndexInItems = find$1(items, function (item) { + return item.value === currentIndex; + }); + _warning(isCurrentIndexInItems !== undefined, "The index named \"".concat(currentIndex, "\" is not listed in the `items` of `sortBy`.")) ; + renderFn(_objectSpread2(_objectSpread2({}, widgetRenderState), {}, { + instantSearchInstance: instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), false); + }, + dispose: function dispose(_ref2) { + var state = _ref2.state; + unmountFn(); + return connectorState.initialIndex ? state.setIndex(connectorState.initialIndex) : state; + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + sortBy: this.getWidgetRenderState(renderOptions) + }); + }, + getWidgetRenderState: function getWidgetRenderState(_ref3) { + var results = _ref3.results, + helper = _ref3.helper, + state = _ref3.state, + parent = _ref3.parent; - var formatToNumber = function formatToNumber(v) { - return Number(Number(v).toFixed(precision)); - }; + if (!connectorState.initialIndex && parent) { + connectorState.initialIndex = parent.getIndexName(); + } - var rangeFormatter = { - from: function from(v) { - return v.toLocaleString(); + if (!connectorState.setIndex) { + connectorState.setIndex = function (indexName) { + helper.setIndex(indexName).search(); + }; + } + + var hasNoResults = results ? results.nbHits === 0 : true; + return { + currentRefinement: state.index, + options: transformItems(items, { + results: results + }), + refine: connectorState.setIndex, + hasNoResults: hasNoResults, + canRefine: !hasNoResults && items.length > 0, + widgetParams: widgetParams + }; }, - to: function to(v) { - return formatToNumber(v).toLocaleString(); + getWidgetUiState: function getWidgetUiState(uiState, _ref4) { + var searchParameters = _ref4.searchParameters; + var currentIndex = searchParameters.index; + return _objectSpread2(_objectSpread2({}, uiState), {}, { + sortBy: currentIndex !== connectorState.initialIndex ? currentIndex : undefined + }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { + var uiState = _ref5.uiState; + return searchParameters.setQueryParameter('index', uiState.sortBy || connectorState.initialIndex || searchParameters.index); } - }; // eslint-disable-next-line complexity - - var getRefinedState = function getRefinedState(helper, currentRange, nextMin, nextMax) { - var resolvedState = helper.state; - var currentRangeMin = currentRange.min, - currentRangeMax = currentRange.max; + }; + }; + }; - var _ref3 = resolvedState.getNumericRefinement(attribute, '>=') || [], - _ref4 = _slicedToArray(_ref3, 1), - min = _ref4[0]; + var withUsage$f = createDocumentationMessageGenerator({ + name: 'rating-menu', + connector: true + }); + var $$type$2 = 'ais.ratingMenu'; + var MAX_VALUES_PER_FACET_API_LIMIT = 1000; + var STEP = 1; - var _ref5 = resolvedState.getNumericRefinement(attribute, '<=') || [], - _ref6 = _slicedToArray(_ref5, 1), - max = _ref6[0]; + var createSendEvent$1 = function createSendEvent(_ref) { + var instantSearchInstance = _ref.instantSearchInstance, + helper = _ref.helper, + getRefinedStar = _ref.getRefinedStar, + attribute = _ref.attribute; + return function () { + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; + } - var isResetMin = nextMin === undefined || nextMin === ''; - var isResetMax = nextMax === undefined || nextMax === ''; + if (args.length === 1) { + instantSearchInstance.sendEventToInsights(args[0]); + return; + } - var _toPrecision = toPrecision({ - min: !isResetMin ? parseFloat(nextMin) : undefined, - max: !isResetMax ? parseFloat(nextMax) : undefined, - precision: precision - }), - nextMinAsNumber = _toPrecision.min, - nextMaxAsNumber = _toPrecision.max; + var eventType = args[0], + facetValue = args[1], + _args$ = args[2], + eventName = _args$ === void 0 ? 'Filter Applied' : _args$; - var newNextMin; + if (eventType !== 'click') { + return; + } - if (!isFiniteNumber(minBound) && currentRangeMin === nextMinAsNumber) { - newNextMin = undefined; - } else if (isFiniteNumber(minBound) && isResetMin) { - newNextMin = minBound; - } else { - newNextMin = nextMinAsNumber; - } + var isRefined = getRefinedStar() === Number(facetValue); - var newNextMax; + if (!isRefined) { + instantSearchInstance.sendEventToInsights({ + insightsMethod: 'clickedFilters', + widgetType: $$type$2, + eventType: eventType, + payload: { + eventName: eventName, + index: helper.getIndex(), + filters: ["".concat(attribute, ">=").concat(facetValue)] + }, + attribute: attribute + }); + } + }; + }; - if (!isFiniteNumber(maxBound) && currentRangeMax === nextMaxAsNumber) { - newNextMax = undefined; - } else if (isFiniteNumber(maxBound) && isResetMax) { - newNextMax = maxBound; - } else { - newNextMax = nextMaxAsNumber; - } + /** + * **StarRating** connector provides the logic to build a custom widget that will let + * the user refine search results based on ratings. + * + * The connector provides to the rendering: `refine()` to select a value and + * `items` that are the values that can be selected. `refine` should be used + * with `items.value`. + */ + var connectRatingMenu = function connectRatingMenu(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$f()); + return function (widgetParams) { + var _ref2 = widgetParams || {}, + attribute = _ref2.attribute, + _ref2$max = _ref2.max, + max = _ref2$max === void 0 ? 5 : _ref2$max; - var isResetNewNextMin = newNextMin === undefined; - var isGreaterThanCurrentRange = isFiniteNumber(currentRangeMin) && currentRangeMin <= newNextMin; - var isMinValid = isResetNewNextMin || isFiniteNumber(newNextMin) && (!isFiniteNumber(currentRangeMin) || isGreaterThanCurrentRange); - var isResetNewNextMax = newNextMax === undefined; - var isLowerThanRange = isFiniteNumber(newNextMax) && currentRangeMax >= newNextMax; - var isMaxValid = isResetNewNextMax || isFiniteNumber(newNextMax) && (!isFiniteNumber(currentRangeMax) || isLowerThanRange); - var hasMinChange = min !== newNextMin; - var hasMaxChange = max !== newNextMax; + var sendEvent; - if ((hasMinChange || hasMaxChange) && isMinValid && isMaxValid) { - resolvedState = resolvedState.removeNumericRefinement(attribute); + if (!attribute) { + throw new Error(withUsage$f('The `attribute` option is required.')); + } - if (isFiniteNumber(newNextMin)) { - resolvedState = resolvedState.addNumericRefinement(attribute, '>=', newNextMin); - } + var _getRefinedStar = function getRefinedStar(state) { + var _values$; - if (isFiniteNumber(newNextMax)) { - resolvedState = resolvedState.addNumericRefinement(attribute, '<=', newNextMax); - } + var values = state.getNumericRefinements(attribute); - return resolvedState.resetPage(); + if (!((_values$ = values['>=']) !== null && _values$ !== void 0 && _values$.length)) { + return undefined; } - return null; - }; - - var sendEventWithRefinedState = function sendEventWithRefinedState(refinedState, instantSearchInstance, helper) { - var eventName = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'Filter Applied'; - var filters = convertNumericRefinementsToFilters(refinedState, attribute); - - if (filters && filters.length > 0) { - instantSearchInstance.sendEventToInsights({ - insightsMethod: 'clickedFilters', - widgetType: $$type$1, - eventType: 'click', - payload: { - eventName: eventName, - index: helper.getIndex(), - filters: filters - }, - attribute: attribute - }); - } + return values['>='][0]; }; - var createSendEvent = function createSendEvent(instantSearchInstance, helper, currentRange) { - return function () { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } - - if (args.length === 1) { - instantSearchInstance.sendEventToInsights(args[0]); - return; - } - - var eventType = args[0], - facetValue = args[1], - eventName = args[2]; - - if (eventType !== 'click') { - return; - } - - var _facetValue = _slicedToArray(facetValue, 2), - nextMin = _facetValue[0], - nextMax = _facetValue[1]; + var getFacetsMaxDecimalPlaces = function getFacetsMaxDecimalPlaces(facetResults) { + var maxDecimalPlaces = 0; + facetResults.forEach(function (facetResult) { + var _facetResult$name$spl = facetResult.name.split('.'), + _facetResult$name$spl2 = _slicedToArray(_facetResult$name$spl, 2), + _facetResult$name$spl3 = _facetResult$name$spl2[1], + decimal = _facetResult$name$spl3 === void 0 ? '' : _facetResult$name$spl3; - var refinedState = getRefinedState(helper, currentRange, nextMin, nextMax); - sendEventWithRefinedState(refinedState, instantSearchInstance, helper, eventName); - }; + maxDecimalPlaces = Math.max(maxDecimalPlaces, decimal.length); + }); + return maxDecimalPlaces; }; - function _getCurrentRange(stats) { - var min; + var getFacetValuesWarningMessage = function getFacetValuesWarningMessage(_ref3) { + var maxDecimalPlaces = _ref3.maxDecimalPlaces, + maxFacets = _ref3.maxFacets, + maxValuesPerFacet = _ref3.maxValuesPerFacet; + var maxDecimalPlacesInRange = Math.max(0, Math.floor(Math.log10(MAX_VALUES_PER_FACET_API_LIMIT / max))); + var maxFacetsInRange = Math.min(MAX_VALUES_PER_FACET_API_LIMIT, Math.pow(10, maxDecimalPlacesInRange) * max); + var solutions = []; - if (isFiniteNumber(minBound)) { - min = minBound; - } else if (isFiniteNumber(stats.min)) { - min = stats.min; - } else { - min = 0; + if (maxFacets > MAX_VALUES_PER_FACET_API_LIMIT) { + solutions.push("- Update your records to lower the precision of the values in the \"".concat(attribute, "\" attribute (for example: ").concat(5.123456789.toPrecision(maxDecimalPlaces + 1), " to ").concat(5.123456789.toPrecision(maxDecimalPlacesInRange + 1), ")")); } - var max; - - if (isFiniteNumber(maxBound)) { - max = maxBound; - } else if (isFiniteNumber(stats.max)) { - max = stats.max; - } else { - max = 0; + if (maxValuesPerFacet < maxFacetsInRange) { + solutions.push("- Increase the maximum number of facet values to ".concat(maxFacetsInRange, " using the \"configure\" widget ").concat(createDocumentationLink({ + name: 'configure' + }), " and the \"maxValuesPerFacet\" parameter https://www.algolia.com/doc/api-reference/api-parameters/maxValuesPerFacet/")); } - return toPrecision({ - min: min, - max: max, - precision: precision - }); - } - - function _getCurrentRefinement(helper) { - var _ref7 = helper.getNumericRefinement(attribute, '>=') || [], - _ref8 = _slicedToArray(_ref7, 1), - minValue = _ref8[0]; - - var _ref9 = helper.getNumericRefinement(attribute, '<=') || [], - _ref10 = _slicedToArray(_ref9, 1), - maxValue = _ref10[0]; - - var min = isFiniteNumber(minValue) ? minValue : -Infinity; - var max = isFiniteNumber(maxValue) ? maxValue : Infinity; - return [min, max]; - } + return "The ".concat(attribute, " attribute can have ").concat(maxFacets, " different values (0 to ").concat(max, " with a maximum of ").concat(maxDecimalPlaces, " decimals = ").concat(maxFacets, ") but you retrieved only ").concat(maxValuesPerFacet, " facet values. Therefore the number of results that match the refinements can be incorrect.\n ").concat(solutions.length ? "To resolve this problem you can:\n".concat(solutions.join('\n')) : ""); + }; - function _refine(instantSearchInstance, helper, currentRange) { - return function () { - var _ref11 = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [undefined, undefined], - _ref12 = _slicedToArray(_ref11, 2), - nextMin = _ref12[0], - nextMax = _ref12[1]; + function getRefinedState(state, facetValue) { + var isRefined = _getRefinedStar(state) === Number(facetValue); + var emptyState = state.resetPage().removeNumericRefinement(attribute); - var refinedState = getRefinedState(helper, currentRange, nextMin, nextMax); + if (!isRefined) { + return emptyState.addNumericRefinement(attribute, '<=', max).addNumericRefinement(attribute, '>=', Number(facetValue)); + } - if (refinedState) { - sendEventWithRefinedState(refinedState, instantSearchInstance, helper); - helper.setState(refinedState).search(); - } - }; + return emptyState; } + var toggleRefinement = function toggleRefinement(helper, facetValue) { + sendEvent('click', facetValue); + helper.setState(getRefinedState(helper.state, facetValue)).search(); + }; + + var connectorState = { + toggleRefinementFactory: function toggleRefinementFactory(helper) { + return toggleRefinement.bind(null, helper); + }, + createURLFactory: function createURLFactory(_ref4) { + var state = _ref4.state, + createURL = _ref4.createURL; + return function (value) { + return createURL(getRefinedState(state, value)); + }; + } + }; return { - $$type: $$type$1, + $$type: $$type$2, init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: initOptions.instantSearchInstance + instantSearchInstance: instantSearchInstance }), true); }, render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: renderOptions.instantSearchInstance + instantSearchInstance: instantSearchInstance }), false); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - range: _objectSpread2(_objectSpread2({}, renderState.range), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) + ratingMenu: _objectSpread2(_objectSpread2({}, renderState.ratingMenu), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref13) { - var results = _ref13.results, - helper = _ref13.helper, - instantSearchInstance = _ref13.instantSearchInstance; - var facetsFromResults = results && results.disjunctiveFacets || []; - var facet = find$1(facetsFromResults, function (facetResult) { - return facetResult.name === attribute; - }); - var stats = facet && facet.stats || { - min: undefined, - max: undefined - }; + getWidgetRenderState: function getWidgetRenderState(_ref5) { + var helper = _ref5.helper, + results = _ref5.results, + state = _ref5.state, + instantSearchInstance = _ref5.instantSearchInstance, + createURL = _ref5.createURL; + var facetValues = []; - var currentRange = _getCurrentRange(stats); + if (!sendEvent) { + sendEvent = createSendEvent$1({ + instantSearchInstance: instantSearchInstance, + helper: helper, + getRefinedStar: function getRefinedStar() { + return _getRefinedStar(helper.state); + }, + attribute: attribute + }); + } - var start = _getCurrentRefinement(helper); + var refinementIsApplied = false; + var totalCount = 0; + var facetResults = results === null || results === void 0 ? void 0 : results.getFacetValues(attribute, {}); - var refine; + if (results && facetResults) { + var maxValuesPerFacet = facetResults.length; + var maxDecimalPlaces = getFacetsMaxDecimalPlaces(facetResults); + var maxFacets = Math.pow(10, maxDecimalPlaces) * max; + _warning(maxFacets <= maxValuesPerFacet || Boolean(results.__isArtificial), getFacetValuesWarningMessage({ + maxDecimalPlaces: maxDecimalPlaces, + maxFacets: maxFacets, + maxValuesPerFacet: maxValuesPerFacet + })) ; - if (!results) { - // On first render pass an empty range - // to be able to bypass the validation - // related to it - refine = _refine(instantSearchInstance, helper, { - min: undefined, - max: undefined - }); - } else { - refine = _refine(instantSearchInstance, helper, currentRange); + var refinedStar = _getRefinedStar(state); + + var _loop = function _loop(star) { + var isRefined = refinedStar === star; + refinementIsApplied = refinementIsApplied || isRefined; + var count = facetResults.filter(function (f) { + return Number(f.name) >= star && Number(f.name) <= max; + }).map(function (f) { + return f.count; + }).reduce(function (sum, current) { + return sum + current; + }, 0); + totalCount += count; + + if (refinedStar && !isRefined && count === 0) { + // skip count==0 when at least 1 refinement is enabled + // eslint-disable-next-line no-continue + return "continue"; + } + + var stars = _toConsumableArray(new Array(Math.floor(max / STEP))).map(function (_v, i) { + return i * STEP < star; + }); + + facetValues.push({ + stars: stars, + name: String(star), + label: String(star), + value: String(star), + count: count, + isRefined: isRefined + }); + }; + + for (var star = STEP; star < max; star += STEP) { + var _ret = _loop(star); + + if (_ret === "continue") continue; + } } + facetValues = facetValues.reverse(); + var hasNoResults = results ? results.nbHits === 0 : true; return { - refine: refine, - canRefine: currentRange.min !== currentRange.max, - format: rangeFormatter, - range: currentRange, - sendEvent: createSendEvent(instantSearchInstance, helper, currentRange), - widgetParams: _objectSpread2(_objectSpread2({}, widgetParams), {}, { - precision: precision + items: facetValues, + hasNoResults: hasNoResults, + canRefine: (!hasNoResults || refinementIsApplied) && totalCount > 0, + refine: connectorState.toggleRefinementFactory(helper), + sendEvent: sendEvent, + createURL: connectorState.createURLFactory({ + state: state, + createURL: createURL }), - start: start + widgetParams: widgetParams }; }, - dispose: function dispose(_ref14) { - var state = _ref14.state; + dispose: function dispose(_ref6) { + var state = _ref6.state; unmountFn(); - return state.removeDisjunctiveFacet(attribute).removeNumericRefinement(attribute); + return state.removeNumericRefinement(attribute); }, - getWidgetUiState: function getWidgetUiState(uiState, _ref15) { - var searchParameters = _ref15.searchParameters; + getWidgetUiState: function getWidgetUiState(uiState, _ref7) { + var searchParameters = _ref7.searchParameters; - var _searchParameters$get = searchParameters.getNumericRefinements(attribute), - _searchParameters$get2 = _searchParameters$get['>='], - min = _searchParameters$get2 === void 0 ? [] : _searchParameters$get2, - _searchParameters$get3 = _searchParameters$get['<='], - max = _searchParameters$get3 === void 0 ? [] : _searchParameters$get3; + var value = _getRefinedStar(searchParameters); - if (min.length === 0 && max.length === 0) { + if (typeof value !== 'number') { return uiState; } return _objectSpread2(_objectSpread2({}, uiState), {}, { - range: _objectSpread2(_objectSpread2({}, uiState.range), {}, _defineProperty({}, attribute, "".concat(min, ":").concat(max))) + ratingMenu: _objectSpread2(_objectSpread2({}, uiState.ratingMenu), {}, _defineProperty({}, attribute, value)) }); }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref16) { - var uiState = _ref16.uiState; - var widgetSearchParameters = searchParameters.addDisjunctiveFacet(attribute).setQueryParameters({ - numericRefinements: _objectSpread2(_objectSpread2({}, searchParameters.numericRefinements), {}, _defineProperty({}, attribute, {})) - }); - - if (isFiniteNumber(minBound)) { - widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '>=', minBound); - } + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref8) { + var uiState = _ref8.uiState; + var value = uiState.ratingMenu && uiState.ratingMenu[attribute]; + var withoutRefinements = searchParameters.clearRefinements(attribute); + var withDisjunctiveFacet = withoutRefinements.addDisjunctiveFacet(attribute); - if (isFiniteNumber(maxBound)) { - widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '<=', maxBound); + if (!value) { + return withDisjunctiveFacet.setQueryParameters({ + numericRefinements: _objectSpread2(_objectSpread2({}, withDisjunctiveFacet.numericRefinements), {}, _defineProperty({}, attribute, {})) + }); } - var value = uiState.range && uiState.range[attribute]; - - if (!value || value.indexOf(':') === -1) { - return widgetSearchParameters; - } + return withDisjunctiveFacet.addNumericRefinement(attribute, '<=', max).addNumericRefinement(attribute, '>=', value); + } + }; + }; + }; - var _value$split$map = value.split(':').map(parseFloat), - _value$split$map2 = _slicedToArray(_value$split$map, 2), - lowerBound = _value$split$map2[0], - upperBound = _value$split$map2[1]; + var withUsage$g = createDocumentationMessageGenerator({ + name: 'stats', + connector: true + }); + /** + * **Stats** connector provides the logic to build a custom widget that will displays + * search statistics (hits number and processing time). + */ - if (isFiniteNumber(lowerBound) && (!isFiniteNumber(minBound) || minBound < lowerBound)) { - widgetSearchParameters = widgetSearchParameters.removeNumericRefinement(attribute, '>='); - widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '>=', lowerBound); - } + var connectStats = function connectStats(renderFn) { + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(renderFn, withUsage$g()); + return function (widgetParams) { + return { + $$type: 'ais.stats', + init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), false); + }, + dispose: function dispose() { + unmountFn(); + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + stats: this.getWidgetRenderState(renderOptions) + }); + }, + getWidgetRenderState: function getWidgetRenderState(_ref) { + var results = _ref.results, + state = _ref.state; - if (isFiniteNumber(upperBound) && (!isFiniteNumber(maxBound) || upperBound < maxBound)) { - widgetSearchParameters = widgetSearchParameters.removeNumericRefinement(attribute, '<='); - widgetSearchParameters = widgetSearchParameters.addNumericRefinement(attribute, '<=', upperBound); + if (!results) { + return { + hitsPerPage: state.hitsPerPage, + nbHits: 0, + nbSortedHits: undefined, + areHitsSorted: false, + nbPages: 0, + page: state.page || 0, + processingTimeMS: -1, + query: state.query || '', + widgetParams: widgetParams + }; } - return widgetSearchParameters; + return { + hitsPerPage: results.hitsPerPage, + nbHits: results.nbHits, + nbSortedHits: results.nbSortedHits, + areHitsSorted: typeof results.appliedRelevancyStrictness !== 'undefined' && results.appliedRelevancyStrictness > 0 && results.nbSortedHits !== results.nbHits, + nbPages: results.nbPages, + page: results.page, + processingTimeMS: results.processingTimeMS, + query: results.query, + widgetParams: widgetParams + }; } }; }; }; - var withUsage$c = createDocumentationMessageGenerator({ - name: 'refinement-list', + var withUsage$h = createDocumentationMessageGenerator({ + name: 'toggle-refinement', connector: true }); - var DEFAULT_SORT$2 = ['isRefined', 'count:desc', 'name:asc']; + var $$type$3 = 'ais.toggleRefinement'; + + var createSendEvent$2 = function createSendEvent(_ref) { + var instantSearchInstance = _ref.instantSearchInstance, + helper = _ref.helper, + attribute = _ref.attribute, + on = _ref.on; + + var sendEventForToggle = function sendEventForToggle() { + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; + } + + if (args.length === 1) { + instantSearchInstance.sendEventToInsights(args[0]); + return; + } + + var eventType = args[0], + isRefined = args[1], + _args$ = args[2], + eventName = _args$ === void 0 ? 'Filter Applied' : _args$; + + if (eventType !== 'click' || on === undefined) { + return; + } // only send an event when the refinement gets applied, + // not when it gets removed + + + if (!isRefined) { + instantSearchInstance.sendEventToInsights({ + insightsMethod: 'clickedFilters', + widgetType: $$type$3, + eventType: eventType, + payload: { + eventName: eventName, + index: helper.getIndex(), + filters: on.map(function (value) { + return "".concat(attribute, ":").concat(value); + }) + }, + attribute: attribute + }); + } + }; + + return sendEventForToggle; + }; /** - * **RefinementList** connector provides the logic to build a custom widget that - * will let the user filter the results based on the values of a specific facet. - * - * **Requirement:** the attribute passed as `attribute` must be present in - * attributesForFaceting of the searched index. + * **Toggle** connector provides the logic to build a custom widget that will provide + * an on/off filtering feature based on an attribute value or values. * - * This connector provides: - * - a `refine()` function to select an item. - * - a `toggleShowMore()` function to display more or less items - * - a `searchForItems()` function to search within the items. + * Two modes are implemented in the custom widget: + * - with or without the value filtered + * - switch between two values. */ - var connectRefinementList = function connectRefinementList(renderFn) { + var connectToggleRefinement = function connectToggleRefinement(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$c()); + checkRendering(renderFn, withUsage$h()); return function (widgetParams) { - var _ref = widgetParams || {}, - attribute = _ref.attribute, - _ref$operator = _ref.operator, - operator = _ref$operator === void 0 ? 'or' : _ref$operator, - _ref$limit = _ref.limit, - limit = _ref$limit === void 0 ? 10 : _ref$limit, - _ref$showMore = _ref.showMore, - showMore = _ref$showMore === void 0 ? false : _ref$showMore, - _ref$showMoreLimit = _ref.showMoreLimit, - showMoreLimit = _ref$showMoreLimit === void 0 ? 20 : _ref$showMoreLimit, - _ref$sortBy = _ref.sortBy, - sortBy = _ref$sortBy === void 0 ? DEFAULT_SORT$2 : _ref$sortBy, - _ref$escapeFacetValue = _ref.escapeFacetValues, - escapeFacetValues = _ref$escapeFacetValue === void 0 ? true : _ref$escapeFacetValue, - _ref$transformItems = _ref.transformItems, - transformItems = _ref$transformItems === void 0 ? function (items) { - return items; - } : _ref$transformItems; + var _ref2 = widgetParams || {}, + attribute = _ref2.attribute, + _ref2$on = _ref2.on, + userOn = _ref2$on === void 0 ? true : _ref2$on, + userOff = _ref2.off; if (!attribute) { - throw new Error(withUsage$c('The `attribute` option is required.')); + throw new Error(withUsage$h('The `attribute` option is required.')); } - if (!/^(and|or)$/.test(operator)) { - throw new Error(withUsage$c("The `operator` must one of: `\"and\"`, `\"or\"` (got \"".concat(operator, "\")."))); - } + var hasAnOffValue = userOff !== undefined; // even though facet values can be numbers and boolean, + // the helper methods only accept string in the type - if (showMore === true && showMoreLimit <= limit) { - throw new Error(withUsage$c('`showMoreLimit` should be greater than `limit`.')); - } + var on = toArray(userOn).map(escapeFacetValue$4); + var off = hasAnOffValue ? toArray(userOff).map(escapeFacetValue$4) : undefined; + var sendEvent; - var formatItems = function formatItems(_ref2) { - var label = _ref2.name, - value = _ref2.escapedValue, - item = _objectWithoutProperties(_ref2, ["name", "escapedValue"]); + var toggleRefinementFactory = function toggleRefinementFactory(helper) { + return function () { + var _ref3 = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : { + isRefined: false + }, + isRefined = _ref3.isRefined; - return _objectSpread2(_objectSpread2({}, item), {}, { - value: value, - label: label, - highlighted: label - }); - }; + if (!isRefined) { + sendEvent('click', isRefined); - var lastResultsFromMainSearch; - var lastItemsFromMainSearch = []; - var hasExhaustiveItems = true; - var triggerRefine; - var sendEvent; - var isShowingMore = false; // Provide the same function to the `renderFn` so that way the user - // has to only bind it once when `isFirstRendering` for instance + if (hasAnOffValue) { + off.forEach(function (v) { + return helper.removeDisjunctiveFacetRefinement(attribute, v); + }); + } - var toggleShowMore = function toggleShowMore() {}; + on.forEach(function (v) { + return helper.addDisjunctiveFacetRefinement(attribute, v); + }); + } else { + on.forEach(function (v) { + return helper.removeDisjunctiveFacetRefinement(attribute, v); + }); - function cachedToggleShowMore() { - toggleShowMore(); - } + if (hasAnOffValue) { + off.forEach(function (v) { + return helper.addDisjunctiveFacetRefinement(attribute, v); + }); + } + } - function createToggleShowMore(renderOptions, widget) { - return function () { - isShowingMore = !isShowingMore; - widget.render(renderOptions); + helper.search(); }; - } - - function getLimit() { - return isShowingMore ? showMoreLimit : limit; - } - - var searchForFacetValues = function searchForFacetValues() { - return function () {}; }; - var createSearchForFacetValues = function createSearchForFacetValues(helper, widget) { - return function (renderOptions) { - return function (query) { - var instantSearchInstance = renderOptions.instantSearchInstance, - searchResults = renderOptions.results; + var connectorState = { + createURLFactory: function createURLFactory(isRefined, _ref4) { + var state = _ref4.state, + createURL = _ref4.createURL; + return function () { + state = state.resetPage(); + var valuesToRemove = isRefined ? on : off; - if (query === '' && lastItemsFromMainSearch) { - // render with previous data from the helper. - renderFn(_objectSpread2(_objectSpread2({}, widget.getWidgetRenderState(_objectSpread2(_objectSpread2({}, renderOptions), {}, { - results: lastResultsFromMainSearch - }))), {}, { - instantSearchInstance: instantSearchInstance - }), false); - } else { - var tags = { - highlightPreTag: escapeFacetValues ? TAG_PLACEHOLDER.highlightPreTag : TAG_REPLACEMENT.highlightPreTag, - highlightPostTag: escapeFacetValues ? TAG_PLACEHOLDER.highlightPostTag : TAG_REPLACEMENT.highlightPostTag - }; - helper.searchForFacetValues(attribute, query, // We cap the `maxFacetHits` value to 100 because the Algolia API - // doesn't support a greater number. - // See https://www.algolia.com/doc/api-reference/api-parameters/maxFacetHits/ - Math.min(getLimit(), 100), tags).then(function (results) { - var facetValues = escapeFacetValues ? escapeFacets(results.facetHits) : results.facetHits; - var normalizedFacetValues = transformItems(facetValues.map(function (_ref3) { - var escapedValue = _ref3.escapedValue, - value = _ref3.value, - item = _objectWithoutProperties(_ref3, ["escapedValue", "value"]); + if (valuesToRemove) { + valuesToRemove.forEach(function (v) { + state = state.removeDisjunctiveFacetRefinement(attribute, v); + }); + } - return _objectSpread2(_objectSpread2({}, item), {}, { - value: escapedValue, - label: value - }); - }), { - results: searchResults - }); - renderFn(_objectSpread2(_objectSpread2({}, widget.getWidgetRenderState(_objectSpread2(_objectSpread2({}, renderOptions), {}, { - results: lastResultsFromMainSearch - }))), {}, { - items: normalizedFacetValues, - canToggleShowMore: false, - canRefine: true, - isFromSearch: true, - instantSearchInstance: instantSearchInstance - }), false); + var valuesToAdd = isRefined ? off : on; + + if (valuesToAdd) { + valuesToAdd.forEach(function (v) { + state = state.addDisjunctiveFacetRefinement(attribute, v); }); } + + return createURL(state); }; - }; + } }; - return { - $$type: 'ais.refinementList', + $$type: $$type$3, init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: initOptions.instantSearchInstance + instantSearchInstance: instantSearchInstance }), true); }, render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: renderOptions.instantSearchInstance + instantSearchInstance: instantSearchInstance }), false); }, + dispose: function dispose(_ref5) { + var state = _ref5.state; + unmountFn(); + return state.removeDisjunctiveFacet(attribute); + }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - refinementList: _objectSpread2(_objectSpread2({}, renderState.refinementList), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) + toggleRefinement: _objectSpread2(_objectSpread2({}, renderState.toggleRefinement), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) }); }, - getWidgetRenderState: function getWidgetRenderState(renderOptions) { - var results = renderOptions.results, - state = renderOptions.state, - _createURL = renderOptions.createURL, - instantSearchInstance = renderOptions.instantSearchInstance, - helper = renderOptions.helper; - var items = []; - var facetValues = []; + getWidgetRenderState: function getWidgetRenderState(_ref6) { + var state = _ref6.state, + helper = _ref6.helper, + results = _ref6.results, + createURL = _ref6.createURL, + instantSearchInstance = _ref6.instantSearchInstance; + var isRefined = results ? on.every(function (v) { + return state.isDisjunctiveFacetRefined(attribute, v); + }) : on.every(function (v) { + return state.isDisjunctiveFacetRefined(attribute, v); + }); + var onFacetValue = { + isRefined: isRefined, + count: 0 + }; + var offFacetValue = { + isRefined: hasAnOffValue && !isRefined, + count: 0 + }; - if (!sendEvent || !triggerRefine || !searchForFacetValues) { - sendEvent = createSendEventForFacet({ - instantSearchInstance: instantSearchInstance, - helper: helper, - attribute: attribute, - widgetType: this.$$type + if (results) { + var offValue = toArray(off || false); + var allFacetValues = results.getFacetValues(attribute, {}) || []; + var onData = on.map(function (v) { + return find$1(allFacetValues, function (_ref7) { + var escapedValue = _ref7.escapedValue; + return escapedValue === escapeFacetValue$4(String(v)); + }); + }).filter(function (v) { + return v !== undefined; }); - - triggerRefine = function triggerRefine(facetValue) { - sendEvent('click', facetValue); - helper.toggleFacetRefinement(attribute, facetValue).search(); + var offData = hasAnOffValue ? offValue.map(function (v) { + return find$1(allFacetValues, function (_ref8) { + var escapedValue = _ref8.escapedValue; + return escapedValue === escapeFacetValue$4(String(v)); + }); + }).filter(function (v) { + return v !== undefined; + }) : []; + onFacetValue = { + isRefined: onData.length ? onData.every(function (v) { + return v.isRefined; + }) : false, + count: onData.reduce(function (acc, v) { + return acc + v.count; + }, 0) || null + }; + offFacetValue = { + isRefined: offData.length ? offData.every(function (v) { + return v.isRefined; + }) : false, + count: offData.reduce(function (acc, v) { + return acc + v.count; + }, 0) || allFacetValues.reduce(function (total, _ref9) { + var count = _ref9.count; + return total + count; + }, 0) }; - - searchForFacetValues = createSearchForFacetValues(helper, this); } - if (results) { - var values = results.getFacetValues(attribute, { - sortBy: sortBy, - facetOrdering: sortBy === DEFAULT_SORT$2 - }); - facetValues = values && Array.isArray(values) ? values : []; - items = transformItems(facetValues.slice(0, getLimit()).map(formatItems), { - results: results + if (!sendEvent) { + sendEvent = createSendEvent$2({ + instantSearchInstance: instantSearchInstance, + attribute: attribute, + on: on, + helper: helper }); - var maxValuesPerFacetConfig = state.maxValuesPerFacet; - var currentLimit = getLimit(); // If the limit is the max number of facet retrieved it is impossible to know - // if the facets are exhaustive. The only moment we are sure it is exhaustive - // is when it is strictly under the number requested unless we know that another - // widget has requested more values (maxValuesPerFacet > getLimit()). - // Because this is used for making the search of facets unable or not, it is important - // to be conservative here. - - hasExhaustiveItems = maxValuesPerFacetConfig > currentLimit ? facetValues.length <= currentLimit : facetValues.length < currentLimit; - lastResultsFromMainSearch = results; - lastItemsFromMainSearch = items; - - if (renderOptions.results) { - toggleShowMore = createToggleShowMore(renderOptions, this); - } - } // Do not mistake searchForFacetValues and searchFacetValues which is the actual search - // function - + } - var searchFacetValues = searchForFacetValues && searchForFacetValues(renderOptions); - var canShowLess = isShowingMore && lastItemsFromMainSearch.length > limit; - var canShowMore = showMore && !hasExhaustiveItems; - var canToggleShowMore = canShowLess || canShowMore; + var nextRefinement = isRefined ? offFacetValue : onFacetValue; return { - createURL: function createURL(facetValue) { - return _createURL(state.resetPage().toggleFacetRefinement(attribute, facetValue)); + value: { + name: attribute, + isRefined: isRefined, + count: results ? nextRefinement.count : null, + onFacetValue: onFacetValue, + offFacetValue: offFacetValue }, - items: items, - refine: triggerRefine, - searchForItems: searchFacetValues, - isFromSearch: false, - canRefine: items.length > 0, - widgetParams: widgetParams, - isShowingMore: isShowingMore, - canToggleShowMore: canToggleShowMore, - toggleShowMore: cachedToggleShowMore, + createURL: connectorState.createURLFactory(isRefined, { + state: state, + createURL: createURL + }), sendEvent: sendEvent, - hasExhaustiveItems: hasExhaustiveItems + canRefine: Boolean(results ? nextRefinement.count : null), + refine: toggleRefinementFactory(helper), + widgetParams: widgetParams }; }, - dispose: function dispose(_ref4) { - var state = _ref4.state; - unmountFn(); - var withoutMaxValuesPerFacet = state.setQueryParameter('maxValuesPerFacet', undefined); - - if (operator === 'and') { - return withoutMaxValuesPerFacet.removeFacet(attribute); - } - - return withoutMaxValuesPerFacet.removeDisjunctiveFacet(attribute); - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref5) { - var searchParameters = _ref5.searchParameters; - var values = operator === 'or' ? searchParameters.getDisjunctiveRefinements(attribute) : searchParameters.getConjunctiveRefinements(attribute); + getWidgetUiState: function getWidgetUiState(uiState, _ref10) { + var searchParameters = _ref10.searchParameters; + var isRefined = on && on.every(function (v) { + return searchParameters.isDisjunctiveFacetRefined(attribute, v); + }); - if (!values.length) { + if (!isRefined) { return uiState; } return _objectSpread2(_objectSpread2({}, uiState), {}, { - refinementList: _objectSpread2(_objectSpread2({}, uiState.refinementList), {}, _defineProperty({}, attribute, values)) + toggle: _objectSpread2(_objectSpread2({}, uiState.toggle), {}, _defineProperty({}, attribute, isRefined)) }); }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { - var uiState = _ref6.uiState; - var isDisjunctive = operator === 'or'; - var values = uiState.refinementList && uiState.refinementList[attribute]; - var withoutRefinements = searchParameters.clearRefinements(attribute); - var withFacetConfiguration = isDisjunctive ? withoutRefinements.addDisjunctiveFacet(attribute) : withoutRefinements.addFacet(attribute); - var currentMaxValuesPerFacet = withFacetConfiguration.maxValuesPerFacet || 0; - var nextMaxValuesPerFacet = Math.max(currentMaxValuesPerFacet, showMore ? showMoreLimit : limit); - var withMaxValuesPerFacet = withFacetConfiguration.setQueryParameter('maxValuesPerFacet', nextMaxValuesPerFacet); + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref11) { + var uiState = _ref11.uiState; + var withFacetConfiguration = searchParameters.clearRefinements(attribute).addDisjunctiveFacet(attribute); + var isRefined = Boolean(uiState.toggle && uiState.toggle[attribute]); - if (!values) { - var key = isDisjunctive ? 'disjunctiveFacetsRefinements' : 'facetsRefinements'; - return withMaxValuesPerFacet.setQueryParameters(_defineProperty({}, key, _objectSpread2(_objectSpread2({}, withMaxValuesPerFacet[key]), {}, _defineProperty({}, attribute, [])))); - } + if (isRefined) { + if (on) { + on.forEach(function (v) { + withFacetConfiguration = withFacetConfiguration.addDisjunctiveFacetRefinement(attribute, v); + }); + } - return values.reduce(function (parameters, value) { - return isDisjunctive ? parameters.addDisjunctiveFacetRefinement(attribute, value) : parameters.addFacetRefinement(attribute, value); - }, withMaxValuesPerFacet); + return withFacetConfiguration; + } // It's not refined with an `off` value + + + if (hasAnOffValue) { + if (off) { + off.forEach(function (v) { + withFacetConfiguration = withFacetConfiguration.addDisjunctiveFacetRefinement(attribute, v); + }); + } + + return withFacetConfiguration; + } // It's not refined without an `off` value + + + return withFacetConfiguration.setQueryParameters({ + disjunctiveFacetsRefinements: _objectSpread2(_objectSpread2({}, searchParameters.disjunctiveFacetsRefinements), {}, _defineProperty({}, attribute, [])) + }); } }; }; }; - var withUsage$d = createDocumentationMessageGenerator({ - name: 'search-box', + var withUsage$i = createDocumentationMessageGenerator({ + name: 'breadcrumb', connector: true }); - var defaultQueryHook = function defaultQueryHook(query, hook) { - return hook(query); - }; - /** - * **SearchBox** connector provides the logic to build a widget that will let the user search for a query. - * - * The connector provides to the rendering: `refine()` to set the query. The behaviour of this function - * may be impacted by the `queryHook` widget parameter. - */ - - - var connectSearchBox = function connectSearchBox(renderFn) { + var connectBreadcrumb = function connectBreadcrumb(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$d()); + checkRendering(renderFn, withUsage$i()); + var connectorState = {}; return function (widgetParams) { var _ref = widgetParams || {}, - _ref$queryHook = _ref.queryHook, - queryHook = _ref$queryHook === void 0 ? defaultQueryHook : _ref$queryHook; + attributes = _ref.attributes, + _ref$separator = _ref.separator, + separator = _ref$separator === void 0 ? ' > ' : _ref$separator, + _ref$rootPath = _ref.rootPath, + rootPath = _ref$rootPath === void 0 ? null : _ref$rootPath, + _ref$transformItems = _ref.transformItems, + transformItems = _ref$transformItems === void 0 ? function (items) { + return items; + } : _ref$transformItems; - var _refine; + if (!attributes || !Array.isArray(attributes) || attributes.length === 0) { + throw new Error(withUsage$i('The `attributes` option expects an array of strings.')); + } - var _clear; + var _attributes = _slicedToArray(attributes, 1), + hierarchicalFacetName = _attributes[0]; + + function getRefinedState(state, facetValue) { + if (!facetValue) { + var breadcrumb = state.getHierarchicalFacetBreadcrumb(hierarchicalFacetName); + + if (breadcrumb.length === 0) { + return state; + } else { + return state.resetPage().toggleFacetRefinement(hierarchicalFacetName, breadcrumb[0]); + } + } + + return state.resetPage().toggleFacetRefinement(hierarchicalFacetName, facetValue); + } return { - $$type: 'ais.searchBox', + $$type: 'ais.breadcrumb', init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance + instantSearchInstance: initOptions.instantSearchInstance }), true); }, render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance + instantSearchInstance: renderOptions.instantSearchInstance }), false); - }, - dispose: function dispose(_ref2) { - var state = _ref2.state; + }, + dispose: function dispose() { unmountFn(); - return state.setQueryParameter('query', undefined); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - searchBox: this.getWidgetRenderState(renderOptions) + breadcrumb: _objectSpread2(_objectSpread2({}, renderState.breadcrumb), {}, _defineProperty({}, hierarchicalFacetName, this.getWidgetRenderState(renderOptions))) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref3) { - var helper = _ref3.helper, - searchMetadata = _ref3.searchMetadata, - state = _ref3.state; + getWidgetRenderState: function getWidgetRenderState(_ref2) { + var helper = _ref2.helper, + createURL = _ref2.createURL, + results = _ref2.results, + state = _ref2.state; - if (!_refine) { - _refine = function _refine(query) { - queryHook(query, function (q) { - return helper.setQuery(q).search(); - }); + function getItems() { + // The hierarchicalFacets condition is required for flavors + // that render immediately with empty results, without relying + // on init() (like React InstantSearch Hooks). + if (!results || state.hierarchicalFacets.length === 0) { + return []; + } + + var _state$hierarchicalFa = _slicedToArray(state.hierarchicalFacets, 1), + facetName = _state$hierarchicalFa[0].name; + + var facetValues = results.getFacetValues(facetName, {}); + var data = Array.isArray(facetValues.data) ? facetValues.data : []; + var items = transformItems(shiftItemsValues(prepareItems(data)), { + results: results + }); + return items; + } + + var items = getItems(); + + if (!connectorState.createURL) { + connectorState.createURL = function (facetValue) { + return createURL(getRefinedState(helper.state, facetValue)); }; + } - _clear = function _clear() { - helper.setQuery('').search(); + if (!connectorState.refine) { + connectorState.refine = function (facetValue) { + helper.setState(getRefinedState(helper.state, facetValue)).search(); }; } return { - query: state.query || '', - refine: _refine, - clear: _clear, - widgetParams: widgetParams, - isSearchStalled: searchMetadata.isSearchStalled + canRefine: items.length > 0, + createURL: connectorState.createURL, + items: items, + refine: connectorState.refine, + widgetParams: widgetParams }; }, - getWidgetUiState: function getWidgetUiState(uiState, _ref4) { - var searchParameters = _ref4.searchParameters; - var query = searchParameters.query || ''; - - if (query === '' || uiState && uiState.query === query) { - return uiState; + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters) { + if (searchParameters.isHierarchicalFacet(hierarchicalFacetName)) { + var facet = searchParameters.getHierarchicalFacetByName(hierarchicalFacetName); + _warning(isEqual(facet.attributes, attributes) && facet.separator === separator && facet.rootPath === rootPath, 'Using Breadcrumb and HierarchicalMenu on the same facet with different options overrides the configuration of the HierarchicalMenu.') ; + return searchParameters; } - return _objectSpread2(_objectSpread2({}, uiState), {}, { - query: query + return searchParameters.addHierarchicalFacet({ + name: hierarchicalFacetName, + attributes: attributes, + separator: separator, + rootPath: rootPath }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { - var uiState = _ref5.uiState; - return searchParameters.setQueryParameter('query', uiState.query || ''); } }; }; }; - var withUsage$e = createDocumentationMessageGenerator({ - name: 'sort-by', + function prepareItems(data) { + return data.reduce(function (result, currentItem) { + if (currentItem.isRefined) { + result.push({ + label: currentItem.name, + value: currentItem.escapedValue + }); + + if (Array.isArray(currentItem.data)) { + result = result.concat(prepareItems(currentItem.data)); + } + } + + return result; + }, []); + } + + function shiftItemsValues(array) { + return array.map(function (x, idx) { + return { + label: x.label, + value: idx + 1 === array.length ? null : array[idx + 1].value + }; + }); + } + + var withUsage$j = createDocumentationMessageGenerator({ + name: 'geo-search', connector: true - }); + }); // in this connector, we assume insideBoundingBox is only a string, + // even though in the helper it's defined as number[][] alone. + // This can be done, since the connector assumes "control" of the parameter + + function getBoundingBoxAsString(state) { + return state.insideBoundingBox || ''; + } + + function setBoundingBoxAsString(state, value) { + return state.setQueryParameter('insideBoundingBox', value); + } + + var $$type$4 = 'ais.geoSearch'; + /** - * The **SortBy** connector provides the logic to build a custom widget that will display a - * list of indices. With Algolia, this is most commonly used for changing ranking strategy. This allows - * a user to change how the hits are being sorted. + * The **GeoSearch** connector provides the logic to build a widget that will display the results on a map. It also provides a way to search for results based on their position. The connector provides functions to manage the search experience (search on map interaction or control the interaction for example). + * + * @requirements + * + * Note that the GeoSearch connector uses the [geosearch](https://www.algolia.com/doc/guides/searching/geo-search) capabilities of Algolia. Your hits **must** have a `_geoloc` attribute in order to be passed to the rendering function. + * + * Currently, the feature is not compatible with multiple values in the _geoloc attribute. */ - - var connectSortBy = function connectSortBy(renderFn) { + var connectGeoSearch = function connectGeoSearch(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$e()); - var connectorState = {}; + checkRendering(renderFn, withUsage$j()); return function (widgetParams) { var _ref = widgetParams || {}, - items = _ref.items, + _ref$enableRefineOnMa = _ref.enableRefineOnMapMove, + enableRefineOnMapMove = _ref$enableRefineOnMa === void 0 ? true : _ref$enableRefineOnMa, _ref$transformItems = _ref.transformItems, - transformItems = _ref$transformItems === void 0 ? function (x) { - return x; + transformItems = _ref$transformItems === void 0 ? function (items) { + return items; } : _ref$transformItems; - if (!Array.isArray(items)) { - throw new Error(withUsage$e('The `items` option expects an array of objects.')); - } + var widgetState = { + isRefineOnMapMove: enableRefineOnMapMove, + // @MAJOR hasMapMoveSinceLastRefine -> hasMapMovedSinceLastRefine + hasMapMoveSinceLastRefine: false, + lastRefinePosition: '', + lastRefineBoundingBox: '', + internalToggleRefineOnMapMove: noop, + internalSetMapMoveSinceLastRefine: noop + }; - return { - $$type: 'ais.sortBy', - init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - var widgetRenderState = this.getWidgetRenderState(initOptions); - var currentIndex = widgetRenderState.currentRefinement; - var isCurrentIndexInItems = find$1(items, function (item) { - return item.value === currentIndex; - }); - _warning(isCurrentIndexInItems !== undefined, "The index named \"".concat(currentIndex, "\" is not listed in the `items` of `sortBy`.")) ; - renderFn(_objectSpread2(_objectSpread2({}, widgetRenderState), {}, { - instantSearchInstance: instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), false); - }, - dispose: function dispose(_ref2) { - var state = _ref2.state; - unmountFn(); - return connectorState.initialIndex ? state.setIndex(connectorState.initialIndex) : state; - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - sortBy: this.getWidgetRenderState(renderOptions) - }); + var getPositionFromState = function getPositionFromState(state) { + return state.aroundLatLng ? aroundLatLngToPosition(state.aroundLatLng) : undefined; + }; + + var getCurrentRefinementFromState = function getCurrentRefinementFromState(state) { + return state.insideBoundingBox && insideBoundingBoxToBoundingBox(state.insideBoundingBox); + }; + + var refine = function refine(helper) { + return function (_ref2) { + var ne = _ref2.northEast, + sw = _ref2.southWest; + var boundingBox = [ne.lat, ne.lng, sw.lat, sw.lng].join(); + helper.setState(setBoundingBoxAsString(helper.state, boundingBox).resetPage()).search(); + widgetState.hasMapMoveSinceLastRefine = false; + widgetState.lastRefineBoundingBox = boundingBox; + }; + }; + + var clearMapRefinement = function clearMapRefinement(helper) { + return function () { + helper.setQueryParameter('insideBoundingBox', undefined).search(); + }; + }; + + var isRefinedWithMap = function isRefinedWithMap(state) { + return function () { + return Boolean(state.insideBoundingBox); + }; + }; + + var toggleRefineOnMapMove = function toggleRefineOnMapMove() { + return widgetState.internalToggleRefineOnMapMove(); + }; + + var createInternalToggleRefinementOnMapMove = function createInternalToggleRefinementOnMapMove(renderOptions, render) { + return function () { + widgetState.isRefineOnMapMove = !widgetState.isRefineOnMapMove; + render(renderOptions); + }; + }; + + var isRefineOnMapMove = function isRefineOnMapMove() { + return widgetState.isRefineOnMapMove; + }; + + var setMapMoveSinceLastRefine = function setMapMoveSinceLastRefine() { + return widgetState.internalSetMapMoveSinceLastRefine(); + }; + + var createInternalSetMapMoveSinceLastRefine = function createInternalSetMapMoveSinceLastRefine(renderOptions, render) { + return function () { + var shouldTriggerRender = widgetState.hasMapMoveSinceLastRefine !== true; + widgetState.hasMapMoveSinceLastRefine = true; + + if (shouldTriggerRender) { + render(renderOptions); + } + }; + }; + + var hasMapMoveSinceLastRefine = function hasMapMoveSinceLastRefine() { + return widgetState.hasMapMoveSinceLastRefine; + }; + + var sendEvent; + return { + $$type: $$type$4, + init: function init(initArgs) { + var instantSearchInstance = initArgs.instantSearchInstance; + var isFirstRendering = true; + widgetState.internalToggleRefineOnMapMove = createInternalToggleRefinementOnMapMove(initArgs, noop); + widgetState.internalSetMapMoveSinceLastRefine = createInternalSetMapMoveSinceLastRefine(initArgs, noop); + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initArgs)), {}, { + instantSearchInstance: instantSearchInstance + }), isFirstRendering); }, - getWidgetRenderState: function getWidgetRenderState(_ref3) { - var results = _ref3.results, - helper = _ref3.helper, - state = _ref3.state, - parent = _ref3.parent; + render: function render(renderArgs) { + var helper = renderArgs.helper, + instantSearchInstance = renderArgs.instantSearchInstance; + var isFirstRendering = false; // We don't use the state provided by the render function because we need + // to be sure that the state is the latest one for the following condition - if (!connectorState.initialIndex && parent) { - connectorState.initialIndex = parent.getIndexName(); + var state = helper.state; + var positionChangedSinceLastRefine = Boolean(state.aroundLatLng) && Boolean(widgetState.lastRefinePosition) && state.aroundLatLng !== widgetState.lastRefinePosition; + var boundingBoxChangedSinceLastRefine = !state.insideBoundingBox && Boolean(widgetState.lastRefineBoundingBox) && state.insideBoundingBox !== widgetState.lastRefineBoundingBox; + + if (positionChangedSinceLastRefine || boundingBoxChangedSinceLastRefine) { + widgetState.hasMapMoveSinceLastRefine = false; } - if (!connectorState.setIndex) { - connectorState.setIndex = function (indexName) { - helper.setIndex(indexName).search(); - }; + widgetState.lastRefinePosition = state.aroundLatLng || ''; + widgetState.lastRefineBoundingBox = getBoundingBoxAsString(state); + widgetState.internalToggleRefineOnMapMove = createInternalToggleRefinementOnMapMove(renderArgs, this.render.bind(this)); + widgetState.internalSetMapMoveSinceLastRefine = createInternalSetMapMoveSinceLastRefine(renderArgs, this.render.bind(this)); + var widgetRenderState = this.getWidgetRenderState(renderArgs); + sendEvent('view', widgetRenderState.items); + renderFn(_objectSpread2(_objectSpread2({}, widgetRenderState), {}, { + instantSearchInstance: instantSearchInstance + }), isFirstRendering); + }, + getWidgetRenderState: function getWidgetRenderState(renderOptions) { + var helper = renderOptions.helper, + results = renderOptions.results, + instantSearchInstance = renderOptions.instantSearchInstance; + var state = helper.state; + var items = results ? transformItems(results.hits.filter(function (hit) { + return hit._geoloc; + }), { + results: results + }) : []; + + if (!sendEvent) { + sendEvent = createSendEventForHits({ + instantSearchInstance: instantSearchInstance, + index: helper.getIndex(), + widgetType: $$type$4 + }); } return { - currentRefinement: state.index, - options: transformItems(items, { - results: results - }), - refine: connectorState.setIndex, - hasNoResults: results ? results.nbHits === 0 : true, + items: items, + position: getPositionFromState(state), + currentRefinement: getCurrentRefinementFromState(state), + refine: refine(helper), + sendEvent: sendEvent, + clearMapRefinement: clearMapRefinement(helper), + isRefinedWithMap: isRefinedWithMap(state), + toggleRefineOnMapMove: toggleRefineOnMapMove, + isRefineOnMapMove: isRefineOnMapMove, + setMapMoveSinceLastRefine: setMapMoveSinceLastRefine, + hasMapMoveSinceLastRefine: hasMapMoveSinceLastRefine, widgetParams: widgetParams }; }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + geoSearch: this.getWidgetRenderState(renderOptions) + }); + }, + dispose: function dispose(_ref3) { + var state = _ref3.state; + unmountFn(); + return state.setQueryParameter('insideBoundingBox', undefined); + }, getWidgetUiState: function getWidgetUiState(uiState, _ref4) { var searchParameters = _ref4.searchParameters; - var currentIndex = searchParameters.index; + var boundingBox = getBoundingBoxAsString(searchParameters); + + if (!boundingBox || uiState && uiState.geoSearch && uiState.geoSearch.boundingBox === boundingBox) { + return uiState; + } + return _objectSpread2(_objectSpread2({}, uiState), {}, { - sortBy: currentIndex !== connectorState.initialIndex ? currentIndex : undefined + geoSearch: { + boundingBox: boundingBox + } }); }, getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { var uiState = _ref5.uiState; - return searchParameters.setQueryParameter('index', uiState.sortBy || connectorState.initialIndex || searchParameters.index); + + if (!uiState || !uiState.geoSearch) { + return searchParameters.setQueryParameter('insideBoundingBox', undefined); + } + + return setBoundingBoxAsString(searchParameters, uiState.geoSearch.boundingBox); } }; }; }; - var withUsage$f = createDocumentationMessageGenerator({ - name: 'rating-menu', + var withUsage$k = createDocumentationMessageGenerator({ + name: 'powered-by', connector: true }); - var $$type$2 = 'ais.ratingMenu'; - var MAX_VALUES_PER_FACET_API_LIMIT = 1000; - var STEP = 1; - - var createSendEvent$1 = function createSendEvent(_ref) { - var instantSearchInstance = _ref.instantSearchInstance, - helper = _ref.helper, - getRefinedStar = _ref.getRefinedStar, - attribute = _ref.attribute; - return function () { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } - - if (args.length === 1) { - instantSearchInstance.sendEventToInsights(args[0]); - return; - } - - var eventType = args[0], - facetValue = args[1], - _args$ = args[2], - eventName = _args$ === void 0 ? 'Filter Applied' : _args$; - - if (eventType !== 'click') { - return; - } - - var isRefined = getRefinedStar() === Number(facetValue); - - if (!isRefined) { - instantSearchInstance.sendEventToInsights({ - insightsMethod: 'clickedFilters', - widgetType: $$type$2, - eventType: eventType, - payload: { - eventName: eventName, - index: helper.getIndex(), - filters: ["".concat(attribute, ">=").concat(facetValue)] - }, - attribute: attribute - }); - } - }; - }; /** - * **StarRating** connector provides the logic to build a custom widget that will let - * the user refine search results based on ratings. - * - * The connector provides to the rendering: `refine()` to select a value and - * `items` that are the values that can be selected. `refine` should be used - * with `items.value`. + * **PoweredBy** connector provides the logic to build a custom widget that will displays + * the logo to redirect to Algolia. */ - var connectRatingMenu = function connectRatingMenu(renderFn) { + var connectPoweredBy = function connectPoweredBy(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$f()); - return function (widgetParams) { - var _ref2 = widgetParams || {}, - attribute = _ref2.attribute, - _ref2$max = _ref2.max, - max = _ref2$max === void 0 ? 5 : _ref2$max; - - var sendEvent; - - if (!attribute) { - throw new Error(withUsage$f('The `attribute` option is required.')); - } - - var _getRefinedStar = function getRefinedStar(state) { - var _values$; - - var values = state.getNumericRefinements(attribute); - - if (!((_values$ = values['>=']) !== null && _values$ !== void 0 && _values$.length)) { - return undefined; - } - - return values['>='][0]; - }; - - var getFacetsMaxDecimalPlaces = function getFacetsMaxDecimalPlaces(facetResults) { - var maxDecimalPlaces = 0; - facetResults.forEach(function (facetResult) { - var _facetResult$name$spl = facetResult.name.split('.'), - _facetResult$name$spl2 = _slicedToArray(_facetResult$name$spl, 2), - _facetResult$name$spl3 = _facetResult$name$spl2[1], - decimal = _facetResult$name$spl3 === void 0 ? '' : _facetResult$name$spl3; - - maxDecimalPlaces = Math.max(maxDecimalPlaces, decimal.length); - }); - return maxDecimalPlaces; - }; - - var getFacetValuesWarningMessage = function getFacetValuesWarningMessage(_ref3) { - var maxDecimalPlaces = _ref3.maxDecimalPlaces, - maxFacets = _ref3.maxFacets, - maxValuesPerFacet = _ref3.maxValuesPerFacet; - var maxDecimalPlacesInRange = Math.max(0, Math.floor(Math.log10(MAX_VALUES_PER_FACET_API_LIMIT / max))); - var maxFacetsInRange = Math.min(MAX_VALUES_PER_FACET_API_LIMIT, Math.pow(10, maxDecimalPlacesInRange) * max); - var solutions = []; - - if (maxFacets > MAX_VALUES_PER_FACET_API_LIMIT) { - solutions.push("- Update your records to lower the precision of the values in the \"".concat(attribute, "\" attribute (for example: ").concat(5.123456789.toPrecision(maxDecimalPlaces + 1), " to ").concat(5.123456789.toPrecision(maxDecimalPlacesInRange + 1), ")")); - } - - if (maxValuesPerFacet < maxFacetsInRange) { - solutions.push("- Increase the maximum number of facet values to ".concat(maxFacetsInRange, " using the \"configure\" widget ").concat(createDocumentationLink({ - name: 'configure' - }), " and the \"maxValuesPerFacet\" parameter https://www.algolia.com/doc/api-reference/api-parameters/maxValuesPerFacet/")); - } - - return "The ".concat(attribute, " attribute can have ").concat(maxFacets, " different values (0 to ").concat(max, " with a maximum of ").concat(maxDecimalPlaces, " decimals = ").concat(maxFacets, ") but you retrieved only ").concat(maxValuesPerFacet, " facet values. Therefore the number of results that match the refinements can be incorrect.\n ").concat(solutions.length ? "To resolve this problem you can:\n".concat(solutions.join('\n')) : ""); - }; - - function getRefinedState(state, facetValue) { - var isRefined = _getRefinedStar(state) === Number(facetValue); - var emptyState = state.resetPage().removeNumericRefinement(attribute); - - if (!isRefined) { - return emptyState.addNumericRefinement(attribute, '<=', max).addNumericRefinement(attribute, '>=', Number(facetValue)); - } - - return emptyState; - } - - var toggleRefinement = function toggleRefinement(helper, facetValue) { - sendEvent('click', facetValue); - helper.setState(getRefinedState(helper.state, facetValue)).search(); - }; + checkRendering(renderFn, withUsage$k()); + var defaultUrl = 'https://www.algolia.com/?' + 'utm_source=instantsearch.js&' + 'utm_medium=website&' + "utm_content=".concat(safelyRunOnBrowser(function (_ref) { + var _window$location; + + var window = _ref.window; + return ((_window$location = window.location) === null || _window$location === void 0 ? void 0 : _window$location.hostname) || ''; + }, { + fallback: function fallback() { + return ''; + } + }), "&") + 'utm_campaign=poweredby'; + return function (widgetParams) { + var _ref2 = widgetParams || {}, + _ref2$url = _ref2.url, + url = _ref2$url === void 0 ? defaultUrl : _ref2$url; - var connectorState = { - toggleRefinementFactory: function toggleRefinementFactory(helper) { - return toggleRefinement.bind(null, helper); - }, - createURLFactory: function createURLFactory(_ref4) { - var state = _ref4.state, - createURL = _ref4.createURL; - return function (value) { - return createURL(getRefinedState(state, value)); - }; - } - }; return { - $$type: $$type$2, + $$type: 'ais.poweredBy', init: function init(initOptions) { var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { @@ -13511,143 +13693,210 @@ }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - ratingMenu: _objectSpread2(_objectSpread2({}, renderState.ratingMenu), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) + poweredBy: this.getWidgetRenderState(renderOptions) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref5) { - var helper = _ref5.helper, - results = _ref5.results, - state = _ref5.state, - instantSearchInstance = _ref5.instantSearchInstance, - createURL = _ref5.createURL; - var facetValues = []; + getWidgetRenderState: function getWidgetRenderState() { + return { + url: url, + widgetParams: widgetParams + }; + }, + dispose: function dispose() { + unmountFn(); + } + }; + }; + }; - if (!sendEvent) { - sendEvent = createSendEvent$1({ - instantSearchInstance: instantSearchInstance, - helper: helper, - getRefinedStar: function getRefinedStar() { - return _getRefinedStar(helper.state); - }, - attribute: attribute - }); - } + /** + * Refine the given search parameters. + */ - if (results) { - var facetResults = results.getFacetValues(attribute, {}); - var maxValuesPerFacet = facetResults.length; - var maxDecimalPlaces = getFacetsMaxDecimalPlaces(facetResults); - var maxFacets = Math.pow(10, maxDecimalPlaces) * max; - _warning(maxFacets <= maxValuesPerFacet, getFacetValuesWarningMessage({ - maxDecimalPlaces: maxDecimalPlaces, - maxFacets: maxFacets, - maxValuesPerFacet: maxValuesPerFacet - })) ; + var withUsage$l = createDocumentationMessageGenerator({ + name: 'configure', + connector: true + }); - var refinedStar = _getRefinedStar(state); + function getInitialSearchParameters(state, widgetParams) { + // We leverage the helper internals to remove the `widgetParams` from + // the state. The function `setQueryParameters` omits the values that + // are `undefined` on the next state. + return state.setQueryParameters(Object.keys(widgetParams.searchParameters).reduce(function (acc, key) { + return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, key, undefined)); + }, {})); + } - var _loop = function _loop(star) { - var isRefined = refinedStar === star; - var count = facetResults.filter(function (f) { - return Number(f.name) >= star && Number(f.name) <= max; - }).map(function (f) { - return f.count; - }).reduce(function (sum, current) { - return sum + current; - }, 0); + var connectConfigure = function connectConfigure() { + var renderFn = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : noop; + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + return function (widgetParams) { + if (!widgetParams || !isPlainObject(widgetParams.searchParameters)) { + throw new Error(withUsage$l('The `searchParameters` option expects an object.')); + } - if (refinedStar && !isRefined && count === 0) { - // skip count==0 when at least 1 refinement is enabled - // eslint-disable-next-line no-continue - return "continue"; - } + var connectorState = {}; - var stars = _toConsumableArray(new Array(Math.floor(max / STEP))).map(function (_v, i) { - return i * STEP < star; - }); + function refine(helper) { + return function (searchParameters) { + // Merge new `searchParameters` with the ones set from other widgets + var actualState = getInitialSearchParameters(helper.state, widgetParams); + var nextSearchParameters = mergeSearchParameters(actualState, new algoliasearchHelper_1.SearchParameters(searchParameters)); // Update original `widgetParams.searchParameters` to the new refined one - facetValues.push({ - stars: stars, - name: String(star), - label: String(star), - value: String(star), - count: count, - isRefined: isRefined - }); - }; + widgetParams.searchParameters = searchParameters; // Trigger a search with the resolved search parameters - for (var star = STEP; star < max; star += STEP) { - var _ret = _loop(star); + helper.setState(nextSearchParameters).search(); + }; + } - if (_ret === "continue") continue; - } + return { + $$type: 'ais.configure', + init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), false); + }, + dispose: function dispose(_ref) { + var state = _ref.state; + unmountFn(); + return getInitialSearchParameters(state, widgetParams); + }, + getRenderState: function getRenderState(renderState, renderOptions) { + var _renderState$configur; + + var widgetRenderState = this.getWidgetRenderState(renderOptions); + return _objectSpread2(_objectSpread2({}, renderState), {}, { + configure: _objectSpread2(_objectSpread2({}, widgetRenderState), {}, { + widgetParams: _objectSpread2(_objectSpread2({}, widgetRenderState.widgetParams), {}, { + searchParameters: mergeSearchParameters(new algoliasearchHelper_1.SearchParameters((_renderState$configur = renderState.configure) === null || _renderState$configur === void 0 ? void 0 : _renderState$configur.widgetParams.searchParameters), new algoliasearchHelper_1.SearchParameters(widgetRenderState.widgetParams.searchParameters)).getQueryParams() + }) + }) + }); + }, + getWidgetRenderState: function getWidgetRenderState(_ref2) { + var helper = _ref2.helper; + + if (!connectorState.refine) { + connectorState.refine = refine(helper); } - facetValues = facetValues.reverse(); return { - items: facetValues, - hasNoResults: results ? results.nbHits === 0 : true, - canRefine: facetValues.length > 0, - refine: connectorState.toggleRefinementFactory(helper), - sendEvent: sendEvent, - createURL: connectorState.createURLFactory({ - state: state, - createURL: createURL - }), + refine: connectorState.refine, widgetParams: widgetParams }; }, - dispose: function dispose(_ref6) { - var state = _ref6.state; - unmountFn(); - return state.removeNumericRefinement(attribute); + getWidgetSearchParameters: function getWidgetSearchParameters(state, _ref3) { + var uiState = _ref3.uiState; + return mergeSearchParameters(state, new algoliasearchHelper_1.SearchParameters(_objectSpread2(_objectSpread2({}, uiState.configure), widgetParams.searchParameters))); }, - getWidgetUiState: function getWidgetUiState(uiState, _ref7) { - var searchParameters = _ref7.searchParameters; + getWidgetUiState: function getWidgetUiState(uiState) { + return _objectSpread2(_objectSpread2({}, uiState), {}, { + configure: _objectSpread2(_objectSpread2({}, uiState.configure), widgetParams.searchParameters) + }); + } + }; + }; + }; - var value = _getRefinedStar(searchParameters); + var withUsage$m = createDocumentationMessageGenerator({ + name: 'configure-related-items', + connector: true + }); - if (typeof value !== 'number') { - return uiState; - } + function createOptionalFilter(_ref) { + var attributeName = _ref.attributeName, + attributeValue = _ref.attributeValue, + attributeScore = _ref.attributeScore; + return "".concat(attributeName, ":").concat(attributeValue, ""); + } - return _objectSpread2(_objectSpread2({}, uiState), {}, { - ratingMenu: _objectSpread2(_objectSpread2({}, uiState.ratingMenu), {}, _defineProperty({}, attribute, value)) - }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref8) { - var uiState = _ref8.uiState; - var value = uiState.ratingMenu && uiState.ratingMenu[attribute]; - var withoutRefinements = searchParameters.clearRefinements(attribute); - var withDisjunctiveFacet = withoutRefinements.addDisjunctiveFacet(attribute); + var connectConfigureRelatedItems = function connectConfigureRelatedItems(renderFn, unmountFn) { + return function (widgetParams) { + var _ref2 = widgetParams || {}, + hit = _ref2.hit, + matchingPatterns = _ref2.matchingPatterns, + _ref2$transformSearch = _ref2.transformSearchParameters, + transformSearchParameters = _ref2$transformSearch === void 0 ? function (x) { + return x; + } : _ref2$transformSearch; - if (!value) { - return withDisjunctiveFacet.setQueryParameters({ - numericRefinements: _objectSpread2(_objectSpread2({}, withDisjunctiveFacet.numericRefinements), {}, _defineProperty({}, attribute, {})) + if (!hit) { + throw new Error(withUsage$m('The `hit` option is required.')); + } + + if (!matchingPatterns) { + throw new Error(withUsage$m('The `matchingPatterns` option is required.')); + } + + var optionalFilters = Object.keys(matchingPatterns).reduce(function (acc, attributeName) { + var attribute = matchingPatterns[attributeName]; + var attributeValue = getPropertyByPath(hit, attributeName); + var attributeScore = attribute.score; + + if (Array.isArray(attributeValue)) { + return [].concat(_toConsumableArray(acc), [attributeValue.map(function (attributeSubValue) { + return createOptionalFilter({ + attributeName: attributeName, + attributeValue: attributeSubValue, + attributeScore: attributeScore }); - } + })]); + } - return withDisjunctiveFacet.addNumericRefinement(attribute, '<=', max).addNumericRefinement(attribute, '>=', value); + if (typeof attributeValue === 'string') { + return [].concat(_toConsumableArray(acc), [createOptionalFilter({ + attributeName: attributeName, + attributeValue: attributeValue, + attributeScore: attributeScore + })]); } - }; + + _warning(false, "\nThe `matchingPatterns` option returned a value of type ".concat(getObjectType(attributeValue), " for the \"").concat(attributeName, "\" key. This value was not sent to Algolia because `optionalFilters` only supports strings and array of strings.\n\nYou can remove the \"").concat(attributeName, "\" key from the `matchingPatterns` option.\n\nSee https://www.algolia.com/doc/api-reference/api-parameters/optionalFilters/\n ")) ; + return acc; + }, []); + + var searchParameters = _objectSpread2({}, transformSearchParameters(new algoliasearchHelper_1.SearchParameters({ + sumOrFiltersScores: true, + facetFilters: ["objectID:-".concat(hit.objectID)], + optionalFilters: optionalFilters + }))); + + var makeWidget = connectConfigure(renderFn, unmountFn); + return _objectSpread2(_objectSpread2({}, makeWidget({ + searchParameters: searchParameters + })), {}, { + $$type: 'ais.configureRelatedItems' + }); }; }; - var withUsage$g = createDocumentationMessageGenerator({ - name: 'stats', + var withUsage$n = createDocumentationMessageGenerator({ + name: 'autocomplete', connector: true }); - /** - * **Stats** connector provides the logic to build a custom widget that will displays - * search statistics (hits number and processing time). - */ - var connectStats = function connectStats(renderFn) { + var connectAutocomplete = function connectAutocomplete(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$g()); + checkRendering(renderFn, withUsage$n()); return function (widgetParams) { + var _ref = widgetParams || {}, + _ref$escapeHTML = _ref.escapeHTML, + escapeHTML = _ref$escapeHTML === void 0 ? true : _ref$escapeHTML; + + _warning(!widgetParams.indices, "\nThe option `indices` has been removed from the Autocomplete connector.\n\nThe indices to target are now inferred from the widgets tree.\n".concat(Array.isArray(widgetParams.indices) ? "\nAn alternative would be:\n\nconst autocomplete = connectAutocomplete(renderer);\n\nsearch.addWidgets([\n ".concat(widgetParams.indices.map(function (_ref2) { + var value = _ref2.value; + return "index({ indexName: '".concat(value, "' }),"); + }).join('\n '), "\n autocomplete()\n]);\n") : '', "\n ")) ; + var connectorState = {}; return { - $$type: 'ais.stats', + $$type: 'ais.autocomplete', init: function init(initOptions) { var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { @@ -13656,2126 +13905,2264 @@ }, render: function render(renderOptions) { var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + var renderState = this.getWidgetRenderState(renderOptions); + renderState.indices.forEach(function (_ref3) { + var sendEvent = _ref3.sendEvent, + hits = _ref3.hits; + sendEvent('view', hits); + }); + renderFn(_objectSpread2(_objectSpread2({}, renderState), {}, { instantSearchInstance: instantSearchInstance }), false); }, - dispose: function dispose() { - unmountFn(); - }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - stats: this.getWidgetRenderState(renderOptions) + autocomplete: this.getWidgetRenderState(renderOptions) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref) { - var results = _ref.results, - state = _ref.state; + getWidgetRenderState: function getWidgetRenderState(_ref4) { + var _this = this; - if (!results) { - return { - hitsPerPage: state.hitsPerPage, - nbHits: 0, - nbSortedHits: undefined, - areHitsSorted: false, - nbPages: 0, - page: state.page || 0, - processingTimeMS: -1, - query: state.query || '', - widgetParams: widgetParams + var helper = _ref4.helper, + state = _ref4.state, + scopedResults = _ref4.scopedResults, + instantSearchInstance = _ref4.instantSearchInstance; + + if (!connectorState.refine) { + connectorState.refine = function (query) { + helper.setQuery(query).search(); }; } + var indices = scopedResults.map(function (scopedResult) { + // We need to escape the hits because highlighting + // exposes HTML tags to the end-user. + scopedResult.results.hits = escapeHTML ? escapeHits(scopedResult.results.hits) : scopedResult.results.hits; + var sendEvent = createSendEventForHits({ + instantSearchInstance: instantSearchInstance, + index: scopedResult.results.index, + widgetType: _this.$$type + }); + return { + indexId: scopedResult.indexId, + indexName: scopedResult.results.index, + hits: scopedResult.results.hits, + results: scopedResult.results, + sendEvent: sendEvent + }; + }); return { - hitsPerPage: results.hitsPerPage, - nbHits: results.nbHits, - nbSortedHits: results.nbSortedHits, - areHitsSorted: typeof results.appliedRelevancyStrictness !== 'undefined' && results.appliedRelevancyStrictness > 0 && results.nbSortedHits !== results.nbHits, - nbPages: results.nbPages, - page: results.page, - processingTimeMS: results.processingTimeMS, - query: results.query, + currentRefinement: state.query || '', + indices: indices, + refine: connectorState.refine, widgetParams: widgetParams }; - } - }; - }; - }; - - var withUsage$h = createDocumentationMessageGenerator({ - name: 'toggle-refinement', - connector: true - }); - var $$type$3 = 'ais.toggleRefinement'; - - var createSendEvent$2 = function createSendEvent(_ref) { - var instantSearchInstance = _ref.instantSearchInstance, - helper = _ref.helper, - attribute = _ref.attribute, - on = _ref.on; + }, + getWidgetUiState: function getWidgetUiState(uiState, _ref5) { + var searchParameters = _ref5.searchParameters; + var query = searchParameters.query || ''; - var sendEventForToggle = function sendEventForToggle() { - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } + if (query === '' || uiState && uiState.query === query) { + return uiState; + } - if (args.length === 1) { - instantSearchInstance.sendEventToInsights(args[0]); - return; - } + return _objectSpread2(_objectSpread2({}, uiState), {}, { + query: query + }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { + var uiState = _ref6.uiState; + var parameters = { + query: uiState.query || '' + }; - var eventType = args[0], - isRefined = args[1], - _args$ = args[2], - eventName = _args$ === void 0 ? 'Filter Applied' : _args$; + if (!escapeHTML) { + return searchParameters.setQueryParameters(parameters); + } - if (eventType !== 'click' || on === undefined) { - return; - } // only send an event when the refinement gets applied, - // not when it gets removed + return searchParameters.setQueryParameters(_objectSpread2(_objectSpread2({}, parameters), TAG_PLACEHOLDER)); + }, + dispose: function dispose(_ref7) { + var state = _ref7.state; + unmountFn(); + var stateWithoutQuery = state.setQueryParameter('query', undefined); + if (!escapeHTML) { + return stateWithoutQuery; + } - if (!isRefined) { - instantSearchInstance.sendEventToInsights({ - insightsMethod: 'clickedFilters', - widgetType: $$type$3, - eventType: eventType, - payload: { - eventName: eventName, - index: helper.getIndex(), - filters: on.map(function (value) { - return "".concat(attribute, ":").concat(value); - }) - }, - attribute: attribute - }); - } + return stateWithoutQuery.setQueryParameters(Object.keys(TAG_PLACEHOLDER).reduce(function (acc, key) { + return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, key, undefined)); + }, {})); + } + }; }; - - return sendEventForToggle; }; - /** - * **Toggle** connector provides the logic to build a custom widget that will provide - * an on/off filtering feature based on an attribute value or values. - * - * Two modes are implemented in the custom widget: - * - with or without the value filtered - * - switch between two values. - */ - var connectToggleRefinement = function connectToggleRefinement(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$h()); - return function (widgetParams) { - var _ref2 = widgetParams || {}, - attribute = _ref2.attribute, - _ref2$on = _ref2.on, - userOn = _ref2$on === void 0 ? true : _ref2$on, - userOff = _ref2.off; - - if (!attribute) { - throw new Error(withUsage$h('The `attribute` option is required.')); - } + var withUsage$o = createDocumentationMessageGenerator({ + name: 'query-rules', + connector: true + }); - var hasAnOffValue = userOff !== undefined; - var on = toArray(userOn).map(escapeFacetValue$4); - var off = hasAnOffValue ? toArray(userOff).map(escapeFacetValue$4) : undefined; - var sendEvent; + function hasStateRefinements(state) { + return [state.disjunctiveFacetsRefinements, state.facetsRefinements, state.hierarchicalFacetsRefinements, state.numericRefinements].some(function (refinement) { + return Boolean(refinement && Object.keys(refinement).length > 0); + }); + } // A context rule must consist only of alphanumeric characters, hyphens, and underscores. + // See https://www.algolia.com/doc/guides/managing-results/refine-results/merchandising-and-promoting/in-depth/implementing-query-rules/#context - var toggleRefinementFactory = function toggleRefinementFactory(helper) { - return function () { - var _ref3 = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : { - isRefined: false - }, - isRefined = _ref3.isRefined; - if (!isRefined) { - sendEvent('click', isRefined); + function escapeRuleContext(ruleName) { + return ruleName.replace(/[^a-z0-9-_]+/gi, '_'); + } - if (hasAnOffValue) { - off.forEach(function (v) { - return helper.removeDisjunctiveFacetRefinement(attribute, v); - }); - } + function getRuleContextsFromTrackedFilters(_ref) { + var helper = _ref.helper, + sharedHelperState = _ref.sharedHelperState, + trackedFilters = _ref.trackedFilters; + var ruleContexts = Object.keys(trackedFilters).reduce(function (facets, facetName) { + var facetRefinements = getRefinements(helper.lastResults || {}, sharedHelperState, true).filter(function (refinement) { + return refinement.attribute === facetName; + }).map(function (refinement) { + return refinement.numericValue || refinement.name; + }); + var getTrackedFacetValues = trackedFilters[facetName]; + var trackedFacetValues = getTrackedFacetValues(facetRefinements); + return [].concat(_toConsumableArray(facets), _toConsumableArray(facetRefinements.filter(function (facetRefinement) { + return trackedFacetValues.includes(facetRefinement); + }).map(function (facetValue) { + return escapeRuleContext("ais-".concat(facetName, "-").concat(facetValue)); + }))); + }, []); + return ruleContexts; + } - on.forEach(function (v) { - return helper.addDisjunctiveFacetRefinement(attribute, v); - }); - } else { - on.forEach(function (v) { - return helper.removeDisjunctiveFacetRefinement(attribute, v); - }); + function applyRuleContexts(event) { + var helper = this.helper, + initialRuleContexts = this.initialRuleContexts, + trackedFilters = this.trackedFilters, + transformRuleContexts = this.transformRuleContexts; + var sharedHelperState = event.state; + var previousRuleContexts = sharedHelperState.ruleContexts || []; + var newRuleContexts = getRuleContextsFromTrackedFilters({ + helper: helper, + sharedHelperState: sharedHelperState, + trackedFilters: trackedFilters + }); + var nextRuleContexts = [].concat(_toConsumableArray(initialRuleContexts), _toConsumableArray(newRuleContexts)); + _warning(nextRuleContexts.length <= 10, "\nThe maximum number of `ruleContexts` is 10. They have been sliced to that limit.\nConsider using `transformRuleContexts` to minimize the number of rules sent to Algolia.\n") ; + var ruleContexts = transformRuleContexts(nextRuleContexts).slice(0, 10); - if (hasAnOffValue) { - off.forEach(function (v) { - return helper.addDisjunctiveFacetRefinement(attribute, v); - }); - } - } + if (!isEqual(previousRuleContexts, ruleContexts)) { + helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread2(_objectSpread2({}, sharedHelperState), {}, { + ruleContexts: ruleContexts + })); + } + } - helper.search(); - }; - }; + var connectQueryRules = function connectQueryRules(_render) { + var unmount = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + checkRendering(_render, withUsage$o()); + return function (widgetParams) { + var _ref2 = widgetParams || {}, + _ref2$trackedFilters = _ref2.trackedFilters, + trackedFilters = _ref2$trackedFilters === void 0 ? {} : _ref2$trackedFilters, + _ref2$transformRuleCo = _ref2.transformRuleContexts, + transformRuleContexts = _ref2$transformRuleCo === void 0 ? function (rules) { + return rules; + } : _ref2$transformRuleCo, + _ref2$transformItems = _ref2.transformItems, + transformItems = _ref2$transformItems === void 0 ? function (items) { + return items; + } : _ref2$transformItems; - var connectorState = { - createURLFactory: function createURLFactory(isRefined, _ref4) { - var state = _ref4.state, - createURL = _ref4.createURL; - return function () { - state = state.resetPage(); - var valuesToRemove = isRefined ? on : off; + Object.keys(trackedFilters).forEach(function (facetName) { + if (typeof trackedFilters[facetName] !== 'function') { + throw new Error(withUsage$o("'The \"".concat(facetName, "\" filter value in the `trackedFilters` option expects a function."))); + } + }); + var hasTrackedFilters = Object.keys(trackedFilters).length > 0; // We store the initial rule contexts applied before creating the widget + // so that we do not override them with the rules created from `trackedFilters`. - if (valuesToRemove) { - valuesToRemove.forEach(function (v) { - state = state.removeDisjunctiveFacetRefinement(attribute, v); + var initialRuleContexts = []; + var onHelperChange; + return { + $$type: 'ais.queryRules', + init: function init(initOptions) { + var helper = initOptions.helper, + state = initOptions.state, + instantSearchInstance = initOptions.instantSearchInstance; + initialRuleContexts = state.ruleContexts || []; + onHelperChange = applyRuleContexts.bind({ + helper: helper, + initialRuleContexts: initialRuleContexts, + trackedFilters: trackedFilters, + transformRuleContexts: transformRuleContexts + }); + + if (hasTrackedFilters) { + // We need to apply the `ruleContexts` based on the `trackedFilters` + // before the helper changes state in some cases: + // - Some filters are applied on the first load (e.g. using `configure`) + // - The `transformRuleContexts` option sets initial `ruleContexts`. + if (hasStateRefinements(state) || Boolean(widgetParams.transformRuleContexts)) { + onHelperChange({ + state: state }); - } + } // We track every change in the helper to override its state and add + // any `ruleContexts` needed based on the `trackedFilters`. - var valuesToAdd = isRefined ? off : on; - if (valuesToAdd) { - valuesToAdd.forEach(function (v) { - state = state.addDisjunctiveFacetRefinement(attribute, v); - }); - } + helper.on('change', onHelperChange); + } - return createURL(state); - }; - } - }; - return { - $$type: $$type$3, - init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + _render(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { instantSearchInstance: instantSearchInstance }), true); }, render: function render(renderOptions) { var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + + _render(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { instantSearchInstance: instantSearchInstance }), false); }, - dispose: function dispose(_ref5) { - var state = _ref5.state; - unmountFn(); - return state.removeDisjunctiveFacet(attribute); + getWidgetRenderState: function getWidgetRenderState(_ref3) { + var results = _ref3.results; + + var _ref4 = results || {}, + _ref4$userData = _ref4.userData, + userData = _ref4$userData === void 0 ? [] : _ref4$userData; + + var items = transformItems(userData, { + results: results + }); + return { + items: items, + widgetParams: widgetParams + }; }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - toggleRefinement: _objectSpread2(_objectSpread2({}, renderState.toggleRefinement), {}, _defineProperty({}, attribute, this.getWidgetRenderState(renderOptions))) + queryRules: this.getWidgetRenderState(renderOptions) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref6) { - var state = _ref6.state, - helper = _ref6.helper, - results = _ref6.results, - createURL = _ref6.createURL, - instantSearchInstance = _ref6.instantSearchInstance; - var isRefined = results ? on.every(function (v) { - return state.isDisjunctiveFacetRefined(attribute, v); - }) : on.every(function (v) { - return state.isDisjunctiveFacetRefined(attribute, v); - }); - var onFacetValue = { - isRefined: isRefined, - count: 0 - }; - var offFacetValue = { - isRefined: hasAnOffValue && !isRefined, - count: 0 - }; + dispose: function dispose(_ref5) { + var helper = _ref5.helper, + state = _ref5.state; + unmount(); - if (results) { - var offValue = toArray(off || false); - var allFacetValues = results.getFacetValues(attribute, {}) || []; - var onData = on.map(function (v) { - return find$1(allFacetValues, function (_ref7) { - var escapedValue = _ref7.escapedValue; - return escapedValue === escapeFacetValue$4(String(v)); - }); - }).filter(function (v) { - return v !== undefined; - }); - var offData = hasAnOffValue ? offValue.map(function (v) { - return find$1(allFacetValues, function (_ref8) { - var escapedValue = _ref8.escapedValue; - return escapedValue === escapeFacetValue$4(String(v)); - }); - }).filter(function (v) { - return v !== undefined; - }) : []; - onFacetValue = { - isRefined: onData.length ? onData.every(function (v) { - return v.isRefined; - }) : false, - count: onData.reduce(function (acc, v) { - return acc + v.count; - }, 0) || null - }; - offFacetValue = { - isRefined: offData.length ? offData.every(function (v) { - return v.isRefined; - }) : false, - count: offData.reduce(function (acc, v) { - return acc + v.count; - }, 0) || allFacetValues.reduce(function (total, _ref9) { - var count = _ref9.count; - return total + count; - }, 0) - }; + if (hasTrackedFilters) { + helper.removeListener('change', onHelperChange); + return state.setQueryParameter('ruleContexts', initialRuleContexts); } - if (!sendEvent) { - sendEvent = createSendEvent$2({ - instantSearchInstance: instantSearchInstance, - attribute: attribute, - on: on, - helper: helper - }); - } + return state; + } + }; + }; + }; + + // `SpeechRecognition` is an API used on the browser so we can safely disable + // the `window` check. + + /* eslint-disable no-restricted-globals */ + + /* global SpeechRecognition SpeechRecognitionEvent */ + var createVoiceSearchHelper = function createVoiceSearchHelper(_ref) { + var searchAsYouSpeak = _ref.searchAsYouSpeak, + language = _ref.language, + onQueryChange = _ref.onQueryChange, + onStateChange = _ref.onStateChange; + var SpeechRecognitionAPI = window.webkitSpeechRecognition || window.SpeechRecognition; + + var getDefaultState = function getDefaultState(status) { + return { + status: status, + transcript: '', + isSpeechFinal: false, + errorCode: undefined + }; + }; + + var state = getDefaultState('initial'); + var recognition; + + var isBrowserSupported = function isBrowserSupported() { + return Boolean(SpeechRecognitionAPI); + }; + + var isListening = function isListening() { + return state.status === 'askingPermission' || state.status === 'waiting' || state.status === 'recognizing'; + }; + + var setState = function setState() { + var newState = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; + state = _objectSpread2(_objectSpread2({}, state), newState); + onStateChange(); + }; + + var getState = function getState() { + return state; + }; + + var resetState = function resetState() { + var status = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'initial'; + setState(getDefaultState(status)); + }; + + var onStart = function onStart() { + setState({ + status: 'waiting' + }); + }; + + var onError = function onError(event) { + setState({ + status: 'error', + errorCode: event.error + }); + }; + + var onResult = function onResult(event) { + setState({ + status: 'recognizing', + transcript: event.results[0] && event.results[0][0] && event.results[0][0].transcript || '', + isSpeechFinal: event.results[0] && event.results[0].isFinal + }); + + if (searchAsYouSpeak && state.transcript) { + onQueryChange(state.transcript); + } + }; + + var onEnd = function onEnd() { + if (!state.errorCode && state.transcript && !searchAsYouSpeak) { + onQueryChange(state.transcript); + } - var nextRefinement = isRefined ? offFacetValue : onFacetValue; - return { - value: { - name: attribute, - isRefined: isRefined, - count: results ? nextRefinement.count : null, - onFacetValue: onFacetValue, - offFacetValue: offFacetValue - }, - createURL: connectorState.createURLFactory(isRefined, { - state: state, - createURL: createURL - }), - sendEvent: sendEvent, - canRefine: Boolean(results ? nextRefinement.count : null), - refine: toggleRefinementFactory(helper), - widgetParams: widgetParams - }; - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref10) { - var searchParameters = _ref10.searchParameters; - var isRefined = on && on.every(function (v) { - return searchParameters.isDisjunctiveFacetRefined(attribute, v); - }); + if (state.status !== 'error') { + setState({ + status: 'finished' + }); + } + }; - if (!isRefined) { - return uiState; - } + var startListening = function startListening() { + recognition = new SpeechRecognitionAPI(); - return _objectSpread2(_objectSpread2({}, uiState), {}, { - toggle: _objectSpread2(_objectSpread2({}, uiState.toggle), {}, _defineProperty({}, attribute, isRefined)) - }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref11) { - var uiState = _ref11.uiState; - var withFacetConfiguration = searchParameters.clearRefinements(attribute).addDisjunctiveFacet(attribute); - var isRefined = Boolean(uiState.toggle && uiState.toggle[attribute]); + if (!recognition) { + return; + } - if (isRefined) { - if (on) { - on.forEach(function (v) { - withFacetConfiguration = withFacetConfiguration.addDisjunctiveFacetRefinement(attribute, v); - }); - } + resetState('askingPermission'); + recognition.interimResults = true; - return withFacetConfiguration; - } // It's not refined with an `off` value + if (language) { + recognition.lang = language; + } + recognition.addEventListener('start', onStart); + recognition.addEventListener('error', onError); + recognition.addEventListener('result', onResult); + recognition.addEventListener('end', onEnd); + recognition.start(); + }; - if (hasAnOffValue) { - if (off) { - off.forEach(function (v) { - withFacetConfiguration = withFacetConfiguration.addDisjunctiveFacetRefinement(attribute, v); - }); - } + var dispose = function dispose() { + if (!recognition) { + return; + } - return withFacetConfiguration; - } // It's not refined without an `off` value + recognition.stop(); + recognition.removeEventListener('start', onStart); + recognition.removeEventListener('error', onError); + recognition.removeEventListener('result', onResult); + recognition.removeEventListener('end', onEnd); + recognition = undefined; + }; + + var stopListening = function stopListening() { + dispose(); // Because `dispose` removes event listeners, `end` listener is not called. + // So we're setting the `status` as `finished` here. + // If we don't do it, it will be still `waiting` or `recognizing`. + resetState('finished'); + }; - return withFacetConfiguration.setQueryParameters({ - disjunctiveFacetsRefinements: _objectSpread2(_objectSpread2({}, searchParameters.disjunctiveFacetsRefinements), {}, _defineProperty({}, attribute, [])) - }); - } - }; + return { + getState: getState, + isBrowserSupported: isBrowserSupported, + isListening: isListening, + startListening: startListening, + stopListening: stopListening, + dispose: dispose }; }; - var withUsage$i = createDocumentationMessageGenerator({ - name: 'breadcrumb', + var withUsage$p = createDocumentationMessageGenerator({ + name: 'voice-search', connector: true }); - var connectBreadcrumb = function connectBreadcrumb(renderFn) { + var connectVoiceSearch = function connectVoiceSearch(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$i()); - var connectorState = {}; + checkRendering(renderFn, withUsage$p()); return function (widgetParams) { - var _ref = widgetParams || {}, - attributes = _ref.attributes, - _ref$separator = _ref.separator, - separator = _ref$separator === void 0 ? ' > ' : _ref$separator, - _ref$rootPath = _ref.rootPath, - rootPath = _ref$rootPath === void 0 ? null : _ref$rootPath, - _ref$transformItems = _ref.transformItems, - transformItems = _ref$transformItems === void 0 ? function (items) { - return items; - } : _ref$transformItems; - - if (!attributes || !Array.isArray(attributes) || attributes.length === 0) { - throw new Error(withUsage$i('The `attributes` option expects an array of strings.')); - } - - var _attributes = _slicedToArray(attributes, 1), - hierarchicalFacetName = _attributes[0]; - - function getRefinedState(state, facetValue) { - if (!facetValue) { - var breadcrumb = state.getHierarchicalFacetBreadcrumb(hierarchicalFacetName); - - if (breadcrumb.length === 0) { - return state; - } else { - return state.resetPage().toggleFacetRefinement(hierarchicalFacetName, breadcrumb[0]); - } - } - - return state.resetPage().toggleFacetRefinement(hierarchicalFacetName, facetValue); - } - + var _widgetParams$searchA = widgetParams.searchAsYouSpeak, + searchAsYouSpeak = _widgetParams$searchA === void 0 ? false : _widgetParams$searchA, + language = widgetParams.language, + additionalQueryParameters = widgetParams.additionalQueryParameters, + _widgetParams$createV = widgetParams.createVoiceSearchHelper, + createVoiceSearchHelper$1 = _widgetParams$createV === void 0 ? createVoiceSearchHelper : _widgetParams$createV; return { - $$type: 'ais.breadcrumb', + $$type: 'ais.voiceSearch', init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: initOptions.instantSearchInstance + instantSearchInstance: instantSearchInstance }), true); }, render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: renderOptions.instantSearchInstance + instantSearchInstance: instantSearchInstance }), false); }, - dispose: function dispose() { - unmountFn(); - }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - breadcrumb: _objectSpread2(_objectSpread2({}, renderState.breadcrumb), {}, _defineProperty({}, hierarchicalFacetName, this.getWidgetRenderState(renderOptions))) + voiceSearch: this.getWidgetRenderState(renderOptions) }); }, - getWidgetRenderState: function getWidgetRenderState(_ref2) { - var helper = _ref2.helper, - createURL = _ref2.createURL, - results = _ref2.results, - state = _ref2.state; + getWidgetRenderState: function getWidgetRenderState(renderOptions) { + var _this = this; - function getItems() { - // The hierarchicalFacets condition is required for flavors - // that render immediately with empty results, without relying - // on init() (like React InstantSearch Hooks). - if (!results || state.hierarchicalFacets.length === 0) { - return []; - } + var helper = renderOptions.helper, + instantSearchInstance = renderOptions.instantSearchInstance; - var _state$hierarchicalFa = _slicedToArray(state.hierarchicalFacets, 1), - facetName = _state$hierarchicalFa[0].name; + if (!this._refine) { + this._refine = function (query) { + if (query !== helper.state.query) { + var queryLanguages = language ? [language.split('-')[0]] : undefined; + helper.setQueryParameter('queryLanguages', queryLanguages); - var facetValues = results.getFacetValues(facetName, {}); - var data = Array.isArray(facetValues.data) ? facetValues.data : []; - var items = transformItems(shiftItemsValues(prepareItems(data)), { - results: results + if (typeof additionalQueryParameters === 'function') { + helper.setState(helper.state.setQueryParameters(_objectSpread2({ + ignorePlurals: true, + removeStopWords: true, + // @ts-ignore (optionalWords only allows array in v3, while string is also valid) + optionalWords: query + }, additionalQueryParameters({ + query: query + })))); + } + + helper.setQuery(query).search(); + } + }; + } + + if (!this._voiceSearchHelper) { + this._voiceSearchHelper = createVoiceSearchHelper$1({ + searchAsYouSpeak: searchAsYouSpeak, + language: language, + onQueryChange: function onQueryChange(query) { + return _this._refine(query); + }, + onStateChange: function onStateChange() { + renderFn(_objectSpread2(_objectSpread2({}, _this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), false); + } }); - return items; } - var items = getItems(); + var _voiceSearchHelper = this._voiceSearchHelper, + isBrowserSupported = _voiceSearchHelper.isBrowserSupported, + isListening = _voiceSearchHelper.isListening, + startListening = _voiceSearchHelper.startListening, + stopListening = _voiceSearchHelper.stopListening, + getState = _voiceSearchHelper.getState; + return { + isBrowserSupported: isBrowserSupported(), + isListening: isListening(), + toggleListening: function toggleListening() { + if (!isBrowserSupported()) { + return; + } + + if (isListening()) { + stopListening(); + } else { + startListening(); + } + }, + voiceListeningState: getState(), + widgetParams: widgetParams + }; + }, + dispose: function dispose(_ref) { + var state = _ref.state; + + this._voiceSearchHelper.dispose(); - if (!connectorState.createURL) { - connectorState.createURL = function (facetValue) { - return createURL(getRefinedState(helper.state, facetValue)); - }; - } + unmountFn(); + var newState = state; - if (!connectorState.refine) { - connectorState.refine = function (facetValue) { - helper.setState(getRefinedState(helper.state, facetValue)).search(); - }; + if (typeof additionalQueryParameters === 'function') { + var additional = additionalQueryParameters({ + query: '' + }); + var toReset = additional ? Object.keys(additional).reduce(function (acc, current) { + // @ts-ignore search parameters is typed as readonly in v4 + acc[current] = undefined; + return acc; + }, {}) : {}; + newState = state.setQueryParameters(_objectSpread2({ + // @ts-ignore (queryLanguages is not added to algoliasearch v3) + queryLanguages: undefined, + ignorePlurals: undefined, + removeStopWords: undefined, + optionalWords: undefined + }, toReset)); } - return { - canRefine: items.length > 0, - createURL: connectorState.createURL, - items: items, - refine: connectorState.refine, - widgetParams: widgetParams - }; + return newState.setQueryParameter('query', undefined); }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters) { - if (searchParameters.isHierarchicalFacet(hierarchicalFacetName)) { - var facet = searchParameters.getHierarchicalFacetByName(hierarchicalFacetName); - _warning(isEqual(facet.attributes, attributes) && facet.separator === separator && facet.rootPath === rootPath, 'Using Breadcrumb and HierarchicalMenu on the same facet with different options overrides the configuration of the HierarchicalMenu.') ; - return searchParameters; + getWidgetUiState: function getWidgetUiState(uiState, _ref2) { + var searchParameters = _ref2.searchParameters; + var query = searchParameters.query || ''; + + if (!query) { + return uiState; } - return searchParameters.addHierarchicalFacet({ - name: hierarchicalFacetName, - attributes: attributes, - separator: separator, - rootPath: rootPath + return _objectSpread2(_objectSpread2({}, uiState), {}, { + query: query }); + }, + getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref3) { + var uiState = _ref3.uiState; + return searchParameters.setQueryParameter('query', uiState.query || ''); } }; }; }; - function prepareItems(data) { - return data.reduce(function (result, currentItem) { - if (currentItem.isRefined) { - result.push({ - label: currentItem.name, - value: currentItem.escapedValue - }); - - if (Array.isArray(currentItem.data)) { - result = result.concat(prepareItems(currentItem.data)); - } - } - - return result; - }, []); - } - - function shiftItemsValues(array) { - return array.map(function (x, idx) { - return { - label: x.label, - value: idx + 1 === array.length ? null : array[idx + 1].value - }; - }); + function hasFindAnswersMethod(answersIndex) { + return typeof answersIndex.findAnswers === 'function'; } - var withUsage$j = createDocumentationMessageGenerator({ - name: 'geo-search', + var withUsage$q = createDocumentationMessageGenerator({ + name: 'answers', connector: true - }); // in this connector, we assume insideBoundingBox is only a string, - // even though in the helper it's defined as number[][] alone. - // This can be done, since the connector assumes "control" of the parameter - - function getBoundingBoxAsString(state) { - return state.insideBoundingBox || ''; - } - - function setBoundingBoxAsString(state, value) { - return state.setQueryParameter('insideBoundingBox', value); - } - - var $$type$4 = 'ais.geoSearch'; + }); - /** - * The **GeoSearch** connector provides the logic to build a widget that will display the results on a map. It also provides a way to search for results based on their position. The connector provides functions to manage the search experience (search on map interaction or control the interaction for example). - * - * @requirements - * - * Note that the GeoSearch connector uses the [geosearch](https://www.algolia.com/doc/guides/searching/geo-search) capabilities of Algolia. Your hits **must** have a `_geoloc` attribute in order to be passed to the rendering function. - * - * Currently, the feature is not compatible with multiple values in the _geoloc attribute. - */ - var connectGeoSearch = function connectGeoSearch(renderFn) { + var connectAnswers = function connectAnswers(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$j()); + checkRendering(renderFn, withUsage$q()); return function (widgetParams) { var _ref = widgetParams || {}, - _ref$enableRefineOnMa = _ref.enableRefineOnMapMove, - enableRefineOnMapMove = _ref$enableRefineOnMa === void 0 ? true : _ref$enableRefineOnMa, - _ref$transformItems = _ref.transformItems, - transformItems = _ref$transformItems === void 0 ? function (items) { - return items; - } : _ref$transformItems; - - var widgetState = { - isRefineOnMapMove: enableRefineOnMapMove, - // @MAJOR hasMapMoveSinceLastRefine -> hasMapMovedSinceLastRefine - hasMapMoveSinceLastRefine: false, - lastRefinePosition: '', - lastRefineBoundingBox: '', - internalToggleRefineOnMapMove: noop, - internalSetMapMoveSinceLastRefine: noop - }; - - var getPositionFromState = function getPositionFromState(state) { - return state.aroundLatLng ? aroundLatLngToPosition(state.aroundLatLng) : undefined; - }; - - var getCurrentRefinementFromState = function getCurrentRefinementFromState(state) { - return state.insideBoundingBox && insideBoundingBoxToBoundingBox(state.insideBoundingBox); - }; - - var refine = function refine(helper) { - return function (_ref2) { - var ne = _ref2.northEast, - sw = _ref2.southWest; - var boundingBox = [ne.lat, ne.lng, sw.lat, sw.lng].join(); - helper.setState(setBoundingBoxAsString(helper.state, boundingBox).resetPage()).search(); - widgetState.hasMapMoveSinceLastRefine = false; - widgetState.lastRefineBoundingBox = boundingBox; - }; - }; - - var clearMapRefinement = function clearMapRefinement(helper) { - return function () { - helper.setQueryParameter('insideBoundingBox', undefined).search(); - }; - }; - - var isRefinedWithMap = function isRefinedWithMap(state) { - return function () { - return Boolean(state.insideBoundingBox); - }; - }; - - var toggleRefineOnMapMove = function toggleRefineOnMapMove() { - return widgetState.internalToggleRefineOnMapMove(); - }; + queryLanguages = _ref.queryLanguages, + attributesForPrediction = _ref.attributesForPrediction, + _ref$nbHits = _ref.nbHits, + nbHits = _ref$nbHits === void 0 ? 1 : _ref$nbHits, + _ref$renderDebounceTi = _ref.renderDebounceTime, + renderDebounceTime = _ref$renderDebounceTi === void 0 ? 100 : _ref$renderDebounceTi, + _ref$searchDebounceTi = _ref.searchDebounceTime, + searchDebounceTime = _ref$searchDebounceTi === void 0 ? 100 : _ref$searchDebounceTi, + _ref$escapeHTML = _ref.escapeHTML, + escapeHTML = _ref$escapeHTML === void 0 ? true : _ref$escapeHTML, + _ref$extraParameters = _ref.extraParameters, + extraParameters = _ref$extraParameters === void 0 ? {} : _ref$extraParameters; // @ts-expect-error checking for the wrong value - var createInternalToggleRefinementOnMapMove = function createInternalToggleRefinementOnMapMove(renderOptions, render) { - return function () { - widgetState.isRefineOnMapMove = !widgetState.isRefineOnMapMove; - render(renderOptions); - }; - }; - var isRefineOnMapMove = function isRefineOnMapMove() { - return widgetState.isRefineOnMapMove; - }; + if (!queryLanguages || queryLanguages.length === 0) { + throw new Error(withUsage$q('The `queryLanguages` expects an array of strings.')); + } - var setMapMoveSinceLastRefine = function setMapMoveSinceLastRefine() { - return widgetState.internalSetMapMoveSinceLastRefine(); - }; + var runConcurrentSafePromise = createConcurrentSafePromise(); + var lastHits = []; + var isLoading = false; + var debouncedRender = debounce(renderFn, renderDebounceTime); // this does not directly use DebouncedFunction, since then the generic will disappear - var createInternalSetMapMoveSinceLastRefine = function createInternalSetMapMoveSinceLastRefine(renderOptions, render) { - return function () { - var shouldTriggerRender = widgetState.hasMapMoveSinceLastRefine !== true; - widgetState.hasMapMoveSinceLastRefine = true; + var debouncedRefine; + return { + $$type: 'ais.answers', + init: function init(initOptions) { + var state = initOptions.state, + instantSearchInstance = initOptions.instantSearchInstance; + var answersIndex = instantSearchInstance.client.initIndex(state.index); - if (shouldTriggerRender) { - render(renderOptions); + if (!hasFindAnswersMethod(answersIndex)) { + throw new Error(withUsage$q('`algoliasearch` >= 4.8.0 required.')); } - }; - }; - - var hasMapMoveSinceLastRefine = function hasMapMoveSinceLastRefine() { - return widgetState.hasMapMoveSinceLastRefine; - }; - var sendEvent; - return { - $$type: $$type$4, - init: function init(initArgs) { - var instantSearchInstance = initArgs.instantSearchInstance; - var isFirstRendering = true; - widgetState.internalToggleRefineOnMapMove = createInternalToggleRefinementOnMapMove(initArgs, noop); - widgetState.internalSetMapMoveSinceLastRefine = createInternalSetMapMoveSinceLastRefine(initArgs, noop); - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initArgs)), {}, { - instantSearchInstance: instantSearchInstance - }), isFirstRendering); + debouncedRefine = debounce(answersIndex.findAnswers, searchDebounceTime); + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: initOptions.instantSearchInstance + }), true); }, - render: function render(renderArgs) { - var helper = renderArgs.helper, - instantSearchInstance = renderArgs.instantSearchInstance; - var isFirstRendering = false; // We don't use the state provided by the render function because we need - // to be sure that the state is the latest one for the following condition + render: function render(renderOptions) { + var _this = this; + + var query = renderOptions.state.query; + + if (!query) { + // renders nothing with empty query + lastHits = []; + isLoading = false; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: renderOptions.instantSearchInstance + }), false); + return; + } // render the loader - var state = helper.state; - var positionChangedSinceLastRefine = Boolean(state.aroundLatLng) && Boolean(widgetState.lastRefinePosition) && state.aroundLatLng !== widgetState.lastRefinePosition; - var boundingBoxChangedSinceLastRefine = !state.insideBoundingBox && Boolean(widgetState.lastRefineBoundingBox) && state.insideBoundingBox !== widgetState.lastRefineBoundingBox; - if (positionChangedSinceLastRefine || boundingBoxChangedSinceLastRefine) { - widgetState.hasMapMoveSinceLastRefine = false; - } + lastHits = []; + isLoading = true; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: renderOptions.instantSearchInstance + }), false); // call /answers API - widgetState.lastRefinePosition = state.aroundLatLng || ''; - widgetState.lastRefineBoundingBox = getBoundingBoxAsString(state); - widgetState.internalToggleRefineOnMapMove = createInternalToggleRefinementOnMapMove(renderArgs, this.render.bind(this)); - widgetState.internalSetMapMoveSinceLastRefine = createInternalSetMapMoveSinceLastRefine(renderArgs, this.render.bind(this)); - var widgetRenderState = this.getWidgetRenderState(renderArgs); - sendEvent('view', widgetRenderState.items); - renderFn(_objectSpread2(_objectSpread2({}, widgetRenderState), {}, { - instantSearchInstance: instantSearchInstance - }), isFirstRendering); - }, - getWidgetRenderState: function getWidgetRenderState(renderOptions) { - var helper = renderOptions.helper, - results = renderOptions.results, - instantSearchInstance = renderOptions.instantSearchInstance; - var state = helper.state; - var items = results ? transformItems(results.hits.filter(function (hit) { - return hit._geoloc; - }), { - results: results - }) : []; + runConcurrentSafePromise(debouncedRefine(query, queryLanguages, _objectSpread2(_objectSpread2({}, extraParameters), {}, { + nbHits: nbHits, + attributesForPrediction: attributesForPrediction + }))).then(function (result) { + if (!result) { + // It's undefined when it's debounced. + return; + } - if (!sendEvent) { - sendEvent = createSendEventForHits({ - instantSearchInstance: instantSearchInstance, - index: helper.getIndex(), - widgetType: $$type$4 - }); - } + if (escapeHTML && result.hits.length > 0) { + result.hits = escapeHits(result.hits); + } - return { - items: items, - position: getPositionFromState(state), - currentRefinement: getCurrentRefinementFromState(state), - refine: refine(helper), - sendEvent: sendEvent, - clearMapRefinement: clearMapRefinement(helper), - isRefinedWithMap: isRefinedWithMap(state), - toggleRefineOnMapMove: toggleRefineOnMapMove, - isRefineOnMapMove: isRefineOnMapMove, - setMapMoveSinceLastRefine: setMapMoveSinceLastRefine, - hasMapMoveSinceLastRefine: hasMapMoveSinceLastRefine, - widgetParams: widgetParams - }; + var hitsWithAbsolutePosition = addAbsolutePosition(result.hits, 0, nbHits); + var hitsWithAbsolutePositionAndQueryID = addQueryID(hitsWithAbsolutePosition, result.queryID); + lastHits = hitsWithAbsolutePositionAndQueryID; + isLoading = false; + debouncedRender(_objectSpread2(_objectSpread2({}, _this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: renderOptions.instantSearchInstance + }), false); + }); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - geoSearch: this.getWidgetRenderState(renderOptions) + answers: this.getWidgetRenderState(renderOptions) }); }, - dispose: function dispose(_ref3) { - var state = _ref3.state; + getWidgetRenderState: function getWidgetRenderState() { + return { + hits: lastHits, + isLoading: isLoading, + widgetParams: widgetParams + }; + }, + dispose: function dispose(_ref2) { + var state = _ref2.state; unmountFn(); - return state.setQueryParameter('insideBoundingBox', undefined); + return state; }, - getWidgetUiState: function getWidgetUiState(uiState, _ref4) { - var searchParameters = _ref4.searchParameters; - var boundingBox = getBoundingBoxAsString(searchParameters); - - if (!boundingBox || uiState && uiState.geoSearch && uiState.geoSearch.boundingBox === boundingBox) { - return uiState; - } + getWidgetSearchParameters: function getWidgetSearchParameters(state) { + return state; + } + }; + }; + }; - return _objectSpread2(_objectSpread2({}, uiState), {}, { - geoSearch: { - boundingBox: boundingBox - } + var connectRelevantSort = function connectRelevantSort() { + var renderFn = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : noop; + var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; + return function (widgetParams) { + var connectorState = {}; + return { + $$type: 'ais.relevantSort', + init: function init(initOptions) { + var instantSearchInstance = initOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + var instantSearchInstance = renderOptions.instantSearchInstance; + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { + instantSearchInstance: instantSearchInstance + }), false); + }, + dispose: function dispose(_ref) { + var state = _ref.state; + unmountFn(); + return state.setQueryParameter('relevancyStrictness', undefined); + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + relevantSort: this.getWidgetRenderState(renderOptions) }); }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref5) { - var uiState = _ref5.uiState; + getWidgetRenderState: function getWidgetRenderState(_ref2) { + var results = _ref2.results, + helper = _ref2.helper; - if (!uiState || !uiState.geoSearch) { - return searchParameters.setQueryParameter('insideBoundingBox', undefined); + if (!connectorState.refine) { + connectorState.refine = function (relevancyStrictness) { + helper.setQueryParameter('relevancyStrictness', relevancyStrictness).search(); + }; } - return setBoundingBoxAsString(searchParameters, uiState.geoSearch.boundingBox); + var _ref3 = results || {}, + appliedRelevancyStrictness = _ref3.appliedRelevancyStrictness; + + var isVirtualReplica = appliedRelevancyStrictness !== undefined; + return { + isRelevantSorted: typeof appliedRelevancyStrictness !== 'undefined' && appliedRelevancyStrictness > 0, + isVirtualReplica: isVirtualReplica, + canRefine: isVirtualReplica, + refine: connectorState.refine, + widgetParams: widgetParams + }; + }, + getWidgetSearchParameters: function getWidgetSearchParameters(state, _ref4) { + var _uiState$relevantSort; + + var uiState = _ref4.uiState; + return state.setQueryParameter('relevancyStrictness', (_uiState$relevantSort = uiState.relevantSort) !== null && _uiState$relevantSort !== void 0 ? _uiState$relevantSort : state.relevancyStrictness); + }, + getWidgetUiState: function getWidgetUiState(uiState, _ref5) { + var searchParameters = _ref5.searchParameters; + return _objectSpread2(_objectSpread2({}, uiState), {}, { + relevantSort: searchParameters.relevancyStrictness || uiState.relevantSort + }); } }; }; }; - var withUsage$k = createDocumentationMessageGenerator({ - name: 'powered-by', + var withUsage$r = createDocumentationMessageGenerator({ + name: 'dynamic-widgets', connector: true }); + var MAX_WILDCARD_FACETS = 20; - /** - * **PoweredBy** connector provides the logic to build a custom widget that will displays - * the logo to redirect to Algolia. - */ - var connectPoweredBy = function connectPoweredBy(renderFn) { + var connectDynamicWidgets = function connectDynamicWidgets(renderFn) { var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$k()); - var defaultUrl = 'https://www.algolia.com/?' + 'utm_source=instantsearch.js&' + 'utm_medium=website&' + "utm_content=".concat(safelyRunOnBrowser(function (_ref) { - var _window$location; + checkRendering(renderFn, withUsage$r()); + return function (widgetParams) { + var widgets = widgetParams.widgets, + _widgetParams$maxValu = widgetParams.maxValuesPerFacet, + maxValuesPerFacet = _widgetParams$maxValu === void 0 ? 20 : _widgetParams$maxValu, + _widgetParams$facets = widgetParams.facets, + facets = _widgetParams$facets === void 0 ? ['*'] : _widgetParams$facets, + _widgetParams$transfo = widgetParams.transformItems, + transformItems = _widgetParams$transfo === void 0 ? function (items) { + return items; + } : _widgetParams$transfo, + fallbackWidget = widgetParams.fallbackWidget; + + if (!(widgets && Array.isArray(widgets) && widgets.every(function (widget) { + return _typeof(widget) === 'object'; + }))) { + throw new Error(withUsage$r('The `widgets` option expects an array of widgets.')); + } + + if (!(Array.isArray(facets) && facets.length <= 1 && (facets[0] === '*' || facets[0] === undefined))) { + throw new Error(withUsage$r("The `facets` option only accepts [] or [\"*\"], you passed ".concat(JSON.stringify(facets)))); + } + + var localWidgets = new Map(); + return { + $$type: 'ais.dynamicWidgets', + init: function init(initOptions) { + widgets.forEach(function (widget) { + var attribute = getWidgetAttribute(widget, initOptions); + localWidgets.set(attribute, { + widget: widget, + isMounted: false + }); + }); + renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { + instantSearchInstance: initOptions.instantSearchInstance + }), true); + }, + render: function render(renderOptions) { + var parent = renderOptions.parent; + var renderState = this.getWidgetRenderState(renderOptions); + var widgetsToUnmount = []; + var widgetsToMount = []; + + if (fallbackWidget) { + renderState.attributesToRender.forEach(function (attribute) { + if (!localWidgets.has(attribute)) { + var widget = fallbackWidget({ + attribute: attribute + }); + localWidgets.set(attribute, { + widget: widget, + isMounted: false + }); + } + }); + } + + localWidgets.forEach(function (_ref, attribute) { + var widget = _ref.widget, + isMounted = _ref.isMounted; + var shouldMount = renderState.attributesToRender.indexOf(attribute) > -1; + + if (!isMounted && shouldMount) { + widgetsToMount.push(widget); + localWidgets.set(attribute, { + widget: widget, + isMounted: true + }); + } else if (isMounted && !shouldMount) { + widgetsToUnmount.push(widget); + localWidgets.set(attribute, { + widget: widget, + isMounted: false + }); + } + }); + parent.addWidgets(widgetsToMount); // make sure this only happens after the regular render, otherwise it + // happens too quick, since render is "deferred" for the next microtask, + // so this needs to be a whole task later - var window = _ref.window; - return ((_window$location = window.location) === null || _window$location === void 0 ? void 0 : _window$location.hostname) || ''; - }, { - fallback: function fallback() { - return ''; - } - }), "&") + 'utm_campaign=poweredby'; - return function (widgetParams) { - var _ref2 = widgetParams || {}, - _ref2$url = _ref2.url, - url = _ref2$url === void 0 ? defaultUrl : _ref2$url; + setTimeout(function () { + return parent.removeWidgets(widgetsToUnmount); + }, 0); + renderFn(_objectSpread2(_objectSpread2({}, renderState), {}, { + instantSearchInstance: renderOptions.instantSearchInstance + }), false); + }, + dispose: function dispose(_ref2) { + var parent = _ref2.parent; + var toRemove = []; + localWidgets.forEach(function (_ref3) { + var widget = _ref3.widget, + isMounted = _ref3.isMounted; - return { - $$type: 'ais.poweredBy', - init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), true); + if (isMounted) { + toRemove.push(widget); + } + }); + parent.removeWidgets(toRemove); + unmountFn(); }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), false); + getWidgetSearchParameters: function getWidgetSearchParameters(state) { + // broadening the scope of facets to avoid conflict between never and * + return facets.reduce(function (acc, curr) { + return acc.addFacet(curr); + }, state.setQueryParameters({ + maxValuesPerFacet: Math.max(maxValuesPerFacet || 0, state.maxValuesPerFacet || 0) + })); }, getRenderState: function getRenderState(renderState, renderOptions) { return _objectSpread2(_objectSpread2({}, renderState), {}, { - poweredBy: this.getWidgetRenderState(renderOptions) + dynamicWidgets: this.getWidgetRenderState(renderOptions) }); }, - getWidgetRenderState: function getWidgetRenderState() { + getWidgetRenderState: function getWidgetRenderState(_ref4) { + var _results$renderingCon, _results$renderingCon2, _results$renderingCon3, _results$renderingCon4; + + var results = _ref4.results, + state = _ref4.state; + + if (!results) { + return { + attributesToRender: [], + widgetParams: widgetParams + }; + } + + var attributesToRender = transformItems((_results$renderingCon = (_results$renderingCon2 = results.renderingContent) === null || _results$renderingCon2 === void 0 ? void 0 : (_results$renderingCon3 = _results$renderingCon2.facetOrdering) === null || _results$renderingCon3 === void 0 ? void 0 : (_results$renderingCon4 = _results$renderingCon3.facets) === null || _results$renderingCon4 === void 0 ? void 0 : _results$renderingCon4.order) !== null && _results$renderingCon !== void 0 ? _results$renderingCon : [], { + results: results + }); + + if (!Array.isArray(attributesToRender)) { + throw new Error(withUsage$r('The `transformItems` option expects a function that returns an Array.')); + } + + _warning(maxValuesPerFacet >= (state.maxValuesPerFacet || 0), "The maxValuesPerFacet set by dynamic widgets (".concat(maxValuesPerFacet, ") is smaller than one of the limits set by a widget (").concat(state.maxValuesPerFacet, "). This causes a mismatch in query parameters and thus an extra network request when that widget is mounted.")) ; + _warning(attributesToRender.length <= MAX_WILDCARD_FACETS || widgetParams.facets !== undefined, "More than ".concat(MAX_WILDCARD_FACETS, " facets are requested to be displayed without explicitly setting which facets to retrieve. This could have a performance impact. Set \"facets\" to [] to do two smaller network requests, or explicitly to ['*'] to avoid this warning.")) ; return { - url: url, + attributesToRender: attributesToRender, widgetParams: widgetParams }; - }, - dispose: function dispose() { - unmountFn(); } }; }; }; - /** - * Refine the given search parameters. - */ + /** @deprecated use connectDynamicWidgets */ - var withUsage$l = createDocumentationMessageGenerator({ - name: 'configure', - connector: true + var EXPERIMENTAL_connectDynamicWidgets = deprecate(connectDynamicWidgets, 'use connectDynamicWidgets'); + + var connectors = /*#__PURE__*/Object.freeze({ + __proto__: null, + connectDynamicWidgets: connectDynamicWidgets, + EXPERIMENTAL_connectDynamicWidgets: EXPERIMENTAL_connectDynamicWidgets, + connectClearRefinements: connectClearRefinements, + connectCurrentRefinements: connectCurrentRefinements, + connectHierarchicalMenu: connectHierarchicalMenu, + connectHits: connectHits, + connectHitsWithInsights: connectHitsWithInsights, + connectHitsPerPage: connectHitsPerPage, + connectInfiniteHits: connectInfiniteHits, + connectInfiniteHitsWithInsights: connectInfiniteHitsWithInsights, + connectMenu: connectMenu, + connectNumericMenu: connectNumericMenu, + connectPagination: connectPagination, + connectRange: connectRange, + connectRefinementList: connectRefinementList, + connectSearchBox: connectSearchBox, + connectSortBy: connectSortBy, + connectRatingMenu: connectRatingMenu, + connectStats: connectStats, + connectToggleRefinement: connectToggleRefinement, + connectBreadcrumb: connectBreadcrumb, + connectGeoSearch: connectGeoSearch, + connectPoweredBy: connectPoweredBy, + connectConfigure: connectConfigure, + EXPERIMENTAL_connectConfigureRelatedItems: connectConfigureRelatedItems, + connectAutocomplete: connectAutocomplete, + connectQueryRules: connectQueryRules, + connectVoiceSearch: connectVoiceSearch, + EXPERIMENTAL_connectAnswers: connectAnswers, + connectRelevantSort: connectRelevantSort }); - function getInitialSearchParameters(state, widgetParams) { - // We leverage the helper internals to remove the `widgetParams` from - // the state. The function `setQueryParameters` omits the values that - // are `undefined` on the next state. - return state.setQueryParameters(Object.keys(widgetParams.searchParameters).reduce(function (acc, key) { - return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, key, undefined)); - }, {})); - } + var withUsage$s = createDocumentationMessageGenerator({ + name: 'analytics' + }); - var connectConfigure = function connectConfigure() { - var renderFn = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : noop; - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - return function (widgetParams) { - if (!widgetParams || !isPlainObject(widgetParams.searchParameters)) { - throw new Error(withUsage$l('The `searchParameters` option expects an object.')); - } + // @major this widget will be removed from the next major version. + var analytics = function analytics(widgetParams) { + var _ref = widgetParams || {}, + pushFunction = _ref.pushFunction, + _ref$delay = _ref.delay, + delay = _ref$delay === void 0 ? 3000 : _ref$delay, + _ref$triggerOnUIInter = _ref.triggerOnUIInteraction, + triggerOnUIInteraction = _ref$triggerOnUIInter === void 0 ? false : _ref$triggerOnUIInter, + _ref$pushInitialSearc = _ref.pushInitialSearch, + pushInitialSearch = _ref$pushInitialSearc === void 0 ? true : _ref$pushInitialSearc, + _ref$pushPagination = _ref.pushPagination, + pushPagination = _ref$pushPagination === void 0 ? false : _ref$pushPagination; - var connectorState = {}; + if (!pushFunction) { + throw new Error(withUsage$s('The `pushFunction` option is required.')); + } - function refine(helper) { - return function (searchParameters) { - // Merge new `searchParameters` with the ones set from other widgets - var actualState = getInitialSearchParameters(helper.state, widgetParams); - var nextSearchParameters = merge$1(actualState, new algoliasearchHelper_1.SearchParameters(searchParameters)); // Update original `widgetParams.searchParameters` to the new refined one + _warning(false, "`analytics` widget has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the `insights` middleware.\n\nFor the migration, visit https://www.algolia.com/doc/guides/building-search-ui/upgrade-guides/js/#analytics-widget") ; + var cachedState = null; - widgetParams.searchParameters = searchParameters; // Trigger a search with the resolved search parameters + var serializeRefinements = function serializeRefinements(parameters) { + var refinements = []; - helper.setState(nextSearchParameters).search(); - }; + for (var parameter in parameters) { + if (parameters.hasOwnProperty(parameter)) { + var values = parameters[parameter].join('+'); + refinements.push("".concat(encodeURIComponent(parameter), "=").concat(encodeURIComponent(parameter), "_").concat(encodeURIComponent(values))); + } } - return { - $$type: 'ais.configure', - init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), false); - }, - dispose: function dispose(_ref) { - var state = _ref.state; - unmountFn(); - return getInitialSearchParameters(state, widgetParams); - }, - getRenderState: function getRenderState(renderState, renderOptions) { - var _renderState$configur; + return refinements.join('&'); + }; - var widgetRenderState = this.getWidgetRenderState(renderOptions); - return _objectSpread2(_objectSpread2({}, renderState), {}, { - configure: _objectSpread2(_objectSpread2({}, widgetRenderState), {}, { - widgetParams: _objectSpread2(_objectSpread2({}, widgetRenderState.widgetParams), {}, { - searchParameters: merge$1(new algoliasearchHelper_1.SearchParameters((_renderState$configur = renderState.configure) === null || _renderState$configur === void 0 ? void 0 : _renderState$configur.widgetParams.searchParameters), new algoliasearchHelper_1.SearchParameters(widgetRenderState.widgetParams.searchParameters)).getQueryParams() - }) - }) - }); - }, - getWidgetRenderState: function getWidgetRenderState(_ref2) { - var helper = _ref2.helper; + var serializeNumericRefinements = function serializeNumericRefinements(numericRefinements) { + var refinements = []; - if (!connectorState.refine) { - connectorState.refine = refine(helper); - } + for (var attribute in numericRefinements) { + if (numericRefinements.hasOwnProperty(attribute)) { + var filter = numericRefinements[attribute]; - return { - refine: connectorState.refine, - widgetParams: widgetParams - }; - }, - getWidgetSearchParameters: function getWidgetSearchParameters(state, _ref3) { - var uiState = _ref3.uiState; - return merge$1(state, new algoliasearchHelper_1.SearchParameters(_objectSpread2(_objectSpread2({}, uiState.configure), widgetParams.searchParameters))); - }, - getWidgetUiState: function getWidgetUiState(uiState) { - return _objectSpread2(_objectSpread2({}, uiState), {}, { - configure: _objectSpread2(_objectSpread2({}, uiState.configure), widgetParams.searchParameters) - }); + if (filter.hasOwnProperty('>=') && filter.hasOwnProperty('<=')) { + if (filter['>='] && filter['>='][0] === filter['<='] && filter['<='][0]) { + refinements.push("".concat(attribute, "=").concat(attribute, "_").concat(filter['>='])); + } else { + refinements.push("".concat(attribute, "=").concat(attribute, "_").concat(filter['>='], "to").concat(filter['<='])); + } + } else if (filter.hasOwnProperty('>=')) { + refinements.push("".concat(attribute, "=").concat(attribute, "_from").concat(filter['>='])); + } else if (filter.hasOwnProperty('<=')) { + refinements.push("".concat(attribute, "=").concat(attribute, "_to").concat(filter['<='])); + } else if (filter.hasOwnProperty('=')) { + var equals = []; + + for (var equal in filter['=']) { + // eslint-disable-next-line max-depth + if (filter['='].hasOwnProperty(equal)) { + // @ts-ignore somehow 'equal' is a string, even though it's a number? + equals.push(filter['='][equal]); + } + } + + refinements.push("".concat(attribute, "=").concat(attribute, "_").concat(equals.join('-'))); + } } - }; + } + + return refinements.join('&'); }; - }; - var withUsage$m = createDocumentationMessageGenerator({ - name: 'configure-related-items', - connector: true - }); + var lastSentData = ''; - function createOptionalFilter(_ref) { - var attributeName = _ref.attributeName, - attributeValue = _ref.attributeValue, - attributeScore = _ref.attributeScore; - return "".concat(attributeName, ":").concat(attributeValue, ""); - } + var sendAnalytics = function sendAnalytics(analyticsState) { + if (analyticsState === null) { + return; + } - var connectConfigureRelatedItems = function connectConfigureRelatedItems(renderFn, unmountFn) { - return function (widgetParams) { - var _ref2 = widgetParams || {}, - hit = _ref2.hit, - matchingPatterns = _ref2.matchingPatterns, - _ref2$transformSearch = _ref2.transformSearchParameters, - transformSearchParameters = _ref2$transformSearch === void 0 ? function (x) { - return x; - } : _ref2$transformSearch; + var serializedParams = []; + var serializedRefinements = serializeRefinements(_objectSpread2(_objectSpread2(_objectSpread2({}, analyticsState.state.disjunctiveFacetsRefinements), analyticsState.state.facetsRefinements), analyticsState.state.hierarchicalFacetsRefinements)); + var serializedNumericRefinements = serializeNumericRefinements(analyticsState.state.numericRefinements); - if (!hit) { - throw new Error(withUsage$m('The `hit` option is required.')); + if (serializedRefinements !== '') { + serializedParams.push(serializedRefinements); } - if (!matchingPatterns) { - throw new Error(withUsage$m('The `matchingPatterns` option is required.')); + if (serializedNumericRefinements !== '') { + serializedParams.push(serializedNumericRefinements); } - var optionalFilters = Object.keys(matchingPatterns).reduce(function (acc, attributeName) { - var attribute = matchingPatterns[attributeName]; - var attributeValue = getPropertyByPath(hit, attributeName); - var attributeScore = attribute.score; + var stringifiedParams = serializedParams.join('&'); + var dataToSend = "Query: ".concat(analyticsState.state.query || '', ", ").concat(stringifiedParams); - if (Array.isArray(attributeValue)) { - return [].concat(_toConsumableArray(acc), [attributeValue.map(function (attributeSubValue) { - return createOptionalFilter({ - attributeName: attributeName, - attributeValue: attributeSubValue, - attributeScore: attributeScore - }); - })]); + if (pushPagination === true) { + dataToSend += ", Page: ".concat(analyticsState.state.page || 0); + } + + if (lastSentData !== dataToSend) { + pushFunction(stringifiedParams, analyticsState.state, analyticsState.results); + lastSentData = dataToSend; + } + }; + + var pushTimeout; + var isInitialSearch = true; + + if (pushInitialSearch === true) { + isInitialSearch = false; + } + + var onClick = function onClick() { + sendAnalytics(cachedState); + }; + + var onUnload = function onUnload() { + sendAnalytics(cachedState); + }; + + return { + $$type: 'ais.analytics', + $$widgetType: 'ais.analytics', + init: function init() { + if (triggerOnUIInteraction === true) { + document.addEventListener('click', onClick); + window.addEventListener('beforeunload', onUnload); } + }, + render: function render(_ref2) { + var results = _ref2.results, + state = _ref2.state; - if (typeof attributeValue === 'string') { - return [].concat(_toConsumableArray(acc), [createOptionalFilter({ - attributeName: attributeName, - attributeValue: attributeValue, - attributeScore: attributeScore - })]); + if (isInitialSearch === true) { + isInitialSearch = false; + return; } - _warning(false, "\nThe `matchingPatterns` option returned a value of type ".concat(getObjectType(attributeValue), " for the \"").concat(attributeName, "\" key. This value was not sent to Algolia because `optionalFilters` only supports strings and array of strings.\n\nYou can remove the \"").concat(attributeName, "\" key from the `matchingPatterns` option.\n\nSee https://www.algolia.com/doc/api-reference/api-parameters/optionalFilters/\n ")) ; - return acc; - }, []); + cachedState = { + results: results, + state: state + }; - var searchParameters = _objectSpread2({}, transformSearchParameters(new algoliasearchHelper_1.SearchParameters({ - sumOrFiltersScores: true, - facetFilters: ["objectID:-".concat(hit.objectID)], - optionalFilters: optionalFilters - }))); + if (pushTimeout) { + clearTimeout(pushTimeout); + } - var makeWidget = connectConfigure(renderFn, unmountFn); - return _objectSpread2(_objectSpread2({}, makeWidget({ - searchParameters: searchParameters - })), {}, { - $$type: 'ais.configureRelatedItems' - }); + pushTimeout = window.setTimeout(function () { + return sendAnalytics(cachedState); + }, delay); + }, + dispose: function dispose() { + if (triggerOnUIInteraction === true) { + document.removeEventListener('click', onClick); + window.removeEventListener('beforeunload', onUnload); + } + }, + getRenderState: function getRenderState(renderState, renderOptions) { + return _objectSpread2(_objectSpread2({}, renderState), {}, { + analytics: this.getWidgetRenderState(renderOptions) + }); + }, + getWidgetRenderState: function getWidgetRenderState() { + return { + widgetParams: widgetParams + }; + } }; }; - var withUsage$n = createDocumentationMessageGenerator({ - name: 'autocomplete', - connector: true - }); + function cx() { + for (var _len = arguments.length, cssClasses = new Array(_len), _key = 0; _key < _len; _key++) { + cssClasses[_key] = arguments[_key]; + } - var connectAutocomplete = function connectAutocomplete(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$n()); - return function (widgetParams) { - var _ref = widgetParams || {}, - _ref$escapeHTML = _ref.escapeHTML, - escapeHTML = _ref$escapeHTML === void 0 ? true : _ref$escapeHTML; + return cssClasses.reduce(function (acc, className) { + if (Array.isArray(className)) { + return acc.concat(className); + } - _warning(!widgetParams.indices, "\nThe option `indices` has been removed from the Autocomplete connector.\n\nThe indices to target are now inferred from the widgets tree.\n".concat(Array.isArray(widgetParams.indices) ? "\nAn alternative would be:\n\nconst autocomplete = connectAutocomplete(renderer);\n\nsearch.addWidgets([\n ".concat(widgetParams.indices.map(function (_ref2) { - var value = _ref2.value; - return "index({ indexName: '".concat(value, "' }),"); - }).join('\n '), "\n autocomplete()\n]);\n") : '', "\n ")) ; - var connectorState = {}; - return { - $$type: 'ais.autocomplete', - init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - var renderState = this.getWidgetRenderState(renderOptions); - renderState.indices.forEach(function (_ref3) { - var sendEvent = _ref3.sendEvent, - hits = _ref3.hits; - sendEvent('view', hits); - }); - renderFn(_objectSpread2(_objectSpread2({}, renderState), {}, { - instantSearchInstance: instantSearchInstance - }), false); - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - autocomplete: this.getWidgetRenderState(renderOptions) - }); - }, - getWidgetRenderState: function getWidgetRenderState(_ref4) { - var _this = this; + return acc.concat([className]); + }, []).filter(Boolean).join(' '); + } - var helper = _ref4.helper, - state = _ref4.state, - scopedResults = _ref4.scopedResults, - instantSearchInstance = _ref4.instantSearchInstance; + function prepareTemplates( // can not use = {} here, since the template could have different constraints + defaultTemplates) { + var templates = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; + var allKeys = uniq([].concat(_toConsumableArray(Object.keys(defaultTemplates || {})), _toConsumableArray(Object.keys(templates)))); + return allKeys.reduce(function (config, key) { + var defaultTemplate = defaultTemplates ? defaultTemplates[key] : undefined; + var customTemplate = templates[key]; + var isCustomTemplate = customTemplate !== undefined && customTemplate !== defaultTemplate; + config.templates[key] = isCustomTemplate ? customTemplate // typescript doesn't recognize that this condition asserts customTemplate is defined + : defaultTemplate; + config.useCustomCompileOptions[key] = isCustomTemplate; + return config; + }, { + // eslint-disable-next-line @typescript-eslint/consistent-type-assertions + templates: {}, + // eslint-disable-next-line @typescript-eslint/consistent-type-assertions + useCustomCompileOptions: {} + }); + } + /** + * Prepares an object to be passed to the Template widget + */ + + + function prepareTemplateProps(_ref) { + var defaultTemplates = _ref.defaultTemplates, + templates = _ref.templates, + templatesConfig = _ref.templatesConfig; + var preparedTemplates = prepareTemplates(defaultTemplates, templates); + return _objectSpread2({ + templatesConfig: templatesConfig + }, preparedTemplates); + } + + function createCommonjsModule(fn, module) { + return module = { exports: {} }, fn(module, module.exports), module.exports; + } + + var compiler = createCommonjsModule(function (module, exports) { + /* + * Copyright 2011 Twitter, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + + (function (Hogan) { + // Setup regex assignments + // remove whitespace according to Mustache spec + var rIsWhitespace = /\S/, + rQuot = /\"/g, + rNewline = /\n/g, + rCr = /\r/g, + rSlash = /\\/g, + rLineSep = /\u2028/, + rParagraphSep = /\u2029/; + + Hogan.tags = { + '#': 1, '^': 2, '<': 3, '$': 4, + '/': 5, '!': 6, '>': 7, '=': 8, '_v': 9, + '{': 10, '&': 11, '_t': 12 + }; - if (!connectorState.refine) { - connectorState.refine = function (query) { - helper.setQuery(query).search(); - }; - } + Hogan.scan = function scan(text, delimiters) { + var len = text.length, + IN_TEXT = 0, + IN_TAG_TYPE = 1, + IN_TAG = 2, + state = IN_TEXT, + tagType = null, + tag = null, + buf = '', + tokens = [], + seenTag = false, + i = 0, + lineStart = 0, + otag = '{{', + ctag = '}}'; - var indices = scopedResults.map(function (scopedResult) { - // We need to escape the hits because highlighting - // exposes HTML tags to the end-user. - scopedResult.results.hits = escapeHTML ? escapeHits(scopedResult.results.hits) : scopedResult.results.hits; - var sendEvent = createSendEventForHits({ - instantSearchInstance: instantSearchInstance, - index: scopedResult.results.index, - widgetType: _this.$$type - }); - return { - indexId: scopedResult.indexId, - indexName: scopedResult.results.index, - hits: scopedResult.results.hits, - results: scopedResult.results, - sendEvent: sendEvent - }; - }); - return { - currentRefinement: state.query || '', - indices: indices, - refine: connectorState.refine, - widgetParams: widgetParams - }; - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref5) { - var searchParameters = _ref5.searchParameters; - var query = searchParameters.query || ''; + function addBuf() { + if (buf.length > 0) { + tokens.push({tag: '_t', text: new String(buf)}); + buf = ''; + } + } - if (query === '' || uiState && uiState.query === query) { - return uiState; + function lineIsWhitespace() { + var isAllWhitespace = true; + for (var j = lineStart; j < tokens.length; j++) { + isAllWhitespace = + (Hogan.tags[tokens[j].tag] < Hogan.tags['_v']) || + (tokens[j].tag == '_t' && tokens[j].text.match(rIsWhitespace) === null); + if (!isAllWhitespace) { + return false; } + } - return _objectSpread2(_objectSpread2({}, uiState), {}, { - query: query - }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref6) { - var uiState = _ref6.uiState; - var parameters = { - query: uiState.query || '' - }; + return isAllWhitespace; + } - if (!escapeHTML) { - return searchParameters.setQueryParameters(parameters); + function filterLine(haveSeenTag, noNewLine) { + addBuf(); + + if (haveSeenTag && lineIsWhitespace()) { + for (var j = lineStart, next; j < tokens.length; j++) { + if (tokens[j].text) { + if ((next = tokens[j+1]) && next.tag == '>') { + // set indent to token value + next.indent = tokens[j].text.toString(); + } + tokens.splice(j, 1); + } } + } else if (!noNewLine) { + tokens.push({tag:'\n'}); + } - return searchParameters.setQueryParameters(_objectSpread2(_objectSpread2({}, parameters), TAG_PLACEHOLDER)); - }, - dispose: function dispose(_ref7) { - var state = _ref7.state; - unmountFn(); - var stateWithoutQuery = state.setQueryParameter('query', undefined); + seenTag = false; + lineStart = tokens.length; + } - if (!escapeHTML) { - return stateWithoutQuery; - } + function changeDelimiters(text, index) { + var close = '=' + ctag, + closeIndex = text.indexOf(close, index), + delimiters = trim( + text.substring(text.indexOf('=', index) + 1, closeIndex) + ).split(' '); - return stateWithoutQuery.setQueryParameters(Object.keys(TAG_PLACEHOLDER).reduce(function (acc, key) { - return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, key, undefined)); - }, {})); - } - }; - }; - }; + otag = delimiters[0]; + ctag = delimiters[delimiters.length - 1]; - var withUsage$o = createDocumentationMessageGenerator({ - name: 'query-rules', - connector: true - }); + return closeIndex + close.length - 1; + } - function hasStateRefinements(state) { - return [state.disjunctiveFacetsRefinements, state.facetsRefinements, state.hierarchicalFacetsRefinements, state.numericRefinements].some(function (refinement) { - return Boolean(refinement && Object.keys(refinement).length > 0); - }); - } // A context rule must consist only of alphanumeric characters, hyphens, and underscores. - // See https://www.algolia.com/doc/guides/managing-results/refine-results/merchandising-and-promoting/in-depth/implementing-query-rules/#context + if (delimiters) { + delimiters = delimiters.split(' '); + otag = delimiters[0]; + ctag = delimiters[1]; + } + for (i = 0; i < len; i++) { + if (state == IN_TEXT) { + if (tagChange(otag, text, i)) { + --i; + addBuf(); + state = IN_TAG_TYPE; + } else { + if (text.charAt(i) == '\n') { + filterLine(seenTag); + } else { + buf += text.charAt(i); + } + } + } else if (state == IN_TAG_TYPE) { + i += otag.length - 1; + tag = Hogan.tags[text.charAt(i + 1)]; + tagType = tag ? text.charAt(i + 1) : '_v'; + if (tagType == '=') { + i = changeDelimiters(text, i); + state = IN_TEXT; + } else { + if (tag) { + i++; + } + state = IN_TAG; + } + seenTag = i; + } else { + if (tagChange(ctag, text, i)) { + tokens.push({tag: tagType, n: trim(buf), otag: otag, ctag: ctag, + i: (tagType == '/') ? seenTag - otag.length : i + ctag.length}); + buf = ''; + i += ctag.length - 1; + state = IN_TEXT; + if (tagType == '{') { + if (ctag == '}}') { + i++; + } else { + cleanTripleStache(tokens[tokens.length - 1]); + } + } + } else { + buf += text.charAt(i); + } + } + } - function escapeRuleContext(ruleName) { - return ruleName.replace(/[^a-z0-9-_]+/gi, '_'); - } + filterLine(seenTag, true); - function getRuleContextsFromTrackedFilters(_ref) { - var helper = _ref.helper, - sharedHelperState = _ref.sharedHelperState, - trackedFilters = _ref.trackedFilters; - var ruleContexts = Object.keys(trackedFilters).reduce(function (facets, facetName) { - var facetRefinements = getRefinements(helper.lastResults || {}, sharedHelperState, true).filter(function (refinement) { - return refinement.attribute === facetName; - }).map(function (refinement) { - return refinement.numericValue || refinement.name; - }); - var getTrackedFacetValues = trackedFilters[facetName]; - var trackedFacetValues = getTrackedFacetValues(facetRefinements); - return [].concat(_toConsumableArray(facets), _toConsumableArray(facetRefinements.filter(function (facetRefinement) { - return trackedFacetValues.includes(facetRefinement); - }).map(function (facetValue) { - return escapeRuleContext("ais-".concat(facetName, "-").concat(facetValue)); - }))); - }, []); - return ruleContexts; - } + return tokens; + }; - function applyRuleContexts(event) { - var helper = this.helper, - initialRuleContexts = this.initialRuleContexts, - trackedFilters = this.trackedFilters, - transformRuleContexts = this.transformRuleContexts; - var sharedHelperState = event.state; - var previousRuleContexts = sharedHelperState.ruleContexts || []; - var newRuleContexts = getRuleContextsFromTrackedFilters({ - helper: helper, - sharedHelperState: sharedHelperState, - trackedFilters: trackedFilters - }); - var nextRuleContexts = [].concat(_toConsumableArray(initialRuleContexts), _toConsumableArray(newRuleContexts)); - _warning(nextRuleContexts.length <= 10, "\nThe maximum number of `ruleContexts` is 10. They have been sliced to that limit.\nConsider using `transformRuleContexts` to minimize the number of rules sent to Algolia.\n") ; - var ruleContexts = transformRuleContexts(nextRuleContexts).slice(0, 10); + function cleanTripleStache(token) { + if (token.n.substr(token.n.length - 1) === '}') { + token.n = token.n.substring(0, token.n.length - 1); + } + } - if (!isEqual(previousRuleContexts, ruleContexts)) { - helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread2(_objectSpread2({}, sharedHelperState), {}, { - ruleContexts: ruleContexts - })); + function trim(s) { + if (s.trim) { + return s.trim(); + } + + return s.replace(/^\s*|\s*$/g, ''); } - } - var connectQueryRules = function connectQueryRules(_render) { - var unmount = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(_render, withUsage$o()); - return function (widgetParams) { - var _ref2 = widgetParams || {}, - _ref2$trackedFilters = _ref2.trackedFilters, - trackedFilters = _ref2$trackedFilters === void 0 ? {} : _ref2$trackedFilters, - _ref2$transformRuleCo = _ref2.transformRuleContexts, - transformRuleContexts = _ref2$transformRuleCo === void 0 ? function (rules) { - return rules; - } : _ref2$transformRuleCo, - _ref2$transformItems = _ref2.transformItems, - transformItems = _ref2$transformItems === void 0 ? function (items) { - return items; - } : _ref2$transformItems; + function tagChange(tag, text, index) { + if (text.charAt(index) != tag.charAt(0)) { + return false; + } - Object.keys(trackedFilters).forEach(function (facetName) { - if (typeof trackedFilters[facetName] !== 'function') { - throw new Error(withUsage$o("'The \"".concat(facetName, "\" filter value in the `trackedFilters` option expects a function."))); + for (var i = 1, l = tag.length; i < l; i++) { + if (text.charAt(index + i) != tag.charAt(i)) { + return false; } - }); - var hasTrackedFilters = Object.keys(trackedFilters).length > 0; // We store the initial rule contexts applied before creating the widget - // so that we do not override them with the rules created from `trackedFilters`. + } - var initialRuleContexts = []; - var onHelperChange; - return { - $$type: 'ais.queryRules', - init: function init(initOptions) { - var helper = initOptions.helper, - state = initOptions.state, - instantSearchInstance = initOptions.instantSearchInstance; - initialRuleContexts = state.ruleContexts || []; - onHelperChange = applyRuleContexts.bind({ - helper: helper, - initialRuleContexts: initialRuleContexts, - trackedFilters: trackedFilters, - transformRuleContexts: transformRuleContexts - }); + return true; + } - if (hasTrackedFilters) { - // We need to apply the `ruleContexts` based on the `trackedFilters` - // before the helper changes state in some cases: - // - Some filters are applied on the first load (e.g. using `configure`) - // - The `transformRuleContexts` option sets initial `ruleContexts`. - if (hasStateRefinements(state) || Boolean(widgetParams.transformRuleContexts)) { - onHelperChange({ - state: state - }); - } // We track every change in the helper to override its state and add - // any `ruleContexts` needed based on the `trackedFilters`. + // the tags allowed inside super templates + var allowedInSuper = {'_t': true, '\n': true, '$': true, '/': true}; + function buildTree(tokens, kind, stack, customTags) { + var instructions = [], + opener = null, + tail = null, + token = null; - helper.on('change', onHelperChange); - } + tail = stack[stack.length - 1]; - _render(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; + while (tokens.length > 0) { + token = tokens.shift(); - _render(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), false); - }, - getWidgetRenderState: function getWidgetRenderState(_ref3) { - var results = _ref3.results; + if (tail && tail.tag == '<' && !(token.tag in allowedInSuper)) { + throw new Error('Illegal content in < super tag.'); + } - var _ref4 = results || {}, - _ref4$userData = _ref4.userData, - userData = _ref4$userData === void 0 ? [] : _ref4$userData; + if (Hogan.tags[token.tag] <= Hogan.tags['$'] || isOpener(token, customTags)) { + stack.push(token); + token.nodes = buildTree(tokens, token.tag, stack, customTags); + } else if (token.tag == '/') { + if (stack.length === 0) { + throw new Error('Closing tag without opener: /' + token.n); + } + opener = stack.pop(); + if (token.n != opener.n && !isCloser(token.n, opener.n, customTags)) { + throw new Error('Nesting error: ' + opener.n + ' vs. ' + token.n); + } + opener.end = token.i; + return instructions; + } else if (token.tag == '\n') { + token.last = (tokens.length == 0) || (tokens[0].tag == '\n'); + } - var items = transformItems(userData, { - results: results - }); - return { - items: items, - widgetParams: widgetParams - }; - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - queryRules: this.getWidgetRenderState(renderOptions) - }); - }, - dispose: function dispose(_ref5) { - var helper = _ref5.helper, - state = _ref5.state; - unmount(); + instructions.push(token); + } - if (hasTrackedFilters) { - helper.removeListener('change', onHelperChange); - return state.setQueryParameter('ruleContexts', initialRuleContexts); - } + if (stack.length > 0) { + throw new Error('missing closing tag: ' + stack.pop().n); + } - return state; + return instructions; + } + + function isOpener(token, tags) { + for (var i = 0, l = tags.length; i < l; i++) { + if (tags[i].o == token.n) { + token.tag = '#'; + return true; } - }; + } + } + + function isCloser(close, open, tags) { + for (var i = 0, l = tags.length; i < l; i++) { + if (tags[i].c == close && tags[i].o == open) { + return true; + } + } + } + + function stringifySubstitutions(obj) { + var items = []; + for (var key in obj) { + items.push('"' + esc(key) + '": function(c,p,t,i) {' + obj[key] + '}'); + } + return "{ " + items.join(",") + " }"; + } + + function stringifyPartials(codeObj) { + var partials = []; + for (var key in codeObj.partials) { + partials.push('"' + esc(key) + '":{name:"' + esc(codeObj.partials[key].name) + '", ' + stringifyPartials(codeObj.partials[key]) + "}"); + } + return "partials: {" + partials.join(",") + "}, subs: " + stringifySubstitutions(codeObj.subs); + } + + Hogan.stringify = function(codeObj, text, options) { + return "{code: function (c,p,i) { " + Hogan.wrapMain(codeObj.code) + " }," + stringifyPartials(codeObj) + "}"; }; - }; - // `SpeechRecognition` is an API used on the browser so we can safely disable - // the `window` check. + var serialNo = 0; + Hogan.generate = function(tree, text, options) { + serialNo = 0; + var context = { code: '', subs: {}, partials: {} }; + Hogan.walk(tree, context); - /* eslint-disable no-restricted-globals */ + if (options.asString) { + return this.stringify(context, text, options); + } - /* global SpeechRecognition SpeechRecognitionEvent */ - var createVoiceSearchHelper = function createVoiceSearchHelper(_ref) { - var searchAsYouSpeak = _ref.searchAsYouSpeak, - language = _ref.language, - onQueryChange = _ref.onQueryChange, - onStateChange = _ref.onStateChange; - var SpeechRecognitionAPI = window.webkitSpeechRecognition || window.SpeechRecognition; + return this.makeTemplate(context, text, options); + }; - var getDefaultState = function getDefaultState(status) { - return { - status: status, - transcript: '', - isSpeechFinal: false, - errorCode: undefined - }; + Hogan.wrapMain = function(code) { + return 'var t=this;t.b(i=i||"");' + code + 'return t.fl();'; }; - var state = getDefaultState('initial'); - var recognition; + Hogan.template = Hogan.Template; - var isBrowserSupported = function isBrowserSupported() { - return Boolean(SpeechRecognitionAPI); + Hogan.makeTemplate = function(codeObj, text, options) { + var template = this.makePartials(codeObj); + template.code = new Function('c', 'p', 'i', this.wrapMain(codeObj.code)); + return new this.template(template, text, this, options); }; - var isListening = function isListening() { - return state.status === 'askingPermission' || state.status === 'waiting' || state.status === 'recognizing'; + Hogan.makePartials = function(codeObj) { + var key, template = {subs: {}, partials: codeObj.partials, name: codeObj.name}; + for (key in template.partials) { + template.partials[key] = this.makePartials(template.partials[key]); + } + for (key in codeObj.subs) { + template.subs[key] = new Function('c', 'p', 't', 'i', codeObj.subs[key]); + } + return template; }; - var setState = function setState() { - var newState = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; - state = _objectSpread2(_objectSpread2({}, state), newState); - onStateChange(); - }; + function esc(s) { + return s.replace(rSlash, '\\\\') + .replace(rQuot, '\\\"') + .replace(rNewline, '\\n') + .replace(rCr, '\\r') + .replace(rLineSep, '\\u2028') + .replace(rParagraphSep, '\\u2029'); + } + + function chooseMethod(s) { + return (~s.indexOf('.')) ? 'd' : 'f'; + } + + function createPartial(node, context) { + var prefix = "<" + (context.prefix || ""); + var sym = prefix + node.n + serialNo++; + context.partials[sym] = {name: node.n, partials: {}}; + context.code += 't.b(t.rp("' + esc(sym) + '",c,p,"' + (node.indent || '') + '"));'; + return sym; + } + + Hogan.codegen = { + '#': function(node, context) { + context.code += 'if(t.s(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,1),' + + 'c,p,0,' + node.i + ',' + node.end + ',"' + node.otag + " " + node.ctag + '")){' + + 't.rs(c,p,' + 'function(c,p,t){'; + Hogan.walk(node.nodes, context); + context.code += '});c.pop();}'; + }, + + '^': function(node, context) { + context.code += 'if(!t.s(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,1),c,p,1,0,0,"")){'; + Hogan.walk(node.nodes, context); + context.code += '};'; + }, + + '>': createPartial, + '<': function(node, context) { + var ctx = {partials: {}, code: '', subs: {}, inPartial: true}; + Hogan.walk(node.nodes, ctx); + var template = context.partials[createPartial(node, context)]; + template.subs = ctx.subs; + template.partials = ctx.partials; + }, + + '$': function(node, context) { + var ctx = {subs: {}, code: '', partials: context.partials, prefix: node.n}; + Hogan.walk(node.nodes, ctx); + context.subs[node.n] = ctx.code; + if (!context.inPartial) { + context.code += 't.sub("' + esc(node.n) + '",c,p,i);'; + } + }, + + '\n': function(node, context) { + context.code += write('"\\n"' + (node.last ? '' : ' + i')); + }, - var getState = function getState() { - return state; - }; + '_v': function(node, context) { + context.code += 't.b(t.v(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,0)));'; + }, - var resetState = function resetState() { - var status = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'initial'; - setState(getDefaultState(status)); - }; + '_t': function(node, context) { + context.code += write('"' + esc(node.text) + '"'); + }, - var onStart = function onStart() { - setState({ - status: 'waiting' - }); - }; + '{': tripleStache, - var onError = function onError(event) { - setState({ - status: 'error', - errorCode: event.error - }); + '&': tripleStache }; - var onResult = function onResult(event) { - setState({ - status: 'recognizing', - transcript: event.results[0] && event.results[0][0] && event.results[0][0].transcript || '', - isSpeechFinal: event.results[0] && event.results[0].isFinal - }); - - if (searchAsYouSpeak && state.transcript) { - onQueryChange(state.transcript); - } - }; + function tripleStache(node, context) { + context.code += 't.b(t.t(t.' + chooseMethod(node.n) + '("' + esc(node.n) + '",c,p,0)));'; + } - var onEnd = function onEnd() { - if (!state.errorCode && state.transcript && !searchAsYouSpeak) { - onQueryChange(state.transcript); - } + function write(s) { + return 't.b(' + s + ');'; + } - if (state.status !== 'error') { - setState({ - status: 'finished' - }); + Hogan.walk = function(nodelist, context) { + var func; + for (var i = 0, l = nodelist.length; i < l; i++) { + func = Hogan.codegen[nodelist[i].tag]; + func && func(nodelist[i], context); } + return context; }; - var startListening = function startListening() { - recognition = new SpeechRecognitionAPI(); - - if (!recognition) { - return; - } - - resetState('askingPermission'); - recognition.interimResults = true; + Hogan.parse = function(tokens, text, options) { + options = options || {}; + return buildTree(tokens, '', [], options.sectionTags || []); + }; - if (language) { - recognition.lang = language; - } + Hogan.cache = {}; - recognition.addEventListener('start', onStart); - recognition.addEventListener('error', onError); - recognition.addEventListener('result', onResult); - recognition.addEventListener('end', onEnd); - recognition.start(); + Hogan.cacheKey = function(text, options) { + return [text, !!options.asString, !!options.disableLambda, options.delimiters, !!options.modelGet].join('||'); }; - var dispose = function dispose() { - if (!recognition) { - return; + Hogan.compile = function(text, options) { + options = options || {}; + var key = Hogan.cacheKey(text, options); + var template = this.cache[key]; + + if (template) { + var partials = template.partials; + for (var name in partials) { + delete partials[name].instance; + } + return template; } - recognition.stop(); - recognition.removeEventListener('start', onStart); - recognition.removeEventListener('error', onError); - recognition.removeEventListener('result', onResult); - recognition.removeEventListener('end', onEnd); - recognition = undefined; + template = this.generate(this.parse(this.scan(text, options.delimiters), text, options), text, options); + return this.cache[key] = template; }; + })( exports ); + }); - var stopListening = function stopListening() { - dispose(); // Because `dispose` removes event listeners, `end` listener is not called. - // So we're setting the `status` as `finished` here. - // If we don't do it, it will be still `waiting` or `recognizing`. - - resetState('finished'); - }; + var template = createCommonjsModule(function (module, exports) { - return { - getState: getState, - isBrowserSupported: isBrowserSupported, - isListening: isListening, - startListening: startListening, - stopListening: stopListening, - dispose: dispose + (function (Hogan) { + Hogan.Template = function (codeObj, text, compiler, options) { + codeObj = codeObj || {}; + this.r = codeObj.code || this.r; + this.c = compiler; + this.options = options || {}; + this.text = text || ''; + this.partials = codeObj.partials || {}; + this.subs = codeObj.subs || {}; + this.buf = ''; }; - }; - - var withUsage$p = createDocumentationMessageGenerator({ - name: 'voice-search', - connector: true - }); - var connectVoiceSearch = function connectVoiceSearch(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$p()); - return function (widgetParams) { - var _widgetParams$searchA = widgetParams.searchAsYouSpeak, - searchAsYouSpeak = _widgetParams$searchA === void 0 ? false : _widgetParams$searchA, - language = widgetParams.language, - additionalQueryParameters = widgetParams.additionalQueryParameters, - _widgetParams$createV = widgetParams.createVoiceSearchHelper, - createVoiceSearchHelper$1 = _widgetParams$createV === void 0 ? createVoiceSearchHelper : _widgetParams$createV; - return { - $$type: 'ais.voiceSearch', - init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), false); - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - voiceSearch: this.getWidgetRenderState(renderOptions) - }); - }, - getWidgetRenderState: function getWidgetRenderState(renderOptions) { - var _this = this; + Hogan.Template.prototype = { + // render: replaced by generated code. + r: function (context, partials, indent) { return ''; }, - var helper = renderOptions.helper, - instantSearchInstance = renderOptions.instantSearchInstance; + // variable escaping + v: hoganEscape, - if (!this._refine) { - this._refine = function (query) { - if (query !== helper.state.query) { - var queryLanguages = language ? [language.split('-')[0]] : undefined; - helper.setQueryParameter('queryLanguages', queryLanguages); + // triple stache + t: coerceToString, - if (typeof additionalQueryParameters === 'function') { - helper.setState(helper.state.setQueryParameters(_objectSpread2({ - ignorePlurals: true, - removeStopWords: true, - // @ts-ignore (optionalWords only allows array in v3, while string is also valid) - optionalWords: query - }, additionalQueryParameters({ - query: query - })))); - } + render: function render(context, partials, indent) { + return this.ri([context], partials || {}, indent); + }, - helper.setQuery(query).search(); - } - }; - } + // render internal -- a hook for overrides that catches partials too + ri: function (context, partials, indent) { + return this.r(context, partials, indent); + }, - if (!this._voiceSearchHelper) { - this._voiceSearchHelper = createVoiceSearchHelper$1({ - searchAsYouSpeak: searchAsYouSpeak, - language: language, - onQueryChange: function onQueryChange(query) { - return _this._refine(query); - }, - onStateChange: function onStateChange() { - renderFn(_objectSpread2(_objectSpread2({}, _this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), false); - } - }); - } + // ensurePartial + ep: function(symbol, partials) { + var partial = this.partials[symbol]; - var _voiceSearchHelper = this._voiceSearchHelper, - isBrowserSupported = _voiceSearchHelper.isBrowserSupported, - isListening = _voiceSearchHelper.isListening, - startListening = _voiceSearchHelper.startListening, - stopListening = _voiceSearchHelper.stopListening, - getState = _voiceSearchHelper.getState; - return { - isBrowserSupported: isBrowserSupported(), - isListening: isListening(), - toggleListening: function toggleListening() { - if (!isBrowserSupported()) { - return; - } + // check to see that if we've instantiated this partial before + var template = partials[partial.name]; + if (partial.instance && partial.base == template) { + return partial.instance; + } - if (isListening()) { - stopListening(); - } else { - startListening(); - } - }, - voiceListeningState: getState(), - widgetParams: widgetParams - }; - }, - dispose: function dispose(_ref) { - var state = _ref.state; + if (typeof template == 'string') { + if (!this.c) { + throw new Error("No compiler available."); + } + template = this.c.compile(template, this.options); + } - this._voiceSearchHelper.dispose(); + if (!template) { + return null; + } - unmountFn(); - var newState = state; + // We use this to check whether the partials dictionary has changed + this.partials[symbol].base = template; - if (typeof additionalQueryParameters === 'function') { - var additional = additionalQueryParameters({ - query: '' - }); - var toReset = additional ? Object.keys(additional).reduce(function (acc, current) { - // @ts-ignore search parameters is typed as readonly in v4 - acc[current] = undefined; - return acc; - }, {}) : {}; - newState = state.setQueryParameters(_objectSpread2({ - // @ts-ignore (queryLanguages is not added to algoliasearch v3) - queryLanguages: undefined, - ignorePlurals: undefined, - removeStopWords: undefined, - optionalWords: undefined - }, toReset)); + if (partial.subs) { + // Make sure we consider parent template now + if (!partials.stackText) partials.stackText = {}; + for (key in partial.subs) { + if (!partials.stackText[key]) { + partials.stackText[key] = (this.activeSub !== undefined && partials.stackText[this.activeSub]) ? partials.stackText[this.activeSub] : this.text; + } } + template = createSpecializedPartial(template, partial.subs, partial.partials, + this.stackSubs, this.stackPartials, partials.stackText); + } + this.partials[symbol].instance = template; - return newState.setQueryParameter('query', undefined); - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref2) { - var searchParameters = _ref2.searchParameters; - var query = searchParameters.query || ''; + return template; + }, - if (!query) { - return uiState; - } + // tries to find a partial in the current scope and render it + rp: function(symbol, context, partials, indent) { + var partial = this.ep(symbol, partials); + if (!partial) { + return ''; + } - return _objectSpread2(_objectSpread2({}, uiState), {}, { - query: query - }); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(searchParameters, _ref3) { - var uiState = _ref3.uiState; - return searchParameters.setQueryParameter('query', uiState.query || ''); + return partial.ri(context, partials, indent); + }, + + // render a section + rs: function(context, partials, section) { + var tail = context[context.length - 1]; + + if (!isArray(tail)) { + section(context, partials, this); + return; } - }; - }; - }; - function hasFindAnswersMethod(answersIndex) { - return typeof answersIndex.findAnswers === 'function'; - } + for (var i = 0; i < tail.length; i++) { + context.push(tail[i]); + section(context, partials, this); + context.pop(); + } + }, - var withUsage$q = createDocumentationMessageGenerator({ - name: 'answers', - connector: true - }); + // maybe start a section + s: function(val, ctx, partials, inverted, start, end, tags) { + var pass; - var connectAnswers = function connectAnswers(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$q()); - return function (widgetParams) { - var _ref = widgetParams || {}, - queryLanguages = _ref.queryLanguages, - attributesForPrediction = _ref.attributesForPrediction, - _ref$nbHits = _ref.nbHits, - nbHits = _ref$nbHits === void 0 ? 1 : _ref$nbHits, - _ref$renderDebounceTi = _ref.renderDebounceTime, - renderDebounceTime = _ref$renderDebounceTi === void 0 ? 100 : _ref$renderDebounceTi, - _ref$searchDebounceTi = _ref.searchDebounceTime, - searchDebounceTime = _ref$searchDebounceTi === void 0 ? 100 : _ref$searchDebounceTi, - _ref$escapeHTML = _ref.escapeHTML, - escapeHTML = _ref$escapeHTML === void 0 ? true : _ref$escapeHTML, - _ref$extraParameters = _ref.extraParameters, - extraParameters = _ref$extraParameters === void 0 ? {} : _ref$extraParameters; // @ts-expect-error checking for the wrong value + if (isArray(val) && val.length === 0) { + return false; + } + if (typeof val == 'function') { + val = this.ms(val, ctx, partials, inverted, start, end, tags); + } - if (!queryLanguages || queryLanguages.length === 0) { - throw new Error(withUsage$q('The `queryLanguages` expects an array of strings.')); - } + pass = !!val; - var runConcurrentSafePromise = createConcurrentSafePromise(); - var lastHits = []; - var isLoading = false; - var debouncedRender = debounce(renderFn, renderDebounceTime); // this does not directly use DebouncedFunction, since then the generic will disappear + if (!inverted && pass && ctx) { + ctx.push((typeof val == 'object') ? val : ctx[ctx.length - 1]); + } - var debouncedRefine; - return { - $$type: 'ais.answers', - init: function init(initOptions) { - var state = initOptions.state, - instantSearchInstance = initOptions.instantSearchInstance; - var answersIndex = instantSearchInstance.client.initIndex(state.index); + return pass; + }, - if (!hasFindAnswersMethod(answersIndex)) { - throw new Error(withUsage$q('`algoliasearch` >= 4.8.0 required.')); + // find values with dotted names + d: function(key, ctx, partials, returnFound) { + var found, + names = key.split('.'), + val = this.f(names[0], ctx, partials, returnFound), + doModelGet = this.options.modelGet, + cx = null; + + if (key === '.' && isArray(ctx[ctx.length - 2])) { + val = ctx[ctx.length - 1]; + } else { + for (var i = 1; i < names.length; i++) { + found = findInScope(names[i], val, doModelGet); + if (found !== undefined) { + cx = val; + val = found; + } else { + val = ''; + } } + } - debouncedRefine = debounce(answersIndex.findAnswers, searchDebounceTime); - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: initOptions.instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var _this = this; + if (returnFound && !val) { + return false; + } - var query = renderOptions.state.query; + if (!returnFound && typeof val == 'function') { + ctx.push(cx); + val = this.mv(val, ctx, partials); + ctx.pop(); + } - if (!query) { - // renders nothing with empty query - lastHits = []; - isLoading = false; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: renderOptions.instantSearchInstance - }), false); - return; - } // render the loader + return val; + }, + // find values with normal names + f: function(key, ctx, partials, returnFound) { + var val = false, + v = null, + found = false, + doModelGet = this.options.modelGet; - lastHits = []; - isLoading = true; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: renderOptions.instantSearchInstance - }), false); // call /answers API + for (var i = ctx.length - 1; i >= 0; i--) { + v = ctx[i]; + val = findInScope(key, v, doModelGet); + if (val !== undefined) { + found = true; + break; + } + } - runConcurrentSafePromise(debouncedRefine(query, queryLanguages, _objectSpread2(_objectSpread2({}, extraParameters), {}, { - nbHits: nbHits, - attributesForPrediction: attributesForPrediction - }))).then(function (result) { - if (!result) { - // It's undefined when it's debounced. - return; - } + if (!found) { + return (returnFound) ? false : ""; + } - if (escapeHTML && result.hits.length > 0) { - result.hits = escapeHits(result.hits); - } + if (!returnFound && typeof val == 'function') { + val = this.mv(val, ctx, partials); + } - var hitsWithAbsolutePosition = addAbsolutePosition(result.hits, 0, nbHits); - var hitsWithAbsolutePositionAndQueryID = addQueryID(hitsWithAbsolutePosition, result.queryID); - lastHits = hitsWithAbsolutePositionAndQueryID; - isLoading = false; - debouncedRender(_objectSpread2(_objectSpread2({}, _this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: renderOptions.instantSearchInstance - }), false); - }); - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - answers: this.getWidgetRenderState(renderOptions) - }); - }, - getWidgetRenderState: function getWidgetRenderState() { - return { - hits: lastHits, - isLoading: isLoading, - widgetParams: widgetParams - }; - }, - dispose: function dispose(_ref2) { - var state = _ref2.state; - unmountFn(); - return state; - }, - getWidgetSearchParameters: function getWidgetSearchParameters(state) { - return state; + return val; + }, + + // higher order templates + ls: function(func, cx, partials, text, tags) { + var oldTags = this.options.delimiters; + + this.options.delimiters = tags; + this.b(this.ct(coerceToString(func.call(cx, text)), cx, partials)); + this.options.delimiters = oldTags; + + return false; + }, + + // compile text + ct: function(text, cx, partials) { + if (this.options.disableLambda) { + throw new Error('Lambda features disabled.'); } - }; - }; - }; + return this.c.compile(text, this.options).render(cx, partials); + }, - var connectRelevantSort = function connectRelevantSort() { - var renderFn = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : noop; - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - return function (widgetParams) { - var connectorState = {}; - return { - $$type: 'ais.relevantSort', - init: function init(initOptions) { - var instantSearchInstance = initOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var instantSearchInstance = renderOptions.instantSearchInstance; - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(renderOptions)), {}, { - instantSearchInstance: instantSearchInstance - }), false); - }, - dispose: function dispose(_ref) { - var state = _ref.state; - unmountFn(); - return state.setQueryParameter('relevancyStrictness', undefined); - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - relevantSort: this.getWidgetRenderState(renderOptions) - }); - }, - getWidgetRenderState: function getWidgetRenderState(_ref2) { - var results = _ref2.results, - helper = _ref2.helper; + // template result buffering + b: function(s) { this.buf += s; }, - if (!connectorState.refine) { - connectorState.refine = function (relevancyStrictness) { - helper.setQueryParameter('relevancyStrictness', relevancyStrictness).search(); - }; + fl: function() { var r = this.buf; this.buf = ''; return r; }, + + // method replace section + ms: function(func, ctx, partials, inverted, start, end, tags) { + var textSource, + cx = ctx[ctx.length - 1], + result = func.call(cx); + + if (typeof result == 'function') { + if (inverted) { + return true; + } else { + textSource = (this.activeSub && this.subsText && this.subsText[this.activeSub]) ? this.subsText[this.activeSub] : this.text; + return this.ls(result, cx, partials, textSource.substring(start, end), tags); } + } - var _ref3 = results || {}, - appliedRelevancyStrictness = _ref3.appliedRelevancyStrictness; + return result; + }, - var isVirtualReplica = appliedRelevancyStrictness !== undefined; - return { - isRelevantSorted: typeof appliedRelevancyStrictness !== 'undefined' && appliedRelevancyStrictness > 0, - isVirtualReplica: isVirtualReplica, - canRefine: isVirtualReplica, - refine: connectorState.refine, - widgetParams: widgetParams - }; - }, - getWidgetSearchParameters: function getWidgetSearchParameters(state, _ref4) { - var _uiState$relevantSort; + // method replace variable + mv: function(func, ctx, partials) { + var cx = ctx[ctx.length - 1]; + var result = func.call(cx); - var uiState = _ref4.uiState; - return state.setQueryParameter('relevancyStrictness', (_uiState$relevantSort = uiState.relevantSort) !== null && _uiState$relevantSort !== void 0 ? _uiState$relevantSort : state.relevancyStrictness); - }, - getWidgetUiState: function getWidgetUiState(uiState, _ref5) { - var searchParameters = _ref5.searchParameters; - return _objectSpread2(_objectSpread2({}, uiState), {}, { - relevantSort: searchParameters.relevancyStrictness || uiState.relevantSort - }); + if (typeof result == 'function') { + return this.ct(coerceToString(result.call(cx)), cx, partials); } - }; - }; - }; - - var withUsage$r = createDocumentationMessageGenerator({ - name: 'dynamic-widgets', - connector: true - }); - var MAX_WILDCARD_FACETS = 20; - var connectDynamicWidgets = function connectDynamicWidgets(renderFn) { - var unmountFn = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : noop; - checkRendering(renderFn, withUsage$r()); - return function (widgetParams) { - var widgets = widgetParams.widgets, - _widgetParams$maxValu = widgetParams.maxValuesPerFacet, - maxValuesPerFacet = _widgetParams$maxValu === void 0 ? 20 : _widgetParams$maxValu, - _widgetParams$facets = widgetParams.facets, - facets = _widgetParams$facets === void 0 ? ['*'] : _widgetParams$facets, - _widgetParams$transfo = widgetParams.transformItems, - transformItems = _widgetParams$transfo === void 0 ? function (items) { - return items; - } : _widgetParams$transfo, - fallbackWidget = widgetParams.fallbackWidget; + return result; + }, - if (!(widgets && Array.isArray(widgets) && widgets.every(function (widget) { - return _typeof(widget) === 'object'; - }))) { - throw new Error(withUsage$r('The `widgets` option expects an array of widgets.')); + sub: function(name, context, partials, indent) { + var f = this.subs[name]; + if (f) { + this.activeSub = name; + f(context, partials, this, indent); + this.activeSub = false; + } } - if (!(Array.isArray(facets) && facets.length <= 1 && (facets[0] === '*' || facets[0] === undefined))) { - throw new Error(withUsage$r("The `facets` option only accepts [] or [\"*\"], you passed ".concat(JSON.stringify(facets)))); - } + }; - var localWidgets = new Map(); - return { - $$type: 'ais.dynamicWidgets', - init: function init(initOptions) { - widgets.forEach(function (widget) { - var attribute = getWidgetAttribute(widget, initOptions); - localWidgets.set(attribute, { - widget: widget, - isMounted: false - }); - }); - renderFn(_objectSpread2(_objectSpread2({}, this.getWidgetRenderState(initOptions)), {}, { - instantSearchInstance: initOptions.instantSearchInstance - }), true); - }, - render: function render(renderOptions) { - var parent = renderOptions.parent; - var renderState = this.getWidgetRenderState(renderOptions); - var widgetsToUnmount = []; - var widgetsToMount = []; + //Find a key in an object + function findInScope(key, scope, doModelGet) { + var val; - if (fallbackWidget) { - renderState.attributesToRender.forEach(function (attribute) { - if (!localWidgets.has(attribute)) { - var widget = fallbackWidget({ - attribute: attribute - }); - localWidgets.set(attribute, { - widget: widget, - isMounted: false - }); - } - }); - } + if (scope && typeof scope == 'object') { - localWidgets.forEach(function (_ref, attribute) { - var widget = _ref.widget, - isMounted = _ref.isMounted; - var shouldMount = renderState.attributesToRender.indexOf(attribute) > -1; + if (scope[key] !== undefined) { + val = scope[key]; - if (!isMounted && shouldMount) { - widgetsToMount.push(widget); - localWidgets.set(attribute, { - widget: widget, - isMounted: true - }); - } else if (isMounted && !shouldMount) { - widgetsToUnmount.push(widget); - localWidgets.set(attribute, { - widget: widget, - isMounted: false - }); - } - }); - parent.addWidgets(widgetsToMount); // make sure this only happens after the regular render, otherwise it - // happens too quick, since render is "deferred" for the next microtask, - // so this needs to be a whole task later + // try lookup with get for backbone or similar model data + } else if (doModelGet && scope.get && typeof scope.get == 'function') { + val = scope.get(key); + } + } - setTimeout(function () { - return parent.removeWidgets(widgetsToUnmount); - }, 0); - renderFn(_objectSpread2(_objectSpread2({}, renderState), {}, { - instantSearchInstance: renderOptions.instantSearchInstance - }), false); - }, - dispose: function dispose(_ref2) { - var parent = _ref2.parent; - var toRemove = []; - localWidgets.forEach(function (_ref3) { - var widget = _ref3.widget, - isMounted = _ref3.isMounted; + return val; + } - if (isMounted) { - toRemove.push(widget); - } - }); - parent.removeWidgets(toRemove); - unmountFn(); - }, - getWidgetSearchParameters: function getWidgetSearchParameters(state) { - // broadening the scope of facets to avoid conflict between never and * - return facets.reduce(function (acc, curr) { - return acc.addFacet(curr); - }, state.setQueryParameters({ - maxValuesPerFacet: Math.max(maxValuesPerFacet || 0, state.maxValuesPerFacet || 0) - })); - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - dynamicWidgets: this.getWidgetRenderState(renderOptions) - }); - }, - getWidgetRenderState: function getWidgetRenderState(_ref4) { - var _results$renderingCon, _results$renderingCon2, _results$renderingCon3, _results$renderingCon4; + function createSpecializedPartial(instance, subs, partials, stackSubs, stackPartials, stackText) { + function PartialTemplate() {} PartialTemplate.prototype = instance; + function Substitutions() {} Substitutions.prototype = instance.subs; + var key; + var partial = new PartialTemplate(); + partial.subs = new Substitutions(); + partial.subsText = {}; //hehe. substext. + partial.buf = ''; + + stackSubs = stackSubs || {}; + partial.stackSubs = stackSubs; + partial.subsText = stackText; + for (key in subs) { + if (!stackSubs[key]) stackSubs[key] = subs[key]; + } + for (key in stackSubs) { + partial.subs[key] = stackSubs[key]; + } - var results = _ref4.results, - state = _ref4.state; + stackPartials = stackPartials || {}; + partial.stackPartials = stackPartials; + for (key in partials) { + if (!stackPartials[key]) stackPartials[key] = partials[key]; + } + for (key in stackPartials) { + partial.partials[key] = stackPartials[key]; + } - if (!results) { - return { - attributesToRender: [], - widgetParams: widgetParams - }; - } + return partial; + } - var attributesToRender = transformItems((_results$renderingCon = (_results$renderingCon2 = results.renderingContent) === null || _results$renderingCon2 === void 0 ? void 0 : (_results$renderingCon3 = _results$renderingCon2.facetOrdering) === null || _results$renderingCon3 === void 0 ? void 0 : (_results$renderingCon4 = _results$renderingCon3.facets) === null || _results$renderingCon4 === void 0 ? void 0 : _results$renderingCon4.order) !== null && _results$renderingCon !== void 0 ? _results$renderingCon : [], { - results: results - }); + var rAmp = /&/g, + rLt = //g, + rApos = /\'/g, + rQuot = /\"/g, + hChars = /[&<>\"\']/; - if (!Array.isArray(attributesToRender)) { - throw new Error(withUsage$r('The `transformItems` option expects a function that returns an Array.')); - } + function coerceToString(val) { + return String((val === null || val === undefined) ? '' : val); + } - _warning(maxValuesPerFacet >= (state.maxValuesPerFacet || 0), "The maxValuesPerFacet set by dynamic widgets (".concat(maxValuesPerFacet, ") is smaller than one of the limits set by a widget (").concat(state.maxValuesPerFacet, "). This causes a mismatch in query parameters and thus an extra network request when that widget is mounted.")) ; - _warning(attributesToRender.length <= MAX_WILDCARD_FACETS || widgetParams.facets !== undefined, "More than ".concat(MAX_WILDCARD_FACETS, " facets are requested to be displayed without explicitly setting which facets to retrieve. This could have a performance impact. Set \"facets\" to [] to do two smaller network requests, or explicitly to ['*'] to avoid this warning.")) ; - return { - attributesToRender: attributesToRender, - widgetParams: widgetParams - }; - } - }; + function hoganEscape(str) { + str = coerceToString(str); + return hChars.test(str) ? + str + .replace(rAmp, '&') + .replace(rLt, '<') + .replace(rGt, '>') + .replace(rApos, ''') + .replace(rQuot, '"') : + str; + } + + var isArray = Array.isArray || function(a) { + return Object.prototype.toString.call(a) === '[object Array]'; }; - }; - /** @deprecated use connectDynamicWidgets */ + })( exports ); + }); - var EXPERIMENTAL_connectDynamicWidgets = deprecate(connectDynamicWidgets, 'use connectDynamicWidgets'); + /* + * Copyright 2011 Twitter, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ - var connectors = /*#__PURE__*/Object.freeze({ - __proto__: null, - connectDynamicWidgets: connectDynamicWidgets, - EXPERIMENTAL_connectDynamicWidgets: EXPERIMENTAL_connectDynamicWidgets, - connectClearRefinements: connectClearRefinements, - connectCurrentRefinements: connectCurrentRefinements, - connectHierarchicalMenu: connectHierarchicalMenu, - connectHits: connectHits, - connectHitsWithInsights: connectHitsWithInsights, - connectHitsPerPage: connectHitsPerPage, - connectInfiniteHits: connectInfiniteHits, - connectInfiniteHitsWithInsights: connectInfiniteHitsWithInsights, - connectMenu: connectMenu, - connectNumericMenu: connectNumericMenu, - connectPagination: connectPagination, - connectRange: connectRange, - connectRefinementList: connectRefinementList, - connectSearchBox: connectSearchBox, - connectSortBy: connectSortBy, - connectRatingMenu: connectRatingMenu, - connectStats: connectStats, - connectToggleRefinement: connectToggleRefinement, - connectBreadcrumb: connectBreadcrumb, - connectGeoSearch: connectGeoSearch, - connectPoweredBy: connectPoweredBy, - connectConfigure: connectConfigure, - EXPERIMENTAL_connectConfigureRelatedItems: connectConfigureRelatedItems, - connectAutocomplete: connectAutocomplete, - connectQueryRules: connectQueryRules, - connectVoiceSearch: connectVoiceSearch, - EXPERIMENTAL_connectAnswers: connectAnswers, - connectRelevantSort: connectRelevantSort - }); + // This file is for use with Node.js. See dist/ for browser files. - var withUsage$s = createDocumentationMessageGenerator({ - name: 'analytics' - }); - // @major this widget will be removed from the next major version. - var analytics = function analytics(widgetParams) { - var _ref = widgetParams || {}, - pushFunction = _ref.pushFunction, - _ref$delay = _ref.delay, - delay = _ref$delay === void 0 ? 3000 : _ref$delay, - _ref$triggerOnUIInter = _ref.triggerOnUIInteraction, - triggerOnUIInteraction = _ref$triggerOnUIInter === void 0 ? false : _ref$triggerOnUIInter, - _ref$pushInitialSearc = _ref.pushInitialSearch, - pushInitialSearch = _ref$pushInitialSearc === void 0 ? true : _ref$pushInitialSearc, - _ref$pushPagination = _ref.pushPagination, - pushPagination = _ref$pushPagination === void 0 ? false : _ref$pushPagination; + compiler.Template = template.Template; + compiler.template = compiler.Template; + var hogan = compiler; - if (!pushFunction) { - throw new Error(withUsage$s('The `pushFunction` option is required.')); - } + var n$1=function(t,s,r,e){var u;s[0]=0;for(var h=1;h=5&&((e||!n&&5===r)&&(h.push(r,0,e,s),r=6),n&&(h.push(r,n,0,s),r=6)),e="";},a=0;a"===t?(r=1,e=""):e=t+e[0]:u?t===u?u="":e+=t:'"'===t||"'"===t?u=t:">"===t?(p(),r=1):r&&("="===t?(r=5,s=e,e=""):"/"===t&&(r<5||">"===n[a][l+1])?(p(),3===r&&(h=h[0]),r=h,(h=h[0]).push(2,0,r),r=0):" "===t||"\t"===t||"\n"===t||"\r"===t?(p(),r=2):e+=t),3===r&&"!--"===e&&(r=4,h=h[0]);}return p(),h}(s)),r),arguments,[])).length>1?r:r[0]} - _warning(false, "`analytics` widget has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the `insights` middleware.\n\nFor the migration, visit https://www.algolia.com/doc/guides/building-search-ui/upgrade-guides/js/#analytics-widget") ; - var cachedState = null; + var m$1=e$1.bind(h); - var serializeRefinements = function serializeRefinements(parameters) { - var refinements = []; + var _extends_1 = createCommonjsModule(function (module) { + function _extends() { + module.exports = _extends = Object.assign || function (target) { + for (var i = 1; i < arguments.length; i++) { + var source = arguments[i]; - for (var parameter in parameters) { - if (parameters.hasOwnProperty(parameter)) { - var values = parameters[parameter].join('+'); - refinements.push("".concat(encodeURIComponent(parameter), "=").concat(encodeURIComponent(parameter), "_").concat(encodeURIComponent(values))); + for (var key in source) { + if (Object.prototype.hasOwnProperty.call(source, key)) { + target[key] = source[key]; + } } } - return refinements.join('&'); + return target; }; - var serializeNumericRefinements = function serializeNumericRefinements(numericRefinements) { - var refinements = []; + return _extends.apply(this, arguments); + } - for (var attribute in numericRefinements) { - if (numericRefinements.hasOwnProperty(attribute)) { - var filter = numericRefinements[attribute]; + module.exports = _extends; + }); - if (filter.hasOwnProperty('>=') && filter.hasOwnProperty('<=')) { - if (filter['>='] && filter['>='][0] === filter['<='] && filter['<='][0]) { - refinements.push("".concat(attribute, "=").concat(attribute, "_").concat(filter['>='])); - } else { - refinements.push("".concat(attribute, "=").concat(attribute, "_").concat(filter['>='], "to").concat(filter['<='])); - } - } else if (filter.hasOwnProperty('>=')) { - refinements.push("".concat(attribute, "=").concat(attribute, "_from").concat(filter['>='])); - } else if (filter.hasOwnProperty('<=')) { - refinements.push("".concat(attribute, "=").concat(attribute, "_to").concat(filter['<='])); - } else if (filter.hasOwnProperty('=')) { - var equals = []; + function _objectWithoutPropertiesLoose$2(source, excluded) { + if (source == null) return {}; + var target = {}; + var sourceKeys = Object.keys(source); + var key, i; - for (var equal in filter['=']) { - // eslint-disable-next-line max-depth - if (filter['='].hasOwnProperty(equal)) { - // @ts-ignore somehow 'equal' is a string, even though it's a number? - equals.push(filter['='][equal]); - } - } + for (i = 0; i < sourceKeys.length; i++) { + key = sourceKeys[i]; + if (excluded.indexOf(key) >= 0) continue; + target[key] = source[key]; + } - refinements.push("".concat(attribute, "=").concat(attribute, "_").concat(equals.join('-'))); - } - } + return target; + } + + var objectWithoutPropertiesLoose = _objectWithoutPropertiesLoose$2; + + function _objectWithoutProperties$1(source, excluded) { + if (source == null) return {}; + var target = objectWithoutPropertiesLoose(source, excluded); + var key, i; + + if (Object.getOwnPropertySymbols) { + var sourceSymbolKeys = Object.getOwnPropertySymbols(source); + + for (i = 0; i < sourceSymbolKeys.length; i++) { + key = sourceSymbolKeys[i]; + if (excluded.indexOf(key) >= 0) continue; + if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; + target[key] = source[key]; } + } - return refinements.join('&'); + return target; + } + + var objectWithoutProperties = _objectWithoutProperties$1; + + var _excluded = ["parts", "highlightedTagName", "nonHighlightedTagName", "separator", "className", "classNames"]; + // This is a minimal subset of the actual types from the `JSX` namespace. + + function createHighlightPartComponent(_ref) { + var createElement = _ref.createElement; + return function HighlightPart(_ref2) { + var classNames = _ref2.classNames, + children = _ref2.children, + highlightedTagName = _ref2.highlightedTagName, + isHighlighted = _ref2.isHighlighted, + nonHighlightedTagName = _ref2.nonHighlightedTagName; + var TagName = isHighlighted ? highlightedTagName : nonHighlightedTagName; + return createElement(TagName, { + className: isHighlighted ? classNames.highlighted : classNames.nonHighlighted + }, children); }; + } - var lastSentData = ''; + function createHighlightComponent(_ref3) { + var createElement = _ref3.createElement, + Fragment = _ref3.Fragment; + var HighlightPart = createHighlightPartComponent({ + createElement: createElement, + Fragment: Fragment + }); + return function Highlight(_ref4) { + var parts = _ref4.parts, + _ref4$highlightedTagN = _ref4.highlightedTagName, + highlightedTagName = _ref4$highlightedTagN === void 0 ? 'mark' : _ref4$highlightedTagN, + _ref4$nonHighlightedT = _ref4.nonHighlightedTagName, + nonHighlightedTagName = _ref4$nonHighlightedT === void 0 ? 'span' : _ref4$nonHighlightedT, + _ref4$separator = _ref4.separator, + separator = _ref4$separator === void 0 ? ', ' : _ref4$separator, + className = _ref4.className, + _ref4$classNames = _ref4.classNames, + classNames = _ref4$classNames === void 0 ? {} : _ref4$classNames, + props = objectWithoutProperties(_ref4, _excluded); + + return createElement("span", _extends_1({}, props, { + className: cx(classNames.root, className) + }), parts.map(function (part, partIndex) { + var isLastPart = partIndex === parts.length - 1; + return createElement(Fragment, { + key: partIndex + }, part.map(function (subPart, subPartIndex) { + return createElement(HighlightPart, { + key: subPartIndex, + classNames: classNames, + highlightedTagName: highlightedTagName, + nonHighlightedTagName: nonHighlightedTagName, + isHighlighted: subPart.isHighlighted + }, subPart.value); + }), !isLastPart && createElement("span", { + className: classNames.separator + }, separator)); + })); + }; + } - var sendAnalytics = function sendAnalytics(analyticsState) { - if (analyticsState === null) { - return; - } + var InternalHighlight = createHighlightComponent({ + createElement: h, + Fragment: p + }); - var serializedParams = []; - var serializedRefinements = serializeRefinements(_objectSpread2(_objectSpread2(_objectSpread2({}, analyticsState.state.disjunctiveFacetsRefinements), analyticsState.state.facetsRefinements), analyticsState.state.hierarchicalFacetsRefinements)); - var serializedNumericRefinements = serializeNumericRefinements(analyticsState.state.numericRefinements); + function Highlight(_ref) { + var _ref$classNames = _ref.classNames, + classNames = _ref$classNames === void 0 ? {} : _ref$classNames, + props = _objectWithoutProperties(_ref, ["classNames"]); - if (serializedRefinements !== '') { - serializedParams.push(serializedRefinements); + return h(InternalHighlight, _extends({ + classNames: { + root: cx('ais-Highlight', classNames.root), + highlighted: cx('ais-Highlight-highlighted', classNames.highlighted), + nonHighlighted: cx('ais-Highlight-nonHighlighted', classNames.nonHighlighted), + separator: cx('ais-Highlight-separator', classNames.separator) } + }, props)); + } - if (serializedNumericRefinements !== '') { - serializedParams.push(serializedNumericRefinements); - } + function Highlight$1(_ref) { + var hit = _ref.hit, + attribute = _ref.attribute, + cssClasses = _ref.cssClasses, + props = _objectWithoutProperties(_ref, ["hit", "attribute", "cssClasses"]); + + var property = getPropertyByPath(hit._highlightResult, attribute) || []; + var properties = toArray(property); + _warning(Boolean(properties.length), "Could not enable highlight for \"".concat(attribute.toString(), "\", will display an empty string.\nPlease check whether this attribute exists and is either searchable or specified in `attributesToHighlight`.\n\nSee: https://alg.li/highlighting\n")) ; + var parts = properties.map(function (_ref2) { + var value = _ref2.value; + return getHighlightedParts(unescape$1(value || '')); + }); + return h(Highlight, _extends({}, props, { + parts: parts, + classNames: cssClasses + })); + } - var stringifiedParams = serializedParams.join('&'); - var dataToSend = "Query: ".concat(analyticsState.state.query || '', ", ").concat(stringifiedParams); + function ReverseHighlight(_ref) { + var _ref$classNames = _ref.classNames, + classNames = _ref$classNames === void 0 ? {} : _ref$classNames, + props = _objectWithoutProperties(_ref, ["classNames"]); - if (pushPagination === true) { - dataToSend += ", Page: ".concat(analyticsState.state.page || 0); + return h(InternalHighlight, _extends({ + classNames: { + root: cx('ais-ReverseHighlight', classNames.root), + highlighted: cx('ais-ReverseHighlight-highlighted', classNames.highlighted), + nonHighlighted: cx('ais-ReverseHighlight-nonHighlighted', classNames.nonHighlighted), + separator: cx('ais-ReverseHighlight-separator', classNames.separator) } + }, props)); + } - if (lastSentData !== dataToSend) { - pushFunction(stringifiedParams, analyticsState.state, analyticsState.results); - lastSentData = dataToSend; + function ReverseHighlight$1(_ref) { + var hit = _ref.hit, + attribute = _ref.attribute, + cssClasses = _ref.cssClasses, + props = _objectWithoutProperties(_ref, ["hit", "attribute", "cssClasses"]); + + var property = getPropertyByPath(hit._highlightResult, attribute) || []; + var properties = toArray(property); + _warning(Boolean(properties.length), "Could not enable highlight for \"".concat(attribute.toString(), "\", will display an empty string.\nPlease check whether this attribute exists and is either searchable or specified in `attributesToHighlight`.\n\nSee: https://alg.li/highlighting\n")) ; + var parts = properties.map(function (_ref2) { + var value = _ref2.value; + return getHighlightedParts(unescape$1(value || '')).map(function (_ref3) { + var isHighlighted = _ref3.isHighlighted, + rest = _objectWithoutProperties(_ref3, ["isHighlighted"]); + + return _objectSpread2(_objectSpread2({}, rest), {}, { + isHighlighted: !isHighlighted + }); + }); + }); + return h(ReverseHighlight, _extends({}, props, { + parts: parts, + classNames: cssClasses + })); + } + + function ReverseSnippet(_ref) { + var _ref$classNames = _ref.classNames, + classNames = _ref$classNames === void 0 ? {} : _ref$classNames, + props = _objectWithoutProperties(_ref, ["classNames"]); + + return h(InternalHighlight, _extends({ + classNames: { + root: cx('ais-ReverseSnippet', classNames.root), + highlighted: cx('ais-ReverseSnippet-highlighted', classNames.highlighted), + nonHighlighted: cx('ais-ReverseSnippet-nonHighlighted', classNames.nonHighlighted), + separator: cx('ais-ReverseSnippet-separator', classNames.separator) } - }; + }, props)); + } - var pushTimeout; - var isInitialSearch = true; + function ReverseSnippet$1(_ref) { + var hit = _ref.hit, + attribute = _ref.attribute, + cssClasses = _ref.cssClasses, + props = _objectWithoutProperties(_ref, ["hit", "attribute", "cssClasses"]); + + var property = getPropertyByPath(hit._snippetResult, attribute) || []; + var properties = toArray(property); + _warning(Boolean(properties.length), "Could not enable snippet for \"".concat(attribute.toString(), "\", will display an empty string.\nPlease check whether this attribute exists and is specified in `attributesToSnippet`.\n\nSee: https://alg.li/highlighting\n")) ; + var parts = properties.map(function (_ref2) { + var value = _ref2.value; + return getHighlightedParts(unescape$1(value || '')).map(function (_ref3) { + var isHighlighted = _ref3.isHighlighted, + rest = _objectWithoutProperties(_ref3, ["isHighlighted"]); + + return _objectSpread2(_objectSpread2({}, rest), {}, { + isHighlighted: !isHighlighted + }); + }); + }); + return h(ReverseSnippet, _extends({}, props, { + parts: parts, + classNames: cssClasses + })); + } - if (pushInitialSearch === true) { - isInitialSearch = false; - } + function Snippet(_ref) { + var _ref$classNames = _ref.classNames, + classNames = _ref$classNames === void 0 ? {} : _ref$classNames, + props = _objectWithoutProperties(_ref, ["classNames"]); - var onClick = function onClick() { - sendAnalytics(cachedState); - }; + return h(InternalHighlight, _extends({ + classNames: { + root: cx('ais-Snippet', classNames.root), + highlighted: cx('ais-Snippet-highlighted', classNames.highlighted), + nonHighlighted: cx('ais-Snippet-nonHighlighted', classNames.nonHighlighted), + separator: cx('ais-Snippet-separator', classNames.separator) + } + }, props)); + } - var onUnload = function onUnload() { - sendAnalytics(cachedState); - }; + function Snippet$1(_ref) { + var hit = _ref.hit, + attribute = _ref.attribute, + cssClasses = _ref.cssClasses, + props = _objectWithoutProperties(_ref, ["hit", "attribute", "cssClasses"]); + + var property = getPropertyByPath(hit._snippetResult, attribute) || []; + var properties = toArray(property); + _warning(Boolean(properties.length), "Could not enable snippet for \"".concat(attribute.toString(), "\", will display an empty string.\nPlease check whether this attribute exists and is specified in `attributesToSnippet`.\n\nSee: https://alg.li/highlighting\n")) ; + var parts = properties.map(function (_ref2) { + var value = _ref2.value; + return getHighlightedParts(unescape$1(value || '')); + }); + return h(Snippet, _extends({}, props, { + parts: parts, + classNames: cssClasses + })); + } - return { - $$type: 'ais.analytics', - $$widgetType: 'ais.analytics', - init: function init() { - if (triggerOnUIInteraction === true) { - document.addEventListener('click', onClick); - window.addEventListener('beforeunload', onUnload); - } - }, - render: function render(_ref2) { - var results = _ref2.results, - state = _ref2.state; + // We add all our template helper methods to the template as lambdas. Note + // that lambdas in Mustache are supposed to accept a second argument of + // `render` to get the rendered value, not the literal `{{value}}`. But + // this is currently broken (see https://github.com/twitter/hogan.js/issues/222). + function transformHelpersToHogan() { + var helpers = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; + var compileOptions = arguments.length > 1 ? arguments[1] : undefined; + var data = arguments.length > 2 ? arguments[2] : undefined; + return Object.keys(helpers).reduce(function (acc, helperKey) { + return _objectSpread2(_objectSpread2({}, acc), {}, _defineProperty({}, helperKey, function () { + var _this = this; - if (isInitialSearch === true) { - isInitialSearch = false; - return; - } + return function (text) { + var render = function render(value) { + return hogan.compile(value, compileOptions).render(_this); + }; - cachedState = { - results: results, - state: state + return helpers[helperKey].call(data, text, render); }; + })); + }, {}); + } - if (pushTimeout) { - clearTimeout(pushTimeout); - } + function renderTemplate(_ref) { + var templates = _ref.templates, + templateKey = _ref.templateKey, + compileOptions = _ref.compileOptions, + helpers = _ref.helpers, + data = _ref.data, + bindEvent = _ref.bindEvent, + sendEvent = _ref.sendEvent; + var template = templates[templateKey]; - pushTimeout = window.setTimeout(function () { - return sendAnalytics(cachedState); - }, delay); - }, - dispose: function dispose() { - if (triggerOnUIInteraction === true) { - document.removeEventListener('click', onClick); - window.removeEventListener('beforeunload', onUnload); - } - }, - getRenderState: function getRenderState(renderState, renderOptions) { - return _objectSpread2(_objectSpread2({}, renderState), {}, { - analytics: this.getWidgetRenderState(renderOptions) - }); - }, - getWidgetRenderState: function getWidgetRenderState() { - return { - widgetParams: widgetParams - }; - } - }; - }; + if (typeof template !== 'string' && typeof template !== 'function') { + throw new Error("Template must be 'string' or 'function', was '".concat(_typeof(template), "' (key: ").concat(templateKey, ")")); + } - var classnames = createCommonjsModule(function (module) { - /*! - Copyright (c) 2017 Jed Watson. - Licensed under the MIT License (MIT), see - http://jedwatson.github.io/classnames - */ - /* global define */ - - (function () { - - var hasOwn = {}.hasOwnProperty; - - function classNames () { - var classes = []; - - for (var i = 0; i < arguments.length; i++) { - var arg = arguments[i]; - if (!arg) continue; - - var argType = typeof arg; - - if (argType === 'string' || argType === 'number') { - classes.push(arg); - } else if (Array.isArray(arg) && arg.length) { - var inner = classNames.apply(null, arg); - if (inner) { - classes.push(inner); - } - } else if (argType === 'object') { - for (var key in arg) { - if (hasOwn.call(arg, key) && arg[key]) { - classes.push(key); - } - } - } - } - - return classes.join(' '); - } - - if ( module.exports) { - classNames.default = classNames; - module.exports = classNames; - } else { - window.classNames = classNames; - } - }()); - }); + if (typeof template === 'function') { + // @MAJOR no longer pass bindEvent when string templates are removed + var params = bindEvent || {}; + params.html = m$1; + params.sendEvent = sendEvent; + params.components = { + Highlight: Highlight$1, + ReverseHighlight: ReverseHighlight$1, + Snippet: Snippet$1, + ReverseSnippet: ReverseSnippet$1 + }; + return template(data, params); + } + + var transformedHelpers = transformHelpersToHogan(helpers, compileOptions, data); + return hogan.compile(template, compileOptions).render(_objectSpread2(_objectSpread2({}, data), {}, { + helpers: transformedHelpers + })).replace(/[ \n\r\t\f\xA0]+/g, function (spaces) { + return spaces.replace(/(^|\xA0+)[^\xA0]+/g, '$1 '); + }).trim(); + } var defaultProps = { data: {}, @@ -15805,6 +16192,11 @@ }, { key: "render", value: function render() { + var _this = this; + + _warning(Object.keys(this.props.templates).every(function (key) { + return typeof _this.props.templates[key] === 'function'; + }), "Hogan.js and string-based templates are deprecated and will not be supported in InstantSearch.js 5.x.\n\nYou can replace them with function-form templates and use either the provided `html` function or JSX templates.\n\nSee: https://www.algolia.com/doc/guides/building-search-ui/upgrade-guides/js/#upgrade-templates") ; var RootTagName = this.props.rootTagName; var useCustomCompileOptions = this.props.useCustomCompileOptions[this.props.templateKey]; var compileOptions = useCustomCompileOptions ? this.props.templatesConfig.compileOptions : {}; @@ -15814,7 +16206,8 @@ compileOptions: compileOptions, helpers: this.props.templatesConfig.helpers, data: this.props.data, - bindEvent: this.props.bindEvent + bindEvent: this.props.bindEvent, + sendEvent: this.props.sendEvent }); if (content === null) { @@ -15823,7 +16216,11 @@ return null; } - return v(RootTagName, _extends({}, this.props.rootProps, { + if (_typeof(content) === 'object') { + return h(RootTagName, this.props.rootProps, content); + } + + return h(RootTagName, _extends({}, this.props.rootProps, { dangerouslySetInnerHTML: { __html: content } @@ -15832,7 +16229,7 @@ }]); return Template; - }(_); + }(d); _defineProperty(Template, "defaultProps", defaultProps); @@ -15842,13 +16239,13 @@ templateProps = _ref.templateProps, createURL = _ref.createURL, refine = _ref.refine; - return v("div", { - className: classnames(cssClasses.root, _defineProperty({}, cssClasses.noRefinementRoot, items.length === 0)) - }, v("ul", { + return h("div", { + className: cx(cssClasses.root, items.length === 0 && cssClasses.noRefinementRoot) + }, h("ul", { className: cssClasses.list - }, v("li", { - className: classnames(cssClasses.item, _defineProperty({}, cssClasses.selectedItem, items.length === 0)) - }, v(Template, _extends({}, templateProps, { + }, h("li", { + className: cx(cssClasses.item, items.length === 0 && cssClasses.selectedItem) + }, h(Template, _extends({}, templateProps, { templateKey: "home", rootTagName: "a", rootProps: { @@ -15861,17 +16258,17 @@ } }))), items.map(function (item, idx) { var isLast = idx === items.length - 1; - return v("li", { + return h("li", { key: item.label + idx, - className: classnames(cssClasses.item, _defineProperty({}, cssClasses.selectedItem, isLast)) - }, v(Template, _extends({}, templateProps, { + className: cx(cssClasses.item, isLast && cssClasses.selectedItem) + }, h(Template, _extends({}, templateProps, { templateKey: "separator", rootTagName: "span", rootProps: { className: cssClasses.separator, 'aria-hidden': true } - })), isLast ? item.label : v("a", { + })), isLast ? item.label : h("a", { className: cssClasses.link, href: createURL(item.value), onClick: function onClick(event) { @@ -15883,8 +16280,12 @@ }; var defaultTemplates = { - home: 'Home', - separator: '>' + home: function home() { + return 'Home'; + }, + separator: function separator() { + return '>'; + } }; var withUsage$t = createDocumentationMessageGenerator({ @@ -15913,7 +16314,7 @@ return; } - S(v(Breadcrumb, { + P(h(Breadcrumb, { canRefine: canRefine, cssClasses: cssClasses, createURL: createURL, @@ -15942,24 +16343,24 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$4(), userCssClasses.root), - noRefinementRoot: classnames(suit$4({ + root: cx(suit$4(), userCssClasses.root), + noRefinementRoot: cx(suit$4({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: classnames(suit$4({ + list: cx(suit$4({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$4({ + item: cx(suit$4({ descendantName: 'item' }), userCssClasses.item), - selectedItem: classnames(suit$4({ + selectedItem: cx(suit$4({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - separator: classnames(suit$4({ + separator: cx(suit$4({ descendantName: 'separator' }), userCssClasses.separator), - link: classnames(suit$4({ + link: cx(suit$4({ descendantName: 'link' }), userCssClasses.link) }; @@ -15970,7 +16371,7 @@ templates: templates }); var makeWidget = connectBreadcrumb(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attributes: attributes, @@ -15987,13 +16388,13 @@ refine = _ref.refine, cssClasses = _ref.cssClasses, templateProps = _ref.templateProps; - return v("div", { + return h("div", { className: cssClasses.root - }, v(Template, _extends({}, templateProps, { + }, h(Template, _extends({}, templateProps, { templateKey: "resetLabel", rootTagName: "button", rootProps: { - className: classnames(cssClasses.button, _defineProperty({}, cssClasses.disabledButton, !hasRefinements)), + className: cx(cssClasses.button, !hasRefinements && cssClasses.disabledButton), onClick: refine, disabled: !hasRefinements }, @@ -16004,7 +16405,9 @@ }; var defaultTemplates$1 = { - resetLabel: 'Clear refinements' + resetLabel: function resetLabel() { + return 'Clear refinements'; + } }; var withUsage$u = createDocumentationMessageGenerator({ @@ -16019,7 +16422,7 @@ templates = _ref.templates; return function (_ref2, isFirstRendering) { var refine = _ref2.refine, - hasRefinements = _ref2.hasRefinements, + canRefine = _ref2.canRefine, instantSearchInstance = _ref2.instantSearchInstance; if (isFirstRendering) { @@ -16031,10 +16434,10 @@ return; } - S(v(ClearRefinements, { + P(h(ClearRefinements, { refine: refine, cssClasses: cssClasses, - hasRefinements: hasRefinements, + hasRefinements: canRefine, templateProps: renderState.templateProps }), containerNode); }; @@ -16057,11 +16460,11 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$5(), userCssClasses.root), - button: classnames(suit$5({ + root: cx(suit$5(), userCssClasses.root), + button: cx(suit$5({ descendantName: 'button' }), userCssClasses.button), - disabledButton: classnames(suit$5({ + disabledButton: cx(suit$5({ descendantName: 'button', modifierName: 'disabled' }), userCssClasses.disabledButton) @@ -16073,7 +16476,7 @@ templates: templates }); var makeWidget = connectClearRefinements(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ includedAttributes: includedAttributes, @@ -16100,8 +16503,6 @@ }); }; - /** @jsx h */ - var createItemKey = function createItemKey(_ref) { var attribute = _ref.attribute, value = _ref.value, @@ -16125,24 +16526,25 @@ var CurrentRefinements = function CurrentRefinements(_ref2) { var items = _ref2.items, - cssClasses = _ref2.cssClasses; - return v("div", { - className: cssClasses.root - }, v("ul", { + cssClasses = _ref2.cssClasses, + canRefine = _ref2.canRefine; + return h("div", { + className: cx(cssClasses.root, !canRefine && cssClasses.noRefinementRoot) + }, h("ul", { className: cssClasses.list }, items.map(function (item, index) { - return v("li", { + return h("li", { key: "".concat(item.indexName, "-").concat(item.attribute, "-").concat(index), className: cssClasses.item - }, v("span", { + }, h("span", { className: cssClasses.label }, capitalize(item.label), ":"), item.refinements.map(function (refinement) { - return v("span", { + return h("span", { key: createItemKey(refinement), className: cssClasses.category - }, v("span", { + }, h("span", { className: cssClasses.categoryLabel - }, refinement.attribute === 'query' ? v("q", null, refinement.label) : refinement.label), v("button", { + }, refinement.attribute === 'query' ? h("q", null, refinement.label) : refinement.label), h("button", { className: cssClasses.delete, onClick: handleClick(item.refine.bind(null, refinement)) }, "\u2715")); @@ -16157,7 +16559,8 @@ var renderer$2 = function renderer(_ref, isFirstRender) { var items = _ref.items, - widgetParams = _ref.widgetParams; + widgetParams = _ref.widgetParams, + canRefine = _ref.canRefine; if (isFirstRender) { return; @@ -16166,9 +16569,10 @@ var _ref2 = widgetParams, container = _ref2.container, cssClasses = _ref2.cssClasses; - S(v(CurrentRefinements, { + P(h(CurrentRefinements, { cssClasses: cssClasses, - items: items + items: items, + canRefine: canRefine }), container); }; @@ -16187,28 +16591,31 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$6(), userCssClasses.root), - list: classnames(suit$6({ + root: cx(suit$6(), userCssClasses.root), + noRefinementRoot: cx(suit$6({ + modifierName: 'noRefinement' + }), userCssClasses.noRefinementRoot), + list: cx(suit$6({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$6({ + item: cx(suit$6({ descendantName: 'item' }), userCssClasses.item), - label: classnames(suit$6({ + label: cx(suit$6({ descendantName: 'label' }), userCssClasses.label), - category: classnames(suit$6({ + category: cx(suit$6({ descendantName: 'category' }), userCssClasses.category), - categoryLabel: classnames(suit$6({ + categoryLabel: cx(suit$6({ descendantName: 'categoryLabel' }), userCssClasses.categoryLabel), - delete: classnames(suit$6({ + delete: cx(suit$6({ descendantName: 'delete' }), userCssClasses.delete) }; var makeWidget = connectCurrentRefinements(renderer$2, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ container: containerNode, @@ -16222,8 +16629,12 @@ }; var defaultTemplates$2 = { - header: '', - loader: '', + header: function header() { + return ''; + }, + loader: function loader() { + return ''; + }, item: function item(_item) { return JSON.stringify(_item); } @@ -16234,9 +16645,9 @@ isLoading = _ref.isLoading, cssClasses = _ref.cssClasses, templateProps = _ref.templateProps; - return v("div", { - className: classnames(cssClasses.root, _defineProperty({}, cssClasses.emptyRoot, hits.length === 0)) - }, v(Template, _extends({}, templateProps, { + return h("div", { + className: cx(cssClasses.root, hits.length === 0 && cssClasses.emptyRoot) + }, h(Template, _extends({}, templateProps, { templateKey: "header", rootProps: { className: cssClasses.header @@ -16245,15 +16656,15 @@ hits: hits, isLoading: isLoading } - })), isLoading ? v(Template, _extends({}, templateProps, { + })), isLoading ? h(Template, _extends({}, templateProps, { templateKey: "loader", rootProps: { className: cssClasses.loader } - })) : v("ul", { + })) : h("ul", { className: cssClasses.list }, hits.map(function (hit, position) { - return v(Template, _extends({}, templateProps, { + return h(Template, _extends({}, templateProps, { templateKey: "item", rootTagName: "li", rootProps: { @@ -16291,7 +16702,7 @@ return; } - S(v(Answers, { + P(h(Answers, { cssClasses: cssClasses, hits: hits, isLoading: isLoading, @@ -16321,20 +16732,20 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$7(), userCssClasses.root), - emptyRoot: classnames(suit$7({ + root: cx(suit$7(), userCssClasses.root), + emptyRoot: cx(suit$7({ modifierName: 'empty' }), userCssClasses.emptyRoot), - header: classnames(suit$7({ + header: cx(suit$7({ descendantName: 'header' }), userCssClasses.header), - loader: classnames(suit$7({ + loader: cx(suit$7({ descendantName: 'loader' }), userCssClasses.loader), - list: classnames(suit$7({ + list: cx(suit$7({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$7({ + item: cx(suit$7({ descendantName: 'item' }), userCssClasses.item) }; @@ -16345,7 +16756,7 @@ renderState: {} }); var makeWidget = connectAnswers(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attributesForPrediction: attributesForPrediction, @@ -16448,32 +16859,28 @@ }); }; - /** @jsx h */ - var GeoSearchButton = function GeoSearchButton(_ref) { var className = _ref.className, _ref$disabled = _ref.disabled, disabled = _ref$disabled === void 0 ? false : _ref$disabled, onClick = _ref.onClick, children = _ref.children; - return v("button", { + return h("button", { className: className, onClick: onClick, disabled: disabled }, children); }; - /** @jsx h */ - var GeoSearchToggle = function GeoSearchToggle(_ref) { var classNameLabel = _ref.classNameLabel, classNameInput = _ref.classNameInput, checked = _ref.checked, onToggle = _ref.onToggle, children = _ref.children; - return v("label", { + return h("label", { className: classNameLabel - }, v("input", { + }, h("input", { className: classNameInput, type: "checkbox", checked: checked, @@ -16493,36 +16900,36 @@ onRefineClick = _ref.onRefineClick, onClearClick = _ref.onClearClick, templateProps = _ref.templateProps; - return v(d, null, enableRefine && v("div", null, enableRefineControl && v("div", { + return h(p, null, enableRefine && h("div", null, enableRefineControl && h("div", { className: cssClasses.control - }, isRefineOnMapMove || !hasMapMoveSinceLastRefine ? v(GeoSearchToggle, { - classNameLabel: classnames(cssClasses.label, _defineProperty({}, cssClasses.selectedLabel, isRefineOnMapMove)), + }, isRefineOnMapMove || !hasMapMoveSinceLastRefine ? h(GeoSearchToggle, { + classNameLabel: cx(cssClasses.label, isRefineOnMapMove && cssClasses.selectedLabel), classNameInput: cssClasses.input, checked: isRefineOnMapMove, onToggle: onRefineToggle - }, v(Template, _extends({}, templateProps, { + }, h(Template, _extends({}, templateProps, { templateKey: "toggle", rootTagName: "span" - }))) : v(GeoSearchButton, { + }))) : h(GeoSearchButton, { className: cssClasses.redo, disabled: !hasMapMoveSinceLastRefine, onClick: onRefineClick - }, v(Template, _extends({}, templateProps, { + }, h(Template, _extends({}, templateProps, { templateKey: "redo", rootTagName: "span" - })))), !enableRefineControl && !isRefineOnMapMove && v("div", { + })))), !enableRefineControl && !isRefineOnMapMove && h("div", { className: cssClasses.control - }, v(GeoSearchButton, { - className: classnames(cssClasses.redo, _defineProperty({}, cssClasses.disabledRedo, !hasMapMoveSinceLastRefine)), + }, h(GeoSearchButton, { + className: cx(cssClasses.redo, !hasMapMoveSinceLastRefine && cssClasses.disabledRedo), disabled: !hasMapMoveSinceLastRefine, onClick: onRefineClick - }, v(Template, _extends({}, templateProps, { + }, h(Template, _extends({}, templateProps, { templateKey: "redo", rootTagName: "span" - })))), enableClearMapRefinement && isRefinedWithMap && v(GeoSearchButton, { + })))), enableClearMapRefinement && isRefinedWithMap && h(GeoSearchButton, { className: cssClasses.reset, onClick: onClearClick - }, v(Template, _extends({}, templateProps, { + }, h(Template, _extends({}, templateProps, { templateKey: "reset", rootTagName: "span" }))))); @@ -16706,7 +17113,7 @@ }); } - S(v(GeoSearchControls, { + P(h(GeoSearchControls, { cssClasses: cssClasses, enableRefine: enableRefine, enableRefineControl: enableRefineControl, @@ -16726,14 +17133,23 @@ }), container.querySelector(".".concat(cssClasses.tree))); }; + var _ref = h("p", null, "Your custom HTML Marker"); + var defaultTemplates$3 = { - HTMLMarker: '

Your custom HTML Marker

', - reset: 'Clear the map refinement', - toggle: 'Search as I move the map', - redo: 'Redo search here' + HTMLMarker: function HTMLMarker() { + return _ref; + }, + reset: function reset() { + return 'Clear the map refinement'; + }, + toggle: function toggle() { + return 'Search as I move the map'; + }, + redo: function redo() { + return 'Redo search here'; + } }; - /* global google EventListener */ var createHTMLMarker = function createHTMLMarker(googleReference) { var HTMLMarker = /*#__PURE__*/function (_googleReference$maps) { _inherits(HTMLMarker, _googleReference$maps); @@ -16777,7 +17193,12 @@ _this.element = document.createElement('div'); _this.element.className = className; _this.element.style.position = 'absolute'; - _this.element.innerHTML = template; + + if (_typeof(template) === 'object') { + P(template, _this.element); + } else { + _this.element.innerHTML = template; + } _this.setMap(map); @@ -16920,35 +17341,35 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$9(), userCssClasses.root), + root: cx(suit$9(), userCssClasses.root), // Required only to mount / unmount the Preact tree tree: suit$9({ descendantName: 'tree' }), - map: classnames(suit$9({ + map: cx(suit$9({ descendantName: 'map' }), userCssClasses.map), - control: classnames(suit$9({ + control: cx(suit$9({ descendantName: 'control' }), userCssClasses.control), - label: classnames(suit$9({ + label: cx(suit$9({ descendantName: 'label' }), userCssClasses.label), - selectedLabel: classnames(suit$9({ + selectedLabel: cx(suit$9({ descendantName: 'label', modifierName: 'selected' }), userCssClasses.selectedLabel), - input: classnames(suit$9({ + input: cx(suit$9({ descendantName: 'input' }), userCssClasses.input), - redo: classnames(suit$9({ + redo: cx(suit$9({ descendantName: 'redo' }), userCssClasses.redo), - disabledRedo: classnames(suit$9({ + disabledRedo: cx(suit$9({ descendantName: 'redo', modifierName: 'disabled' }), userCssClasses.disabledRedo), - reset: classnames(suit$9({ + reset: cx(suit$9({ descendantName: 'reset' }), userCssClasses.reset) }; @@ -16981,7 +17402,7 @@ return new HTMLMarker(_objectSpread2(_objectSpread2(_objectSpread2({}, customHTMLMarker.createOptions(item)), rest), {}, { __id: item.objectID, position: item._geoloc, - className: classnames(suit$9({ + className: cx(suit$9({ descendantName: 'marker' })), template: renderTemplate({ @@ -16995,7 +17416,7 @@ var createMarker = !customHTMLMarker ? createBuiltInMarker : createCustomHTMLMarker; var markerOptions = !customHTMLMarker ? builtInMarker : customHTMLMarker; var makeWidget = connectGeoSearch(renderer$4, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget(_objectSpread2(_objectSpread2({}, otherWidgetParams), {}, { renderState: {}, @@ -17024,7 +17445,7 @@ templateKey = _ref.templateKey, templateData = _ref.templateData, subItems = _ref.subItems; - return v("li", { + return h("li", { className: className, onClick: function onClick(originalEvent) { handleClick({ @@ -17033,7 +17454,7 @@ originalEvent: originalEvent }); } - }, v(Template, _extends({}, templateProps, { + }, h(Template, _extends({}, templateProps, { templateKey: templateKey, data: templateData })), subItems); @@ -17075,7 +17496,7 @@ focused: false }); - _defineProperty(_assertThisInitialized(_this), "input", p()); + _defineProperty(_assertThisInitialized(_this), "input", y()); _defineProperty(_assertThisInitialized(_this), "onInput", function (event) { var _this$props = _this.props, @@ -17190,16 +17611,16 @@ showLoadingIndicator = _this$props4.showLoadingIndicator, templates = _this$props4.templates, isSearchStalled = _this$props4.isSearchStalled; - return v("div", { + return h("div", { className: cssClasses.root - }, v("form", { + }, h("form", { action: "", role: "search", className: cssClasses.form, noValidate: true, onSubmit: this.onSubmit, onReset: this.onReset - }, v("input", { + }, h("input", { ref: this.input, value: this.state.query, disabled: this.props.disabled, @@ -17216,7 +17637,7 @@ onInput: this.onInput, onBlur: this.onBlur, onFocus: this.onFocus - }), v(Template, { + }), h(Template, { templateKey: "submit", rootTagName: "button", rootProps: { @@ -17229,7 +17650,7 @@ data: { cssClasses: cssClasses } - }), v(Template, { + }), h(Template, { templateKey: "reset", rootTagName: "button", rootProps: { @@ -17242,7 +17663,7 @@ data: { cssClasses: cssClasses } - }), showLoadingIndicator && v(Template, { + }), showLoadingIndicator && h(Template, { templateKey: "loadingIndicator", rootTagName: "span", rootProps: { @@ -17258,7 +17679,7 @@ }]); return SearchBox; - }(_); + }(d); _defineProperty(SearchBox, "defaultProps", defaultProps$1); @@ -17283,7 +17704,7 @@ _this = _super.call(this, props); - _defineProperty(_assertThisInitialized(_this), "searchBox", p()); + _defineProperty(_assertThisInitialized(_this), "searchBox", y()); _this.handleItemClick = _this.handleItemClick.bind(_assertThisInitialized(_this)); return _this; @@ -17303,8 +17724,6 @@ }, { key: "_generateFacetItem", value: function _generateFacetItem(facetValue) { - var _cx; - var subItems; if (isHierarchicalMenuItem(facetValue) && Array.isArray(facetValue.data) && facetValue.data.length > 0) { @@ -17312,7 +17731,7 @@ root = _this$props$cssClasse.root, cssClasses = _objectWithoutProperties(_this$props$cssClasse, ["root"]); - subItems = v(RefinementList, _extends({}, this.props, { + subItems = h(RefinementList, _extends({}, this.props, { // We want to keep `root` required for external usage but not for the // sub items. cssClasses: cssClasses, @@ -17342,8 +17761,8 @@ key += "/".concat(facetValue.count); } - var refinementListItemClassName = classnames(this.props.cssClasses.item, (_cx = {}, _defineProperty(_cx, this.props.cssClasses.selectedItem, facetValue.isRefined), _defineProperty(_cx, this.props.cssClasses.disabledItem, !facetValue.count), _defineProperty(_cx, this.props.cssClasses.parentItem, isHierarchicalMenuItem(facetValue) && Array.isArray(facetValue.data) && facetValue.data.length > 0), _cx)); - return v(RefinementListItem, { + var refinementListItemClassName = cx(this.props.cssClasses.item, facetValue.isRefined && this.props.cssClasses.selectedItem, !facetValue.count && this.props.cssClasses.disabledItem, Boolean(isHierarchicalMenuItem(facetValue) && Array.isArray(facetValue.data) && facetValue.data.length > 0) && this.props.cssClasses.parentItem); + return h(RefinementListItem, { templateKey: "item", key: key, facetValueToRefine: facetValue.value, @@ -17436,8 +17855,8 @@ value: function render() { var _this2 = this; - var showMoreButtonClassName = classnames(this.props.cssClasses.showMore, _defineProperty({}, this.props.cssClasses.disabledShowMore, !(this.props.showMore === true && this.props.canToggleShowMore))); - var showMoreButton = this.props.showMore === true && v(Template, _extends({}, this.props.templateProps, { + var showMoreButtonClassName = cx(this.props.cssClasses.showMore, !(this.props.showMore === true && this.props.canToggleShowMore) && this.props.cssClasses.disabledShowMore); + var showMoreButton = this.props.showMore === true && h(Template, _extends({}, this.props.templateProps, { templateKey: "showMoreText", rootTagName: "button", rootProps: { @@ -17450,9 +17869,9 @@ } })); var shouldDisableSearchBox = this.props.searchIsAlwaysActive !== true && !(this.props.isFromSearch || !this.props.hasExhaustiveItems); - var searchBox = this.props.searchFacetValues && v("div", { + var searchBox = this.props.searchFacetValues && h("div", { className: this.props.cssClasses.searchBox - }, v(SearchBox, { + }, h(SearchBox, { ref: this.searchBox, placeholder: this.props.searchPlaceholder, disabled: shouldDisableSearchBox, @@ -17471,30 +17890,47 @@ , searchAsYouType: false })); - var facetValues = this.props.facetValues && this.props.facetValues.length > 0 && v("ul", { + var facetValues = this.props.facetValues && this.props.facetValues.length > 0 && h("ul", { className: this.props.cssClasses.list }, this.props.facetValues.map(this._generateFacetItem, this)); - var noResults = this.props.searchFacetValues && this.props.isFromSearch && (!this.props.facetValues || this.props.facetValues.length === 0) && v(Template, _extends({}, this.props.templateProps, { + var noResults = this.props.searchFacetValues && this.props.isFromSearch && (!this.props.facetValues || this.props.facetValues.length === 0) && h(Template, _extends({}, this.props.templateProps, { templateKey: "searchableNoResults", rootProps: { className: this.props.cssClasses.noResults } })); - var rootClassName = classnames(this.props.cssClasses.root, _defineProperty({}, this.props.cssClasses.noRefinementRoot, !this.props.facetValues || this.props.facetValues.length === 0), this.props.className); - return v("div", { + var rootClassName = cx(this.props.cssClasses.root, (!this.props.facetValues || this.props.facetValues.length === 0) && this.props.cssClasses.noRefinementRoot, this.props.className); + return h("div", { className: rootClassName }, this.props.children, searchBox, facetValues, noResults, showMoreButton); } }]); return RefinementList; - }(_); + }(d); _defineProperty(RefinementList$1, "defaultProps", defaultProps$2); var defaultTemplates$4 = { - item: '' + '{{label}}' + '{{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}}' + '', - showMoreText: "\n {{#isShowingMore}}\n Show less\n {{/isShowingMore}}\n {{^isShowingMore}}\n Show more\n {{/isShowingMore}}\n " + item: function item(_ref) { + var url = _ref.url, + label = _ref.label, + count = _ref.count, + cssClasses = _ref.cssClasses, + isRefined = _ref.isRefined; + return h("a", { + className: cx(cx(cssClasses.link), cx(isRefined ? cssClasses.selectedItemLink : undefined)), + href: url + }, h("span", { + className: cx(cssClasses.label) + }, label), h("span", { + className: cx(cssClasses.count) + }, formatNumber(count))); + }, + showMoreText: function showMoreText(_ref2) { + var isShowingMore = _ref2.isShowingMore; + return isShowingMore ? 'Show less' : 'Show more'; + } }; var withUsage$z = createDocumentationMessageGenerator({ @@ -17526,7 +17962,7 @@ return; } - S(v(RefinementList$1, { + P(h(RefinementList$1, { createURL: createURL, cssClasses: cssClasses, facetValues: items, @@ -17616,41 +18052,45 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$a(), userCssClasses.root), - noRefinementRoot: classnames(suit$a({ + root: cx(suit$a(), userCssClasses.root), + noRefinementRoot: cx(suit$a({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: classnames(suit$a({ + list: cx(suit$a({ descendantName: 'list' }), userCssClasses.list), - childList: classnames(suit$a({ + childList: cx(suit$a({ descendantName: 'list', modifierName: 'child' }), userCssClasses.childList), - item: classnames(suit$a({ + item: cx(suit$a({ descendantName: 'item' }), userCssClasses.item), - selectedItem: classnames(suit$a({ + selectedItem: cx(suit$a({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - parentItem: classnames(suit$a({ + parentItem: cx(suit$a({ descendantName: 'item', modifierName: 'parent' }), userCssClasses.parentItem), - link: classnames(suit$a({ + link: cx(suit$a({ descendantName: 'link' }), userCssClasses.link), - label: classnames(suit$a({ + selectedItemLink: cx(suit$a({ + descendantName: 'link', + modifierName: 'selected' + }), userCssClasses.selectedItemLink), + label: cx(suit$a({ descendantName: 'label' }), userCssClasses.label), - count: classnames(suit$a({ + count: cx(suit$a({ descendantName: 'count' }), userCssClasses.count), - showMore: classnames(suit$a({ + showMore: cx(suit$a({ descendantName: 'showMore' }), userCssClasses.showMore), - disabledShowMore: classnames(suit$a({ + disabledShowMore: cx(suit$a({ descendantName: 'showMore', modifierName: 'disabled' }), userCssClasses.disabledShowMore) @@ -17663,7 +18103,7 @@ renderState: {} }); var makeWidget = connectHierarchicalMenu(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attributes: attributes, @@ -17684,25 +18124,26 @@ var results = _ref.results, hits = _ref.hits, bindEvent = _ref.bindEvent, + sendEvent = _ref.sendEvent, cssClasses = _ref.cssClasses, templateProps = _ref.templateProps; if (results.hits.length === 0) { - return v(Template, _extends({}, templateProps, { + return h(Template, _extends({}, templateProps, { templateKey: "empty", rootProps: { - className: classnames(cssClasses.root, cssClasses.emptyRoot) + className: cx(cssClasses.root, cssClasses.emptyRoot) }, data: results })); } - return v("div", { + return h("div", { className: cssClasses.root - }, v("ol", { + }, h("ol", { className: cssClasses.list }, hits.map(function (hit, index) { - return v(Template, _extends({}, templateProps, { + return h(Template, _extends({}, templateProps, { templateKey: "item", rootTagName: "li", rootProps: { @@ -17712,13 +18153,16 @@ data: _objectSpread2(_objectSpread2({}, hit), {}, { __hitIndex: index }), - bindEvent: bindEvent + bindEvent: bindEvent, + sendEvent: sendEvent })); }))); }; var defaultTemplates$5 = { - empty: 'No results', + empty: function empty() { + return 'No results'; + }, item: function item(data) { return JSON.stringify(data, null, 2); } @@ -17751,7 +18195,7 @@ return; } - S(v(HitsWithInsightsListener, { + P(h(HitsWithInsightsListener, { cssClasses: cssClasses, hits: receivedHits, results: results, @@ -17781,14 +18225,14 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$b(), userCssClasses.root), - emptyRoot: classnames(suit$b({ + root: cx(suit$b(), userCssClasses.root), + emptyRoot: cx(suit$b({ modifierName: 'empty' }), userCssClasses.emptyRoot), - list: classnames(suit$b({ + list: cx(suit$b({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$b({ + item: cx(suit$b({ descendantName: 'item' }), userCssClasses.item) }; @@ -17799,7 +18243,7 @@ templates: templates }); var makeWidget = withInsights(connectHits)(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ escapeHTML: escapeHTML, @@ -17809,22 +18253,20 @@ }); }; - /** @jsx h */ - function Selector(_ref) { var currentValue = _ref.currentValue, options = _ref.options, cssClasses = _ref.cssClasses, setValue = _ref.setValue; - return v("select", { - className: classnames(cssClasses.select), + return h("select", { + className: cx(cssClasses.select), onChange: function onChange(event) { return setValue(event.target.value); }, value: "".concat(currentValue) }, options.map(function (option) { - return v("option", { - className: classnames(cssClasses.option), + return h("option", { + className: cx(cssClasses.option), key: option.label + option.value, value: "".concat(option.value) }, option.label); @@ -17850,9 +18292,9 @@ }) || {}, currentValue = _ref3.value; - S(v("div", { + P(h("div", { className: cssClasses.root - }, v(Selector, { + }, h(Selector, { cssClasses: cssClasses, currentValue: currentValue, options: items, @@ -17875,11 +18317,11 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$c(), userCssClasses.root), - select: classnames(suit$c({ + root: cx(suit$c(), userCssClasses.root), + select: cx(suit$c({ descendantName: 'select' }), userCssClasses.select), - option: classnames(suit$c({ + option: cx(suit$c({ descendantName: 'option' }), userCssClasses.option) }; @@ -17888,7 +18330,7 @@ cssClasses: cssClasses }); var makeWidget = connectHitsPerPage(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ items: items, @@ -17902,6 +18344,7 @@ var results = _ref.results, hits = _ref.hits, bindEvent = _ref.bindEvent, + sendEvent = _ref.sendEvent, hasShowPrevious = _ref.hasShowPrevious, showPrevious = _ref.showPrevious, showMore = _ref.showMore, @@ -17911,29 +18354,29 @@ templateProps = _ref.templateProps; if (results.hits.length === 0) { - return v(Template, _extends({}, templateProps, { + return h(Template, _extends({}, templateProps, { templateKey: "empty", rootProps: { - className: classnames(cssClasses.root, cssClasses.emptyRoot) + className: cx(cssClasses.root, cssClasses.emptyRoot) }, data: results })); } - return v("div", { + return h("div", { className: cssClasses.root - }, hasShowPrevious && v(Template, _extends({}, templateProps, { + }, hasShowPrevious && h(Template, _extends({}, templateProps, { templateKey: "showPreviousText", rootTagName: "button", rootProps: { - className: classnames(cssClasses.loadPrevious, _defineProperty({}, cssClasses.disabledLoadPrevious, isFirstPage)), + className: cx(cssClasses.loadPrevious, isFirstPage && cssClasses.disabledLoadPrevious), disabled: isFirstPage, onClick: showPrevious } - })), v("ol", { + })), h("ol", { className: cssClasses.list }, hits.map(function (hit, position) { - return v(Template, _extends({}, templateProps, { + return h(Template, _extends({}, templateProps, { templateKey: "item", rootTagName: "li", rootProps: { @@ -17943,13 +18386,14 @@ data: _objectSpread2(_objectSpread2({}, hit), {}, { __hitIndex: position }), - bindEvent: bindEvent + bindEvent: bindEvent, + sendEvent: sendEvent })); - })), v(Template, _extends({}, templateProps, { + })), h(Template, _extends({}, templateProps, { templateKey: "showMoreText", rootTagName: "button", rootProps: { - className: classnames(cssClasses.loadMore, _defineProperty({}, cssClasses.disabledLoadMore, isLastPage)), + className: cx(cssClasses.loadMore, isLastPage && cssClasses.disabledLoadMore), disabled: isLastPage, onClick: showMore } @@ -17957,9 +18401,15 @@ }; var defaultTemplates$6 = { - empty: 'No results', - showPreviousText: 'Show previous results', - showMoreText: 'Show more results', + empty: function empty() { + return 'No results'; + }, + showPreviousText: function showPreviousText() { + return 'Show previous results'; + }, + showMoreText: function showMoreText() { + return 'Show more results'; + }, item: function item(data) { return JSON.stringify(data, null, 2); } @@ -17997,7 +18447,7 @@ return; } - S(v(InfiniteHitsWithInsightsListener, { + P(h(InfiniteHitsWithInsightsListener, { cssClasses: cssClasses, hits: hits, results: results, @@ -18034,27 +18484,27 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$d(), userCssClasses.root), - emptyRoot: classnames(suit$d({ + root: cx(suit$d(), userCssClasses.root), + emptyRoot: cx(suit$d({ modifierName: 'empty' }), userCssClasses.emptyRoot), - item: classnames(suit$d({ + item: cx(suit$d({ descendantName: 'item' }), userCssClasses.item), - list: classnames(suit$d({ + list: cx(suit$d({ descendantName: 'list' }), userCssClasses.list), - loadPrevious: classnames(suit$d({ + loadPrevious: cx(suit$d({ descendantName: 'loadPrevious' }), userCssClasses.loadPrevious), - disabledLoadPrevious: classnames(suit$d({ + disabledLoadPrevious: cx(suit$d({ descendantName: 'loadPrevious', modifierName: 'disabled' }), userCssClasses.disabledLoadPrevious), - loadMore: classnames(suit$d({ + loadMore: cx(suit$d({ descendantName: 'loadMore' }), userCssClasses.loadMore), - disabledLoadMore: classnames(suit$d({ + disabledLoadMore: cx(suit$d({ descendantName: 'loadMore', modifierName: 'disabled' }), userCssClasses.disabledLoadMore) @@ -18067,7 +18517,7 @@ renderState: {} }); var makeWidget = withInsights(connectInfiniteHits)(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ escapeHTML: escapeHTML, @@ -18080,8 +18530,24 @@ }; var defaultTemplates$7 = { - item: '' + '{{label}}' + '{{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}}' + '', - showMoreText: "\n {{#isShowingMore}}\n Show less\n {{/isShowingMore}}\n {{^isShowingMore}}\n Show more\n {{/isShowingMore}}\n " + item: function item(_ref) { + var cssClasses = _ref.cssClasses, + url = _ref.url, + label = _ref.label, + count = _ref.count; + return h("a", { + className: cx(cssClasses.link), + href: url + }, h("span", { + className: cx(cssClasses.label) + }, label), h("span", { + className: cx(cssClasses.count) + }, formatNumber(count))); + }, + showMoreText: function showMoreText(_ref2) { + var isShowingMore = _ref2.isShowingMore; + return isShowingMore ? 'Show less' : 'Show more'; + } }; var withUsage$D = createDocumentationMessageGenerator({ @@ -18118,7 +18584,7 @@ url: createURL(facetValue.value) }); }); - S(v(RefinementList$1, { + P(h(RefinementList$1, { createURL: createURL, cssClasses: cssClasses, facetValues: facetValues, @@ -18152,33 +18618,33 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$e(), userCssClasses.root), - noRefinementRoot: classnames(suit$e({ + root: cx(suit$e(), userCssClasses.root), + noRefinementRoot: cx(suit$e({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: classnames(suit$e({ + list: cx(suit$e({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$e({ + item: cx(suit$e({ descendantName: 'item' }), userCssClasses.item), - selectedItem: classnames(suit$e({ + selectedItem: cx(suit$e({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - link: classnames(suit$e({ + link: cx(suit$e({ descendantName: 'link' }), userCssClasses.link), - label: classnames(suit$e({ + label: cx(suit$e({ descendantName: 'label' }), userCssClasses.label), - count: classnames(suit$e({ + count: cx(suit$e({ descendantName: 'count' }), userCssClasses.count), - showMore: classnames(suit$e({ + showMore: cx(suit$e({ descendantName: 'showMore' }), userCssClasses.showMore), - disabledShowMore: classnames(suit$e({ + disabledShowMore: cx(suit$e({ descendantName: 'showMore', modifierName: 'disabled' }), userCssClasses.disabledShowMore) @@ -18191,7 +18657,7 @@ showMore: showMore }); var makeWidget = connectMenu(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -18218,15 +18684,15 @@ }, selectedValue = _ref2.value; - return v("div", { - className: classnames(cssClasses.root, _defineProperty({}, cssClasses.noRefinementRoot, items.length === 0)) - }, v("select", { + return h("div", { + className: cx(cssClasses.root, items.length === 0 && cssClasses.noRefinementRoot) + }, h("select", { className: cssClasses.select, value: selectedValue, onChange: function onChange(event) { refine(event.target.value); } - }, v(Template, _extends({}, templateProps, { + }, h(Template, _extends({}, templateProps, { templateKey: "defaultOption", rootTagName: "option", rootProps: { @@ -18234,7 +18700,7 @@ className: cssClasses.option } })), items.map(function (item) { - return v(Template, _extends({}, templateProps, { + return h(Template, _extends({}, templateProps, { templateKey: "item", rootTagName: "option", rootProps: { @@ -18248,8 +18714,14 @@ } var defaultTemplates$8 = { - item: '{{label}} ({{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}})', - defaultOption: 'See all' + item: function item(_ref) { + var label = _ref.label, + count = _ref.count; + return "".concat(label, " (").concat(formatNumber(count), ")"); + }, + defaultOption: function defaultOption() { + return 'See all'; + } }; var withUsage$E = createDocumentationMessageGenerator({ @@ -18276,7 +18748,7 @@ return; } - S(v(MenuSelect, { + P(h(MenuSelect, { cssClasses: cssClasses, items: items, refine: refine, @@ -18305,14 +18777,14 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$f(), userCssClasses.root), - noRefinementRoot: classnames(suit$f({ + root: cx(suit$f(), userCssClasses.root), + noRefinementRoot: cx(suit$f({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - select: classnames(suit$f({ + select: cx(suit$f({ descendantName: 'select' }), userCssClasses.select), - option: classnames(suit$f({ + option: cx(suit$f({ descendantName: 'option' }), userCssClasses.option) }; @@ -18323,7 +18795,7 @@ templates: templates }); var makeWidget = connectMenu(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -18336,7 +18808,22 @@ }; var defaultTemplates$9 = { - item: "" + item: function item(_ref) { + var cssClasses = _ref.cssClasses, + attribute = _ref.attribute, + label = _ref.label, + isRefined = _ref.isRefined; + return h("label", { + className: cssClasses.label + }, h("input", { + type: "radio", + className: cssClasses.radio, + name: attribute, + defaultChecked: isRefined + }), h("span", { + className: cssClasses.labelText + }, label)); + } }; var withUsage$F = createDocumentationMessageGenerator({ @@ -18365,7 +18852,7 @@ return; } - S(v(RefinementList$1, { + P(h(RefinementList$1, { createURL: createURL, cssClasses: cssClasses, facetValues: items, @@ -18393,27 +18880,27 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$g(), userCssClasses.root), - noRefinementRoot: classnames(suit$g({ + root: cx(suit$g(), userCssClasses.root), + noRefinementRoot: cx(suit$g({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: classnames(suit$g({ + list: cx(suit$g({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$g({ + item: cx(suit$g({ descendantName: 'item' }), userCssClasses.item), - selectedItem: classnames(suit$g({ + selectedItem: cx(suit$g({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - label: classnames(suit$g({ + label: cx(suit$g({ descendantName: 'label' }), userCssClasses.label), - radio: classnames(suit$g({ + radio: cx(suit$g({ descendantName: 'radio' }), userCssClasses.radio), - labelText: classnames(suit$g({ + labelText: cx(suit$g({ descendantName: 'labelText' }), userCssClasses.labelText) }; @@ -18425,7 +18912,7 @@ templates: templates }); var makeWidget = connectNumericMenu(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -18450,11 +18937,11 @@ }; } - return v("div", { - className: classnames(props.cssClasses.root, _defineProperty({}, props.cssClasses.noRefinementRoot, props.nbPages <= 1)) - }, v("ul", { + return h("div", { + className: cx(props.cssClasses.root, props.nbPages <= 1 && props.cssClasses.noRefinementRoot) + }, h("ul", { className: props.cssClasses.list - }, props.showFirst && v(PaginationLink, { + }, props.showFirst && h(PaginationLink, { ariaLabel: "First", className: props.cssClasses.firstPageItem, isDisabled: props.isFirstPage, @@ -18463,7 +18950,7 @@ createURL: props.createURL, cssClasses: props.cssClasses, createClickHandler: createClickHandler - }), props.showPrevious && v(PaginationLink, { + }), props.showPrevious && h(PaginationLink, { ariaLabel: "Previous", className: props.cssClasses.previousPageItem, isDisabled: props.isFirstPage, @@ -18473,7 +18960,7 @@ cssClasses: props.cssClasses, createClickHandler: createClickHandler }), props.pages.map(function (pageNumber) { - return v(PaginationLink, { + return h(PaginationLink, { key: pageNumber, ariaLabel: "".concat(pageNumber + 1), className: props.cssClasses.pageItem, @@ -18484,7 +18971,7 @@ cssClasses: props.cssClasses, createClickHandler: createClickHandler }); - }), props.showNext && v(PaginationLink, { + }), props.showNext && h(PaginationLink, { ariaLabel: "Next", className: props.cssClasses.nextPageItem, isDisabled: props.isLastPage, @@ -18493,7 +18980,7 @@ createURL: props.createURL, cssClasses: props.cssClasses, createClickHandler: createClickHandler - }), props.showLast && v(PaginationLink, { + }), props.showLast && h(PaginationLink, { ariaLabel: "Last", className: props.cssClasses.lastPageItem, isDisabled: props.isLastPage, @@ -18517,14 +19004,14 @@ cssClasses = _ref.cssClasses, createURL = _ref.createURL, createClickHandler = _ref.createClickHandler; - return v("li", { - className: classnames(cssClasses.item, className, isDisabled && cssClasses.disabledItem, isSelected && cssClasses.selectedItem) - }, isDisabled ? v("span", { + return h("li", { + className: cx(cssClasses.item, className, isDisabled && cssClasses.disabledItem, isSelected && cssClasses.selectedItem) + }, isDisabled ? h("span", { className: cssClasses.link, dangerouslySetInnerHTML: { __html: label } - }) : v("a", { + }) : h("a", { className: cssClasses.link, "aria-label": ariaLabel, href: createURL(pageNumber), @@ -18573,7 +19060,7 @@ } }; - S(v(Pagination, { + P(h(Pagination, { createURL: createURL, cssClasses: cssClasses, currentPage: currentRefinement, @@ -18619,45 +19106,45 @@ var scrollTo = userScrollTo === true ? 'body' : userScrollTo; var scrollToNode = scrollTo !== false ? getContainerNode(scrollTo) : false; var cssClasses = { - root: classnames(suit$h(), userCssClasses.root), - noRefinementRoot: classnames(suit$h({ + root: cx(suit$h(), userCssClasses.root), + noRefinementRoot: cx(suit$h({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: classnames(suit$h({ + list: cx(suit$h({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$h({ + item: cx(suit$h({ descendantName: 'item' }), userCssClasses.item), - firstPageItem: classnames(suit$h({ + firstPageItem: cx(suit$h({ descendantName: 'item', modifierName: 'firstPage' }), userCssClasses.firstPageItem), - lastPageItem: classnames(suit$h({ + lastPageItem: cx(suit$h({ descendantName: 'item', modifierName: 'lastPage' }), userCssClasses.lastPageItem), - previousPageItem: classnames(suit$h({ + previousPageItem: cx(suit$h({ descendantName: 'item', modifierName: 'previousPage' }), userCssClasses.previousPageItem), - nextPageItem: classnames(suit$h({ + nextPageItem: cx(suit$h({ descendantName: 'item', modifierName: 'nextPage' }), userCssClasses.nextPageItem), - pageItem: classnames(suit$h({ + pageItem: cx(suit$h({ descendantName: 'item', modifierName: 'page' }), userCssClasses.pageItem), - selectedItem: classnames(suit$h({ + selectedItem: cx(suit$h({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - disabledItem: classnames(suit$h({ + disabledItem: cx(suit$h({ descendantName: 'item', modifierName: 'disabled' }), userCssClasses.disabledItem), - link: classnames(suit$h({ + link: cx(suit$h({ descendantName: 'link' }), userCssClasses.link) }; @@ -18675,7 +19162,7 @@ scrollToNode: scrollToNode }); var makeWidget = connectPagination(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ totalPages: totalPages, @@ -18685,23 +19172,21 @@ }); }; - var t$1,u$1,r$1,o$1=0,i=[],c$1=l.__b,f$1=l.__r,e$1=l.diffed,a$1=l.__c,v$1=l.unmount;function m$1(t,r){l.__h&&l.__h(u$1,t,o$1||r),o$1=0;var i=u$1.__H||(u$1.__H={__:[],__h:[]});return t>=i.__.length&&i.__.push({}),i.__[t]}function l$1(n){return o$1=1,p$1(w$1,n)}function p$1(n,r,o){var i=m$1(t$1++,2);return i.t=n,i.__c||(i.__=[o?o(r):w$1(void 0,r),function(n){var t=i.t(i.__[0],n);i.__[0]!==t&&(i.__=[t,i.__[1]],i.__c.setState({}));}],i.__c=u$1),i.__}function y$1(r,o){var i=m$1(t$1++,3);!l.__s&&k$1(i.__H,o)&&(i.__=r,i.__H=o,u$1.__H.__h.push(i));}function s$1(n){return o$1=5,d$1(function(){return {current:n}},[])}function d$1(n,u){var r=m$1(t$1++,7);return k$1(r.__H,u)&&(r.__=n(),r.__H=u,r.__h=n),r.__}function x$1(){var t;for(i.sort(function(n,t){return n.__v.__b-t.__v.__b});t=i.pop();)if(t.__P)try{t.__H.__h.forEach(g$1),t.__H.__h.forEach(j$1),t.__H.__h=[];}catch(u){t.__H.__h=[],l.__e(u,t.__v);}}l.__b=function(n){u$1=null,c$1&&c$1(n);},l.__r=function(n){f$1&&f$1(n),t$1=0;var r=(u$1=n.__c).__H;r&&(r.__h.forEach(g$1),r.__h.forEach(j$1),r.__h=[]);},l.diffed=function(t){e$1&&e$1(t);var o=t.__c;o&&o.__H&&o.__H.__h.length&&(1!==i.push(o)&&r$1===l.requestAnimationFrame||((r$1=l.requestAnimationFrame)||function(n){var t,u=function(){clearTimeout(r),b$1&&cancelAnimationFrame(t),setTimeout(n);},r=setTimeout(u,100);b$1&&(t=requestAnimationFrame(u));})(x$1)),u$1=null;},l.__c=function(t,u){u.some(function(t){try{t.__h.forEach(g$1),t.__h=t.__h.filter(function(n){return !n.__||j$1(n)});}catch(r){u.some(function(n){n.__h&&(n.__h=[]);}),u=[],l.__e(r,t.__v);}}),a$1&&a$1(t,u);},l.unmount=function(t){v$1&&v$1(t);var u,r=t.__c;r&&r.__H&&(r.__H.__.forEach(function(n){try{g$1(n);}catch(n){u=n;}}),u&&l.__e(u,r.__v));};var b$1="function"==typeof requestAnimationFrame;function g$1(n){var t=u$1,r=n.__c;"function"==typeof r&&(n.__c=void 0,r()),u$1=t;}function j$1(n){var t=u$1;n.__c=n.__(),u$1=t;}function k$1(n,t){return !n||n.length!==t.length||t.some(function(t,u){return t!==n[u]})}function w$1(n,t){return "function"==typeof t?t(n):t} + var t$2,r$1,u$1,i,o$1=0,c$1=[],f$1=[],e$2=l.__b,a$1=l.__r,v$1=l.diffed,l$1=l.__c,m$2=l.unmount;function d$1(t,u){l.__h&&l.__h(r$1,t,o$1||u),o$1=0;var i=r$1.__H||(r$1.__H={__:[],__h:[]});return t>=i.__.length&&i.__.push({__V:f$1}),i.__[t]}function p$1(n){return o$1=1,y$1(z$1,n)}function y$1(n,u,i){var o=d$1(t$2++,2);if(o.t=n,!o.__c&&(o.__=[i?i(u):z$1(void 0,u),function(n){var t=o.__N?o.__N[0]:o.__[0],r=o.t(t,n);t!==r&&(o.__N=[r,o.__[1]],o.__c.setState({}));}],o.__c=r$1,!r$1.u)){r$1.u=!0;var c=r$1.shouldComponentUpdate;r$1.shouldComponentUpdate=function(n,t,r){if(!o.__c.__H)return !0;var u=o.__c.__H.__.filter(function(n){return n.__c});return u.every(function(n){return !n.__N})?!c||c.call(this,n,t,r):!u.every(function(n){if(!n.__N)return !0;var t=n.__[0];return n.__=n.__N,n.__N=void 0,t===n.__[0]})&&(!c||c.call(this,n,t,r))};}return o.__N||o.__}function h$1(u,i){var o=d$1(t$2++,3);!l.__s&&w$1(o.__H,i)&&(o.__=u,o.i=i,r$1.__H.__h.push(o));}function _$1(n){return o$1=5,F(function(){return {current:n}},[])}function F(n,r){var u=d$1(t$2++,7);return w$1(u.__H,r)?(u.__V=n(),u.i=r,u.__h=n,u.__V):u.__}function b$1(){for(var t;t=c$1.shift();)if(t.__P&&t.__H)try{t.__H.__h.forEach(j$1),t.__H.__h.forEach(k$1),t.__H.__h=[];}catch(r){t.__H.__h=[],l.__e(r,t.__v);}}l.__b=function(n){r$1=null,e$2&&e$2(n);},l.__r=function(n){a$1&&a$1(n),t$2=0;var i=(r$1=n.__c).__H;i&&(u$1===r$1?(i.__h=[],r$1.__h=[],i.__.forEach(function(n){n.__N&&(n.__=n.__N),n.__V=f$1,n.__N=n.i=void 0;})):(i.__h.forEach(j$1),i.__h.forEach(k$1),i.__h=[])),u$1=r$1;},l.diffed=function(t){v$1&&v$1(t);var o=t.__c;o&&o.__H&&(o.__H.__h.length&&(1!==c$1.push(o)&&i===l.requestAnimationFrame||((i=l.requestAnimationFrame)||function(n){var t,r=function(){clearTimeout(u),g$1&&cancelAnimationFrame(t),setTimeout(n);},u=setTimeout(r,100);g$1&&(t=requestAnimationFrame(r));})(b$1)),o.__H.__.forEach(function(n){n.i&&(n.__H=n.i),n.__V!==f$1&&(n.__=n.__V),n.i=void 0,n.__V=f$1;})),u$1=r$1=null;},l.__c=function(t,r){r.some(function(t){try{t.__h.forEach(j$1),t.__h=t.__h.filter(function(n){return !n.__||k$1(n)});}catch(u){r.some(function(n){n.__h&&(n.__h=[]);}),r=[],l.__e(u,t.__v);}}),l$1&&l$1(t,r);},l.unmount=function(t){m$2&&m$2(t);var r,u=t.__c;u&&u.__H&&(u.__H.__.forEach(function(n){try{j$1(n);}catch(n){r=n;}}),r&&l.__e(r,u.__v));};var g$1="function"==typeof requestAnimationFrame;function j$1(n){var t=r$1,u=n.__c;"function"==typeof u&&(n.__c=void 0,u()),r$1=t;}function k$1(n){var t=r$1;n.__c=n.__(),r$1=t;}function w$1(n,t){return !n||n.length!==t.length||t.some(function(t,r){return t!==n[r]})}function z$1(n,t){return "function"==typeof t?t(n):t} function Panel(props) { - var _cx; - - var _useState = l$1(props.isCollapsed), + var _useState = p$1(props.isCollapsed), _useState2 = _slicedToArray(_useState, 2), isCollapsed = _useState2[0], setIsCollapsed = _useState2[1]; - var _useState3 = l$1(false), + var _useState3 = p$1(false), _useState4 = _slicedToArray(_useState3, 2), isControlled = _useState4[0], setIsControlled = _useState4[1]; - var bodyRef = s$1(null); - y$1(function () { + var bodyRef = _$1(null); + h$1(function () { var node = bodyRef.current; if (!node) { @@ -18718,17 +19203,17 @@ setIsCollapsed(props.isCollapsed); } - return v("div", { - className: classnames(props.cssClasses.root, (_cx = {}, _defineProperty(_cx, props.cssClasses.noRefinementRoot, props.hidden), _defineProperty(_cx, props.cssClasses.collapsibleRoot, props.collapsible), _defineProperty(_cx, props.cssClasses.collapsedRoot, isCollapsed), _cx)), + return h("div", { + className: cx(props.cssClasses.root, props.hidden && props.cssClasses.noRefinementRoot, props.collapsible && props.cssClasses.collapsibleRoot, isCollapsed && props.cssClasses.collapsedRoot), hidden: props.hidden - }, props.templates.header && v("div", { + }, props.templates.header && h("div", { className: props.cssClasses.header - }, v(Template, { + }, h(Template, { templates: props.templates, templateKey: "header", rootTagName: "span", data: props.data - }), props.collapsible && v("button", { + }), props.collapsible && h("button", { className: props.cssClasses.collapseButton, "aria-expanded": !isCollapsed, onClick: function onClick(event) { @@ -18738,17 +19223,17 @@ return !prevIsCollapsed; }); } - }, v(Template, { + }, h(Template, { templates: props.templates, templateKey: "collapseButtonText", rootTagName: "span", data: { collapsed: isCollapsed } - }))), v("div", { + }))), h("div", { className: props.cssClasses.body, ref: bodyRef - }), props.templates.footer && v(Template, { + }), props.templates.footer && h(Template, { templates: props.templates, templateKey: "footer", rootProps: { @@ -18773,7 +19258,7 @@ hidden = _ref2.hidden, collapsible = _ref2.collapsible, collapsed = _ref2.collapsed; - S(v(Panel, { + P(h(Panel, { cssClasses: cssClasses, hidden: hidden, collapsible: collapsible, @@ -18809,29 +19294,29 @@ return false; }; var cssClasses = { - root: classnames(suit$i(), userCssClasses.root), - noRefinementRoot: classnames(suit$i({ + root: cx(suit$i(), userCssClasses.root), + noRefinementRoot: cx(suit$i({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - collapsibleRoot: classnames(suit$i({ + collapsibleRoot: cx(suit$i({ modifierName: 'collapsible' }), userCssClasses.collapsibleRoot), - collapsedRoot: classnames(suit$i({ + collapsedRoot: cx(suit$i({ modifierName: 'collapsed' }), userCssClasses.collapsedRoot), - collapseButton: classnames(suit$i({ + collapseButton: cx(suit$i({ descendantName: 'collapseButton' }), userCssClasses.collapseButton), - collapseIcon: classnames(suit$i({ + collapseIcon: cx(suit$i({ descendantName: 'collapseIcon' }), userCssClasses.collapseIcon), - body: classnames(suit$i({ + body: cx(suit$i({ descendantName: 'body' }), userCssClasses.body), - header: classnames(suit$i({ + header: cx(suit$i({ descendantName: 'header' }), userCssClasses.header), - footer: classnames(suit$i({ + footer: cx(suit$i({ descendantName: 'footer' }), userCssClasses.footer) }; @@ -18909,7 +19394,7 @@ } }, dispose: function dispose() { - S(null, containerNode); + P(null, containerNode); if (typeof widget.dispose === 'function') { var _widget$dispose; @@ -19031,45 +19516,33 @@ }; }; - /** @jsx h */ - - var _ref2 = v("path", { - fill: "#5468FF", - d: "M78.99.94h16.6a2.97 2.97 0 012.96 2.96v16.6a2.97 2.97 0 01-2.97 2.96h-16.6a2.97 2.97 0 01-2.96-2.96V3.9A2.96 2.96 0 0179 .94" - }); - - var _ref3 = v("path", { - fill: "#FFF", - d: "M89.63 5.97v-.78a.98.98 0 00-.98-.97h-2.28a.98.98 0 00-.97.97V6c0 .09.08.15.17.13a7.13 7.13 0 013.9-.02c.08.02.16-.04.16-.13m-6.25 1L83 6.6a.98.98 0 00-1.38 0l-.46.46a.97.97 0 000 1.38l.38.39c.06.06.15.04.2-.02a7.49 7.49 0 011.63-1.62c.07-.04.08-.14.02-.2m4.16 2.45v3.34c0 .1.1.17.2.12l2.97-1.54c.06-.03.08-.12.05-.18a3.7 3.7 0 00-3.08-1.87c-.07 0-.14.06-.14.13m0 8.05a4.49 4.49 0 110-8.98 4.49 4.49 0 010 8.98m0-10.85a6.37 6.37 0 100 12.74 6.37 6.37 0 000-12.74" - }); - var PoweredBy = function PoweredBy(_ref) { var url = _ref.url, theme = _ref.theme, cssClasses = _ref.cssClasses; - return v("div", { + return h("div", { className: cssClasses.root - }, v("a", { + }, h("a", { href: url, target: "_blank", className: cssClasses.link, "aria-label": "Search by Algolia", rel: "noopener noreferrer" - }, v("svg", { + }, h("svg", { height: "1.2em", className: cssClasses.logo, - viewBox: "0 0 168 24" // This style is necessary as long as it's not included in InstantSearch.css. + viewBox: "0 0 572 64" // This style is necessary as long as it's not included in InstantSearch.css. // For now, InstantSearch.css sets a maximum width of 70px. , style: { width: 'auto' } - }, v("path", { - fill: theme === 'dark' ? '#FFF' : '#5D6494', - d: "M6.97 6.68V8.3a4.47 4.47 0 00-2.42-.67 2.2 2.2 0 00-1.38.4c-.34.26-.5.6-.5 1.02 0 .43.16.77.49 1.03.33.25.83.53 1.51.83a7.04 7.04 0 011.9 1.08c.34.24.58.54.73.89.15.34.23.74.23 1.18 0 .95-.33 1.7-1 2.24a4 4 0 01-2.6.81 5.71 5.71 0 01-2.94-.68v-1.71c.84.63 1.81.94 2.92.94.58 0 1.05-.14 1.39-.4.34-.28.5-.65.5-1.13 0-.29-.1-.55-.3-.8a2.2 2.2 0 00-.65-.53 23.03 23.03 0 00-1.64-.78 13.67 13.67 0 01-1.11-.64c-.12-.1-.28-.22-.46-.4a1.72 1.72 0 01-.39-.5 4.46 4.46 0 01-.22-.6c-.07-.23-.1-.48-.1-.75 0-.91.33-1.63 1-2.17a4 4 0 012.57-.8c.97 0 1.8.18 2.47.52zm7.47 5.7v-.3a2.26 2.26 0 00-.5-1.44c-.3-.35-.74-.53-1.32-.53-.53 0-.99.2-1.37.58a2.9 2.9 0 00-.72 1.68h3.91zm1 2.79v1.4c-.6.34-1.38.51-2.36.51a4.02 4.02 0 01-3-1.13 4.04 4.04 0 01-1.11-2.97c0-1.3.34-2.32 1.02-3.06a3.38 3.38 0 012.6-1.1c1.03 0 1.85.32 2.46.96.6.64.9 1.57.9 2.78 0 .33-.03.68-.09 1.04h-5.31c.1.7.4 1.24.89 1.61.49.38 1.1.56 1.85.56.86 0 1.58-.2 2.15-.6zm6.61-1.78h-1.21c-.6 0-1.05.12-1.35.36-.3.23-.46.53-.46.89 0 .37.12.66.36.88.23.2.57.32 1.02.32.5 0 .9-.15 1.2-.43.3-.28.44-.65.44-1.1v-.92zm-4.07-2.55V9.33a4.96 4.96 0 012.5-.55c2.1 0 3.17 1.03 3.17 3.08V17H22.1v-.96c-.42.68-1.15 1.02-2.19 1.02-.76 0-1.38-.22-1.84-.66-.46-.44-.7-1-.7-1.68 0-.78.3-1.38.88-1.81.59-.43 1.4-.65 2.46-.65h1.34v-.46c0-.55-.13-.97-.4-1.25-.26-.29-.7-.43-1.32-.43-.86 0-1.65.24-2.35.72zm9.34-1.93v1.42c.39-1 1.1-1.5 2.12-1.5.15 0 .31.02.5.05v1.53c-.23-.1-.48-.14-.76-.14-.54 0-.99.24-1.34.71a2.8 2.8 0 00-.52 1.71V17h-1.57V8.91h1.57zm5 4.09a3 3 0 00.76 2.01c.47.53 1.14.8 2 .8.64 0 1.24-.18 1.8-.53v1.4c-.53.32-1.2.48-2 .48a3.98 3.98 0 01-4.17-4.18c0-1.16.38-2.15 1.14-2.98a4 4 0 013.1-1.23c.7 0 1.34.15 1.92.44v1.44a3.24 3.24 0 00-1.77-.5A2.65 2.65 0 0032.33 13zm7.92-7.28v4.58c.46-1 1.3-1.5 2.5-1.5.8 0 1.42.24 1.9.73.48.5.72 1.17.72 2.05V17H43.8v-5.1c0-.56-.14-.99-.43-1.29-.28-.3-.65-.45-1.1-.45-.54 0-1 .2-1.42.6-.4.4-.61 1.02-.61 1.85V17h-1.56V5.72h1.56zM55.2 15.74c.6 0 1.1-.25 1.5-.76.4-.5.6-1.16.6-1.95 0-.92-.2-1.62-.6-2.12-.4-.5-.92-.74-1.55-.74-.56 0-1.05.22-1.5.67-.44.45-.66 1.13-.66 2.06 0 .96.22 1.67.64 2.14.43.47.95.7 1.57.7zM53 5.72v4.42a2.74 2.74 0 012.43-1.34c1.03 0 1.86.38 2.51 1.15.65.76.97 1.78.97 3.05 0 1.13-.3 2.1-.92 2.9-.62.81-1.47 1.21-2.54 1.21s-1.9-.45-2.46-1.34V17h-1.58V5.72H53zm9.9 11.1l-3.22-7.9h1.74l1 2.62 1.26 3.42c.1-.32.48-1.46 1.15-3.42l.91-2.63h1.66l-2.92 7.87c-.78 2.07-1.96 3.1-3.56 3.1-.28 0-.53-.02-.73-.07v-1.34c.17.04.35.06.54.06 1.03 0 1.76-.57 2.17-1.7z" - }), _ref2, _ref3, v("path", { - fill: theme === 'dark' ? '#FFF' : '#5468FF', - d: "M120.92 18.8c-4.38.02-4.38-3.54-4.38-4.1V1.36l2.67-.42v13.25c0 .32 0 2.36 1.71 2.37v2.24zm-10.84-2.18c.82 0 1.43-.04 1.85-.12v-2.72a5.48 5.48 0 00-1.57-.2c-.3 0-.6.02-.9.07-.3.04-.57.12-.81.24-.24.11-.44.28-.58.49a.93.93 0 00-.22.65c0 .63.22 1 .61 1.23.4.24.94.36 1.62.36zm-.23-9.7c.88 0 1.62.11 2.23.33.6.22 1.09.53 1.44.92.36.4.61.92.76 1.48.16.56.23 1.17.23 1.85v6.87a21.69 21.69 0 01-4.68.5c-.69 0-1.32-.07-1.9-.2a4 4 0 01-1.46-.63 3.3 3.3 0 01-.96-1.13 4.3 4.3 0 01-.34-1.8 3.13 3.13 0 011.43-2.63c.45-.3.95-.5 1.54-.62a8.8 8.8 0 013.79.05v-.44c0-.3-.04-.6-.11-.87a1.78 1.78 0 00-1.1-1.22 3.2 3.2 0 00-1.15-.2 9.75 9.75 0 00-2.95.46l-.33-2.19a11.43 11.43 0 013.56-.53zm52.84 9.63c.82 0 1.43-.05 1.85-.13V13.7a5.42 5.42 0 00-1.57-.2c-.3 0-.6.02-.9.07-.3.04-.57.12-.81.24-.24.12-.44.28-.58.5a.93.93 0 00-.22.65c0 .63.22.99.61 1.23.4.24.94.36 1.62.36zm-.23-9.7c.88 0 1.63.11 2.23.33.6.22 1.1.53 1.45.92.35.39.6.92.76 1.48.15.56.23 1.18.23 1.85v6.88c-.41.08-1.03.19-1.87.31-.83.12-1.77.18-2.81.18-.7 0-1.33-.06-1.9-.2a4 4 0 01-1.47-.63c-.4-.3-.72-.67-.95-1.13a4.3 4.3 0 01-.34-1.8c0-.66.13-1.08.38-1.53.26-.45.61-.82 1.05-1.1.44-.3.95-.5 1.53-.62a8.8 8.8 0 013.8.05v-.43c0-.31-.04-.6-.12-.88-.07-.28-.2-.52-.38-.73a1.78 1.78 0 00-.73-.5c-.3-.1-.68-.2-1.14-.2a9.85 9.85 0 00-2.95.47l-.32-2.19a11.63 11.63 0 013.55-.53zm-8.03-1.27a1.62 1.62 0 000-3.24 1.62 1.62 0 100 3.24zm1.35 13.22h-2.7V7.27l2.7-.42V18.8zm-4.72 0c-4.38.02-4.38-3.54-4.38-4.1l-.01-13.34 2.67-.42v13.25c0 .32 0 2.36 1.72 2.37v2.24zm-8.7-5.9a4.7 4.7 0 00-.74-2.79 2.4 2.4 0 00-2.07-1 2.4 2.4 0 00-2.06 1 4.7 4.7 0 00-.74 2.8c0 1.16.25 1.94.74 2.62a2.4 2.4 0 002.07 1.02c.88 0 1.57-.34 2.07-1.02a4.2 4.2 0 00.73-2.63zm2.74 0a6.46 6.46 0 01-1.52 4.23c-.49.53-1.07.94-1.76 1.22-.68.29-1.73.45-2.26.45a6.6 6.6 0 01-2.25-.45 5.1 5.1 0 01-2.88-3.13 7.3 7.3 0 01-.01-4.84 5.13 5.13 0 012.9-3.1 5.67 5.67 0 012.22-.42c.81 0 1.56.14 2.24.42.69.29 1.28.69 1.75 1.22.49.52.87 1.15 1.14 1.89a7 7 0 01.43 2.5zm-20.14 0c0 1.11.25 2.36.74 2.88.5.52 1.13.78 1.91.78a4.07 4.07 0 002.12-.6V9.33c-.19-.04-.99-.2-1.76-.23a2.67 2.67 0 00-2.23 1 4.73 4.73 0 00-.78 2.8zm7.44 5.27c0 1.82-.46 3.16-1.4 4-.94.85-2.37 1.27-4.3 1.27-.7 0-2.17-.13-3.34-.4l.43-2.11c.98.2 2.27.26 2.95.26 1.08 0 1.84-.22 2.3-.66.46-.43.68-1.08.68-1.94v-.44a5.2 5.2 0 01-2.54.6 5.6 5.6 0 01-2.01-.36 4.2 4.2 0 01-2.58-2.71 9.88 9.88 0 01.02-5.35 4.92 4.92 0 012.93-2.96 6.6 6.6 0 012.43-.46 19.64 19.64 0 014.43.66v10.6z" + }, h("path", { + fill: theme === 'dark' ? '#FFF' : '#36395A', + d: "M16 48.3c-3.4 0-6.3-.6-8.7-1.7A12.4 12.4 0 0 1 1.9 42C.6 40 0 38 0 35.4h6.5a6.7 6.7 0 0 0 3.9 6c1.4.7 3.3 1.1 5.6 1.1 2.2 0 4-.3 5.4-1a7 7 0 0 0 3-2.4 6 6 0 0 0 1-3.4c0-1.5-.6-2.8-1.9-3.7-1.3-1-3.3-1.6-5.9-1.8l-4-.4c-3.7-.3-6.6-1.4-8.8-3.4a10 10 0 0 1-3.3-7.9c0-2.4.6-4.6 1.8-6.4a12 12 0 0 1 5-4.3c2.2-1 4.7-1.6 7.5-1.6s5.5.5 7.6 1.6a12 12 0 0 1 5 4.4c1.2 1.8 1.8 4 1.8 6.7h-6.5a6.4 6.4 0 0 0-3.5-5.9c-1-.6-2.6-1-4.4-1s-3.2.3-4.4 1c-1.1.6-2 1.4-2.6 2.4-.5 1-.8 2-.8 3.1a5 5 0 0 0 1.5 3.6c1 1 2.6 1.7 4.7 1.9l4 .3c2.8.2 5.2.8 7.2 1.8 2.1 1 3.7 2.2 4.9 3.8a9.7 9.7 0 0 1 1.7 5.8c0 2.5-.7 4.7-2 6.6a13 13 0 0 1-5.6 4.4c-2.4 1-5.2 1.6-8.4 1.6Zm35.6 0c-2.6 0-4.8-.4-6.7-1.3a13 13 0 0 1-4.7-3.5 17.1 17.1 0 0 1-3.6-10.4v-1c0-2 .3-3.8 1-5.6a13 13 0 0 1 7.3-8.3 15 15 0 0 1 6.3-1.4A13.2 13.2 0 0 1 64 24.3c1 2.2 1.6 4.6 1.6 7.2V34H39.4v-4.3h21.8l-1.8 2.2c0-2-.3-3.7-.9-5.1a7.3 7.3 0 0 0-2.7-3.4c-1.2-.7-2.7-1.1-4.6-1.1s-3.4.4-4.7 1.3a8 8 0 0 0-2.9 3.6c-.6 1.5-.9 3.3-.9 5.4 0 2 .3 3.7 1 5.3a7.9 7.9 0 0 0 2.8 3.7c1.3.8 3 1.3 5 1.3s3.8-.5 5.1-1.3c1.3-1 2.1-2 2.4-3.2h6a11.8 11.8 0 0 1-7 8.7 16 16 0 0 1-6.4 1.2ZM80 48c-2.2 0-4-.3-5.7-1a8.4 8.4 0 0 1-3.7-3.3 9.7 9.7 0 0 1-1.3-5.2c0-2 .5-3.8 1.5-5.2a9 9 0 0 1 4.3-3.1c1.8-.7 4-1 6.7-1H89v4.1h-7.5c-2 0-3.4.5-4.4 1.4-1 1-1.6 2.1-1.6 3.6s.5 2.7 1.6 3.6c1 1 2.5 1.4 4.4 1.4 1.1 0 2.2-.2 3.2-.7 1-.4 1.9-1 2.6-2 .6-1 1-2.4 1-4.2l1.7 2.1c-.2 2-.7 3.8-1.5 5.2a9 9 0 0 1-3.4 3.3 12 12 0 0 1-5.3 1Zm9.5-.7v-8.8h-1v-10c0-1.8-.5-3.2-1.4-4.1-1-1-2.4-1.4-4.2-1.4a142.9 142.9 0 0 0-10.2.4v-5.6a74.8 74.8 0 0 1 8.6-.4c3 0 5.5.4 7.5 1.2s3.4 2 4.4 3.6c1 1.7 1.4 4 1.4 6.7v18.4h-5Zm12.9 0V17.8h5v12.3h-.2c0-4.2 1-7.4 2.8-9.5a11 11 0 0 1 8.3-3.1h1v5.6h-2a9 9 0 0 0-6.3 2.2c-1.5 1.5-2.2 3.6-2.2 6.4v15.6h-6.4Zm34.4 1a15 15 0 0 1-6.6-1.3c-1.9-.9-3.4-2-4.7-3.5a15.5 15.5 0 0 1-2.7-5c-.6-1.7-1-3.6-1-5.4v-1c0-2 .4-3.8 1-5.6a15 15 0 0 1 2.8-4.9c1.3-1.5 2.8-2.6 4.6-3.5a16.4 16.4 0 0 1 13.3.2c2 1 3.5 2.3 4.8 4a12 12 0 0 1 2 6H144c-.2-1.6-1-3-2.2-4.1a7.5 7.5 0 0 0-5.2-1.7 8 8 0 0 0-4.7 1.3 8 8 0 0 0-2.8 3.6 13.8 13.8 0 0 0 0 10.3c.6 1.5 1.5 2.7 2.8 3.6s2.8 1.3 4.8 1.3c1.5 0 2.7-.2 3.8-.8a7 7 0 0 0 2.6-2c.7-1 1-2 1.2-3.2h6.2a11 11 0 0 1-2 6.2 15.1 15.1 0 0 1-11.8 5.5Zm19.7-1v-40h6.4V31h-1.3c0-3 .4-5.5 1.1-7.6a9.7 9.7 0 0 1 3.5-4.8A9.9 9.9 0 0 1 172 17h.3c3.5 0 6 1.1 7.9 3.5 1.7 2.3 2.6 5.7 2.6 10v16.8h-6.4V29.6c0-2.1-.6-3.8-1.8-5a6.4 6.4 0 0 0-4.8-1.8c-2 0-3.7.7-5 2a7.8 7.8 0 0 0-1.9 5.5v17h-6.4Zm63.8 1a12.2 12.2 0 0 1-10.9-6.2 19 19 0 0 1-1.8-7.3h1.4v12.5h-5.1v-40h6.4v19.8l-2 3.5c.2-3.1.8-5.7 1.9-7.7a11 11 0 0 1 4.4-4.5c1.8-1 3.9-1.5 6.1-1.5a13.4 13.4 0 0 1 12.8 9.1c.7 1.9 1 3.8 1 6v1c0 2.2-.3 4.1-1 6a13.6 13.6 0 0 1-13.2 9.4Zm-1.2-5.5a8.4 8.4 0 0 0 7.9-5c.7-1.5 1.1-3.3 1.1-5.3s-.4-3.8-1.1-5.3a8.7 8.7 0 0 0-3.2-3.6 9.6 9.6 0 0 0-9.2-.2 8.5 8.5 0 0 0-3.3 3.2c-.8 1.4-1.3 3-1.3 5v2.3a9 9 0 0 0 1.3 4.8 9 9 0 0 0 3.4 3c1.4.7 2.8 1 4.4 1Zm27.3 3.9-10-28.9h6.5l9.5 28.9h-6Zm-7.5 12.2v-5.7h4.9c1 0 2-.1 2.9-.4a4 4 0 0 0 2-1.4c.4-.7.9-1.6 1.2-2.7l8.6-30.9h6.2l-9.3 32.4a14 14 0 0 1-2.5 5 8.9 8.9 0 0 1-4 2.8c-1.5.6-3.4.9-5.6.9h-4.4Zm9-12.2v-5.2h6.4v5.2H248Z" + }), h("path", { + fill: theme === 'dark' ? '#FFF' : '#003DFF', + d: "M534.4 9.1H528a.8.8 0 0 1-.7-.7V1.8c0-.4.2-.7.6-.8l6.5-1c.4 0 .8.2.9.6v7.8c0 .4-.4.7-.8.7zM428 35.2V.8c0-.5-.3-.8-.7-.8h-.2l-6.4 1c-.4 0-.7.4-.7.8v35c0 1.6 0 11.8 12.3 12.2.5 0 .8-.4.8-.8V43c0-.4-.3-.7-.6-.8-4.5-.5-4.5-6-4.5-7zm106.5-21.8H528c-.4 0-.7.4-.7.8v34c0 .4.3.8.7.8h6.5c.4 0 .8-.4.8-.8v-34c0-.5-.4-.8-.8-.8zm-17.7 21.8V.8c0-.5-.3-.8-.8-.8l-6.5 1c-.4 0-.7.4-.7.8v35c0 1.6 0 11.8 12.3 12.2.4 0 .8-.4.8-.8V43c0-.4-.3-.7-.7-.8-4.4-.5-4.4-6-4.4-7zm-22.2-20.6a16.5 16.5 0 0 1 8.6 9.3c.8 2.2 1.3 4.8 1.3 7.5a19.4 19.4 0 0 1-4.6 12.6 14.8 14.8 0 0 1-5.2 3.6c-2 .9-5.2 1.4-6.8 1.4a21 21 0 0 1-6.7-1.4 15.4 15.4 0 0 1-8.6-9.3 21.3 21.3 0 0 1 0-14.4 15.2 15.2 0 0 1 8.6-9.3c2-.8 4.3-1.2 6.7-1.2s4.6.4 6.7 1.2zm-6.7 27.6c2.7 0 4.7-1 6.2-3s2.2-4.3 2.2-7.8-.7-6.3-2.2-8.3-3.5-3-6.2-3-4.7 1-6.1 3c-1.5 2-2.2 4.8-2.2 8.3s.7 5.8 2.2 7.8 3.5 3 6.2 3zm-88.8-28.8c-6.2 0-11.7 3.3-14.8 8.2a18.6 18.6 0 0 0 4.8 25.2c1.8 1.2 4 1.8 6.2 1.7s.1 0 .1 0h.9c4.2-.7 8-4 9.1-8.1v7.4c0 .4.3.7.8.7h6.4a.7.7 0 0 0 .7-.7V14.2c0-.5-.3-.8-.7-.8h-13.5zm6.3 26.5a9.8 9.8 0 0 1-5.7 2h-.5a10 10 0 0 1-9.2-14c1.4-3.7 5-6.3 9-6.3h6.4v18.3zm152.3-26.5h13.5c.5 0 .8.3.8.7v33.7c0 .4-.3.7-.8.7h-6.4a.7.7 0 0 1-.8-.7v-7.4c-1.2 4-4.8 7.4-9 8h-.1a4.2 4.2 0 0 1-.5.1h-.9a10.3 10.3 0 0 1-7-2.6c-4-3.3-6.5-8.4-6.5-14.2 0-3.7 1-7.2 3-10 3-5 8.5-8.3 14.7-8.3zm.6 28.4c2.2-.1 4.2-.6 5.7-2V21.7h-6.3a9.8 9.8 0 0 0-9 6.4 10.2 10.2 0 0 0 9.1 13.9h.5zM452.8 13.4c-6.2 0-11.7 3.3-14.8 8.2a18.5 18.5 0 0 0 3.6 24.3 10.4 10.4 0 0 0 13 .6c2.2-1.5 3.8-3.7 4.5-6.1v7.8c0 2.8-.8 5-2.2 6.3-1.5 1.5-4 2.2-7.5 2.2l-6-.3c-.3 0-.7.2-.8.5l-1.6 5.5c-.1.4.1.8.5 1h.1c2.8.4 5.5.6 7 .6 6.3 0 11-1.4 14-4.1 2.7-2.5 4.2-6.3 4.5-11.4V14.2c0-.5-.4-.8-.8-.8h-13.5zm6.3 8.2v18.3a9.6 9.6 0 0 1-5.6 2h-1a10.3 10.3 0 0 1-8.8-14c1.4-3.7 5-6.3 9-6.3h6.4zM291 31.5A32 32 0 0 1 322.8 0h30.8c.6 0 1.2.5 1.2 1.2v61.5c0 1.1-1.3 1.7-2.2 1l-19.2-17a18 18 0 0 1-11 3.4 18.1 18.1 0 1 1 18.2-14.8c-.1.4-.5.7-.9.6-.1 0-.3 0-.4-.2l-3.8-3.4c-.4-.3-.6-.8-.7-1.4a12 12 0 1 0-2.4 8.3c.4-.4 1-.5 1.6-.2l14.7 13.1v-46H323a26 26 0 1 0 10 49.7c.8-.4 1.6-.2 2.3.3l3 2.7c.3.2.3.7 0 1l-.2.2a32 32 0 0 1-47.2-28.6z" })))); }; @@ -19088,7 +19561,7 @@ if (isFirstRendering) { var _widgetParams$theme = widgetParams.theme, theme = _widgetParams$theme === void 0 ? 'light' : _widgetParams$theme; - S(v(PoweredBy, { + P(h(PoweredBy, { cssClasses: cssClasses, url: url, theme: theme @@ -19112,13 +19585,13 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$j(), suit$j({ + root: cx(suit$j(), suit$j({ modifierName: theme === 'dark' ? 'dark' : 'light' }), userCssClasses.root), - link: classnames(suit$j({ + link: cx(suit$j({ descendantName: 'link' }), userCssClasses.link), - logo: classnames(suit$j({ + logo: cx(suit$j({ descendantName: 'logo' }), userCssClasses.logo) }; @@ -19127,7 +19600,7 @@ cssClasses: cssClasses }); var makeWidget = connectPoweredBy(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ theme: theme @@ -19152,13 +19625,11 @@ }); }; - /** @jsx h */ - var QueryRuleCustomData = function QueryRuleCustomData(_ref) { var cssClasses = _ref.cssClasses, templates = _ref.templates, items = _ref.items; - return v(Template, { + return h(Template, { templateKey: "default", templates: templates, rootProps: { @@ -19187,7 +19658,7 @@ templates = _ref2.templates; return function (_ref3) { var items = _ref3.items; - S(v(QueryRuleCustomData, { + P(h(QueryRuleCustomData, { cssClasses: cssClasses, templates: templates, items: items @@ -19212,7 +19683,7 @@ } var cssClasses = { - root: classnames(suit$k(), userCssClasses.root) + root: cx(suit$k(), userCssClasses.root) }; var containerNode = getContainerNode(container); @@ -19225,7 +19696,7 @@ templates: templates }); var makeWidget = connectQueryRules(specializedRenderer, function () { - S(null, containerNode); + P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ transformItems: transformItems @@ -19295,16 +19766,16 @@ templateProps = _this$props.templateProps; var isDisabled = min && max ? min >= max : false; var hasRefinements = Boolean(minValue || maxValue); - var rootClassNames = classnames(cssClasses.root, _defineProperty({}, cssClasses.noRefinement, !hasRefinements)); - return v("div", { + var rootClassNames = cx(cssClasses.root, !hasRefinements && cssClasses.noRefinement); + return h("div", { className: rootClassNames - }, v("form", { + }, h("form", { className: cssClasses.form, onSubmit: this.onSubmit - }, v("label", { + }, h("label", { className: cssClasses.label - }, v("input", { - className: classnames(cssClasses.input, cssClasses.inputMin), + }, h("input", { + className: cx(cssClasses.input, cssClasses.inputMin), type: "number", min: min, max: max, @@ -19313,16 +19784,16 @@ onInput: this.onInput('min'), placeholder: min === null || min === void 0 ? void 0 : min.toString(), disabled: isDisabled - })), v(Template, _extends({}, templateProps, { + })), h(Template, _extends({}, templateProps, { templateKey: "separatorText", rootTagName: "span", rootProps: { className: cssClasses.separator } - })), v("label", { + })), h("label", { className: cssClasses.label - }, v("input", { - className: classnames(cssClasses.input, cssClasses.inputMax), + }, h("input", { + className: cx(cssClasses.input, cssClasses.inputMax), type: "number", min: min, max: max, @@ -19331,7 +19802,7 @@ onInput: this.onInput('max'), placeholder: max === null || max === void 0 ? void 0 : max.toString(), disabled: isDisabled - })), v(Template, _extends({}, templateProps, { + })), h(Template, _extends({}, templateProps, { templateKey: "submitText", rootTagName: "button", rootProps: { @@ -19344,15 +19815,19 @@ }]); return RangeInput; - }(_); + }(d); var withUsage$L = createDocumentationMessageGenerator({ name: 'range-input' }); var suit$l = component('RangeInput'); var defaultTemplates$c = { - separatorText: 'to', - submitText: 'Go' + separatorText: function separatorText() { + return 'to'; + }, + submitText: function submitText() { + return 'Go'; + } }; var renderer$g = function renderer(_ref) { @@ -19388,7 +19863,7 @@ min: minValue !== -Infinity && minValue !== rangeMin ? minValue : undefined, max: maxValue !== Infinity && maxValue !== rangeMax ? maxValue : undefined }; - S(v(RangeInput, { + P(h(RangeInput, { min: rangeMin, max: rangeMax, step: step, @@ -19419,31 +19894,31 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$l(), userCssClasses.root), - noRefinement: classnames(suit$l({ + root: cx(suit$l(), userCssClasses.root), + noRefinement: cx(suit$l({ modifierName: 'noRefinement' })), - form: classnames(suit$l({ + form: cx(suit$l({ descendantName: 'form' }), userCssClasses.form), - label: classnames(suit$l({ + label: cx(suit$l({ descendantName: 'label' }), userCssClasses.label), - input: classnames(suit$l({ + input: cx(suit$l({ descendantName: 'input' }), userCssClasses.input), - inputMin: classnames(suit$l({ + inputMin: cx(suit$l({ descendantName: 'input', modifierName: 'min' }), userCssClasses.inputMin), - inputMax: classnames(suit$l({ + inputMax: cx(suit$l({ descendantName: 'input', modifierName: 'max' }), userCssClasses.inputMax), - separator: classnames(suit$l({ + separator: cx(suit$l({ descendantName: 'separator' }), userCssClasses.separator), - submit: classnames(suit$l({ + submit: cx(suit$l({ descendantName: 'submit' }), userCssClasses.submit) }; @@ -19454,7 +19929,7 @@ renderState: {} }); var makeWidget = connectRange(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -19512,14 +19987,14 @@ } function Button(props) { - return v("button", _extends({}, props, { + return h("button", _extends({}, props, { type: "button" })); } // Preact doesn't have builtin types for Style, JSX.HTMLAttributes['style'] is just object // maybe migrate to csstype later? - var _ref6 = v("div", { + var _ref6 = h("div", { className: "rheostat-background" }); @@ -19549,7 +20024,7 @@ values: _this.props.values }); - _defineProperty(_assertThisInitialized(_this), "rheostat", p()); + _defineProperty(_assertThisInitialized(_this), "rheostat", y()); _this.getPublicState = _this.getPublicState.bind(_assertThisInitialized(_this)); _this.getSliderBoundingBox = _this.getSliderBoundingBox.bind(_assertThisInitialized(_this)); @@ -20053,7 +20528,7 @@ className = _this$state6.className, handlePos = _this$state6.handlePos, values = _this$state6.values; - return v("div", { + return h("div", { className: className, ref: this.rheostat, onClick: disabled ? undefined : this.handleClick, @@ -20068,7 +20543,7 @@ left: "".concat(pos, "%"), position: 'absolute' }; - return v(Handle, { + return h(Handle, { "aria-valuemax": _this7.getMaxValue(idx), "aria-valuemin": _this7.getMinValue(idx), "aria-valuenow": values[idx], @@ -20089,7 +20564,7 @@ return null; } - return v(ProgressBar, { + return h(ProgressBar, { className: "rheostat-progress", key: "progress-bar-".concat(idx), style: _this7.getProgressStyle(idx) @@ -20103,7 +20578,7 @@ left: "".concat(pos, "%"), position: 'absolute' }; - return v(PitComponent, { + return h(PitComponent, { key: "pit-".concat(n), style: pitStyle }, n); @@ -20112,7 +20587,7 @@ }]); return Rheostat; - }(_); + }(d); _defineProperty(Rheostat, "defaultProps", { className: '', @@ -20145,14 +20620,12 @@ var shouldDisplayValue = [0, 50, 100].includes(positionValue); var value = children; var pitValue = Math.round(parseInt(value, 10) * 100) / 100; - return v("div", { + return h("div", { style: _objectSpread2(_objectSpread2({}, style), {}, { marginLeft: positionValue === 100 ? '-2px' : 0 }), - className: classnames('rheostat-marker', 'rheostat-marker-horizontal', { - 'rheostat-marker-large': shouldDisplayValue - }) - }, shouldDisplayValue && v("div", { + className: cx('rheostat-marker', 'rheostat-marker-horizontal', shouldDisplayValue && 'rheostat-marker-large') + }, shouldDisplayValue && h("div", { className: 'rheostat-value' }, pitValue)); }; @@ -20188,13 +20661,10 @@ var roundedValue = Math.round( // have to cast as a string, as the value given to the prop is a number, but becomes a string when read parseFloat(props['aria-valuenow']) * 100) / 100; var value = _typeof(tooltips) === 'object' && tooltips.format ? tooltips.format(roundedValue) : roundedValue; - var className = classnames(props.className, { - 'rheostat-handle-lower': props['data-handle-key'] === 0, - 'rheostat-handle-upper': props['data-handle-key'] === 1 - }); - return v("div", _extends({}, props, { + var className = cx(props.className, props['data-handle-key'] === 0 && 'rheostat-handle-lower', props['data-handle-key'] === 1 && 'rheostat-handle-upper'); + return h("div", _extends({}, props, { className: className - }), tooltips && v("div", { + }), tooltips && h("div", { className: "rheostat-tooltip" }, value)); }; @@ -20264,9 +20734,9 @@ min: min, max: max }); - return v("div", { - className: classnames(cssClasses.root, _defineProperty({}, cssClasses.disabledRoot, this.isDisabled)) - }, v(Rheostat, { + return h("div", { + className: cx(cssClasses.root, this.isDisabled && cssClasses.disabledRoot) + }, h(Rheostat, { handle: this.createHandleComponent(tooltips), onChange: this.handleChange, min: min, @@ -20282,7 +20752,7 @@ }]); return Slider; - }(_); + }(d); var withUsage$M = createDocumentationMessageGenerator({ name: 'range-slider' @@ -20318,7 +20788,7 @@ // backward compatible so we still need to pass [-Infinity, Infinity] var values = [minFinite > maxRange ? maxRange : minFinite, maxFinite < minRange ? minRange : maxFinite]; - S(v(Slider, { + P(h(Slider, { cssClasses: cssClasses, refine: refine, min: minRange, @@ -20364,8 +20834,8 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$m(), userCssClasses.root), - disabledRoot: classnames(suit$m({ + root: cx(suit$m(), userCssClasses.root), + disabledRoot: cx(suit$m({ modifierName: 'disabled' }), userCssClasses.disabledRoot) }; @@ -20377,7 +20847,7 @@ cssClasses: cssClasses }); var makeWidget = connectRange(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -20390,8 +20860,56 @@ }); }; + function ItemWrapper(_ref) { + var children = _ref.children, + count = _ref.count, + value = _ref.value, + url = _ref.url, + cssClasses = _ref.cssClasses; + + if (count) { + return h("a", { + className: cx(cssClasses.link), + "aria-label": "".concat(value, " & up"), + href: url + }, children); + } + + return h("div", { + className: cx(cssClasses.link), + "aria-label": "".concat(value, " & up"), + disabled: true + }, children); + } + var defaultTemplates$d = { - item: "{{#count}}{{/count}}{{^count}}{{/count}}" + item: function item(_ref2) { + var count = _ref2.count, + value = _ref2.value, + url = _ref2.url, + stars = _ref2.stars, + cssClasses = _ref2.cssClasses; + return h(ItemWrapper, { + count: count, + value: value, + url: url, + cssClasses: cssClasses + }, stars.map(function (isFull, index) { + return h("svg", { + key: index, + className: cx(cssClasses.starIcon, isFull ? cssClasses.fullStarIcon : cssClasses.emptyStarIcon), + "aria-hidden": "true", + width: "24", + height: "24" + }, h("use", { + xlinkHref: isFull ? '#ais-RatingMenu-starSymbol' : '#ais-RatingMenu-starEmptySymbol' + })); + }), h("span", { + className: cx(cssClasses.label) + }, "& Up"), count && h("span", { + className: cx(cssClasses.count) + }, formatNumber(count))); + } }; var withUsage$N = createDocumentationMessageGenerator({ @@ -20399,11 +20917,11 @@ }); var suit$n = component('RatingMenu'); - var _ref3$1 = v("path", { + var _ref3 = h("path", { d: "M12 .288l2.833 8.718h9.167l-7.417 5.389 2.833 8.718-7.416-5.388-7.417 5.388 2.833-8.718-7.416-5.389h9.167z" }); - var _ref4 = v("path", { + var _ref4 = h("path", { d: "M12 6.76l1.379 4.246h4.465l-3.612 2.625 1.379 4.246-3.611-2.625-3.612 2.625 1.379-4.246-3.612-2.625h4.465l1.38-4.246zm0-6.472l-2.833 8.718h-9.167l7.416 5.389-2.833 8.718 7.417-5.388 7.416 5.388-2.833-8.718 7.417-5.389h-9.167l-2.833-8.718z" }); @@ -20427,20 +20945,20 @@ return; } - S(v(RefinementList$1, { + P(h(RefinementList$1, { createURL: createURL, cssClasses: cssClasses, facetValues: items, templateProps: renderState.templateProps, toggleRefinement: refine - }, v("svg", { + }, h("svg", { style: "display:none;" - }, v("symbol", { + }, h("symbol", { id: suit$n({ descendantName: 'starSymbol' }), viewBox: "0 0 24 24" - }, _ref3$1), v("symbol", { + }, _ref3), h("symbol", { id: suit$n({ descendantName: 'starEmptySymbol' }), @@ -20493,42 +21011,42 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$n(), userCssClasses.root), - noRefinementRoot: classnames(suit$n({ + root: cx(suit$n(), userCssClasses.root), + noRefinementRoot: cx(suit$n({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: classnames(suit$n({ + list: cx(suit$n({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$n({ + item: cx(suit$n({ descendantName: 'item' }), userCssClasses.item), - selectedItem: classnames(suit$n({ + selectedItem: cx(suit$n({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - disabledItem: classnames(suit$n({ + disabledItem: cx(suit$n({ descendantName: 'item', modifierName: 'disabled' }), userCssClasses.disabledItem), - link: classnames(suit$n({ + link: cx(suit$n({ descendantName: 'link' }), userCssClasses.link), - starIcon: classnames(suit$n({ + starIcon: cx(suit$n({ descendantName: 'starIcon' }), userCssClasses.starIcon), - fullStarIcon: classnames(suit$n({ + fullStarIcon: cx(suit$n({ descendantName: 'starIcon', modifierName: 'full' }), userCssClasses.fullStarIcon), - emptyStarIcon: classnames(suit$n({ + emptyStarIcon: cx(suit$n({ descendantName: 'starIcon', modifierName: 'empty' }), userCssClasses.emptyStarIcon), - label: classnames(suit$n({ + label: cx(suit$n({ descendantName: 'label' }), userCssClasses.label), - count: classnames(suit$n({ + count: cx(suit$n({ descendantName: 'count' }), userCssClasses.count) }; @@ -20539,7 +21057,7 @@ templates: templates }); var makeWidget = connectRatingMenu(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -20549,16 +21067,98 @@ }); }; + var _ref2 = h("path", { + d: "M8.114 10L.944 2.83 0 1.885 1.886 0l.943.943L10 8.113l7.17-7.17.944-.943L20 1.886l-.943.943-7.17 7.17 7.17 7.17.943.944L18.114 20l-.943-.943-7.17-7.17-7.17 7.17-.944.943L0 18.114l.943-.943L8.113 10z" + }); + + var _ref4$1 = h("path", { + d: "M26.804 29.01c-2.832 2.34-6.465 3.746-10.426 3.746C7.333 32.756 0 25.424 0 16.378 0 7.333 7.333 0 16.378 0c9.046 0 16.378 7.333 16.378 16.378 0 3.96-1.406 7.594-3.746 10.426l10.534 10.534c.607.607.61 1.59-.004 2.202-.61.61-1.597.61-2.202.004L26.804 29.01zm-10.426.627c7.323 0 13.26-5.936 13.26-13.26 0-7.32-5.937-13.257-13.26-13.257C9.056 3.12 3.12 9.056 3.12 16.378c0 7.323 5.936 13.26 13.258 13.26z" + }); + + var _ref6$1 = h("g", { + fill: "none", + fillRule: "evenodd" + }, h("g", { + transform: "translate(1 1)", + strokeWidth: "2" + }, h("circle", { + strokeOpacity: ".5", + cx: "18", + cy: "18", + r: "18" + }), h("path", { + d: "M36 18c0-9.94-8.06-18-18-18" + }, h("animateTransform", { + attributeName: "transform", + type: "rotate", + from: "0 18 18", + to: "360 18 18", + dur: "1s", + repeatCount: "indefinite" + })))); + var defaultTemplate = { - reset: "\n\n \n\n ", - submit: "\n\n \n\n ", - loadingIndicator: "\n\n \n \n \n \n \n \n \n \n\n " + reset: function reset(_ref) { + var cssClasses = _ref.cssClasses; + return h("svg", { + className: cssClasses.resetIcon, + viewBox: "0 0 20 20", + width: "10", + height: "10" + }, _ref2); + }, + submit: function submit(_ref3) { + var cssClasses = _ref3.cssClasses; + return h("svg", { + className: cssClasses.submitIcon, + width: "10", + height: "10", + viewBox: "0 0 40 40" + }, _ref4$1); + }, + loadingIndicator: function loadingIndicator(_ref5) { + var cssClasses = _ref5.cssClasses; + return h("svg", { + className: cssClasses.loadingIcon, + width: "16", + height: "16", + viewBox: "0 0 38 38", + stroke: "#444" + }, _ref6$1); + } }; var defaultTemplates$e = { - item: "", - showMoreText: "\n {{#isShowingMore}}\n Show less\n {{/isShowingMore}}\n {{^isShowingMore}}\n Show more\n {{/isShowingMore}}\n ", - searchableNoResults: 'No results' + item: function item(_ref) { + var cssClasses = _ref.cssClasses, + count = _ref.count, + value = _ref.value, + highlighted = _ref.highlighted, + isRefined = _ref.isRefined, + isFromSearch = _ref.isFromSearch; + return h("label", { + className: cx(cssClasses.label) + }, h("input", { + type: "checkbox", + className: cx(cssClasses.checkbox), + value: value, + defaultChecked: isRefined + }), h("span", { + className: cx(cssClasses.labelText), + dangerouslySetInnerHTML: isFromSearch ? { + __html: highlighted + } : undefined + }, !isFromSearch && highlighted), h("span", { + className: cx(cssClasses.count) + }, formatNumber(count))); + }, + showMoreText: function showMoreText(_ref2) { + var isShowingMore = _ref2.isShowingMore; + return isShowingMore ? 'Show less' : 'Show more'; + }, + searchableNoResults: function searchableNoResults() { + return 'No results'; + } }; var withUsage$O = createDocumentationMessageGenerator({ @@ -20603,7 +21203,7 @@ return; } - S(v(RefinementList$1, { + P(h(RefinementList$1, { createURL: createURL, cssClasses: cssClasses, facetValues: items, @@ -20672,69 +21272,69 @@ var escapeFacetValues = searchable ? Boolean(searchableEscapeFacetValues) : false; var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$o(), userCssClasses.root), - noRefinementRoot: classnames(suit$o({ + root: cx(suit$o(), userCssClasses.root), + noRefinementRoot: cx(suit$o({ modifierName: 'noRefinement' }), userCssClasses.noRefinementRoot), - list: classnames(suit$o({ + list: cx(suit$o({ descendantName: 'list' }), userCssClasses.list), - item: classnames(suit$o({ + item: cx(suit$o({ descendantName: 'item' }), userCssClasses.item), - selectedItem: classnames(suit$o({ + selectedItem: cx(suit$o({ descendantName: 'item', modifierName: 'selected' }), userCssClasses.selectedItem), - searchBox: classnames(suit$o({ + searchBox: cx(suit$o({ descendantName: 'searchBox' }), userCssClasses.searchBox), - label: classnames(suit$o({ + label: cx(suit$o({ descendantName: 'label' }), userCssClasses.label), - checkbox: classnames(suit$o({ + checkbox: cx(suit$o({ descendantName: 'checkbox' }), userCssClasses.checkbox), - labelText: classnames(suit$o({ + labelText: cx(suit$o({ descendantName: 'labelText' }), userCssClasses.labelText), - count: classnames(suit$o({ + count: cx(suit$o({ descendantName: 'count' }), userCssClasses.count), - noResults: classnames(suit$o({ + noResults: cx(suit$o({ descendantName: 'noResults' }), userCssClasses.noResults), - showMore: classnames(suit$o({ + showMore: cx(suit$o({ descendantName: 'showMore' }), userCssClasses.showMore), - disabledShowMore: classnames(suit$o({ + disabledShowMore: cx(suit$o({ descendantName: 'showMore', modifierName: 'disabled' }), userCssClasses.disabledShowMore), searchable: { - root: classnames(searchBoxSuit(), userCssClasses.searchableRoot), - form: classnames(searchBoxSuit({ + root: cx(searchBoxSuit(), userCssClasses.searchableRoot), + form: cx(searchBoxSuit({ descendantName: 'form' }), userCssClasses.searchableForm), - input: classnames(searchBoxSuit({ + input: cx(searchBoxSuit({ descendantName: 'input' }), userCssClasses.searchableInput), - submit: classnames(searchBoxSuit({ + submit: cx(searchBoxSuit({ descendantName: 'submit' }), userCssClasses.searchableSubmit), - submitIcon: classnames(searchBoxSuit({ + submitIcon: cx(searchBoxSuit({ descendantName: 'submitIcon' }), userCssClasses.searchableSubmitIcon), - reset: classnames(searchBoxSuit({ + reset: cx(searchBoxSuit({ descendantName: 'reset' }), userCssClasses.searchableReset), - resetIcon: classnames(searchBoxSuit({ + resetIcon: cx(searchBoxSuit({ descendantName: 'resetIcon' }), userCssClasses.searchableResetIcon), - loadingIndicator: classnames(searchBoxSuit({ + loadingIndicator: cx(searchBoxSuit({ descendantName: 'loadingIndicator' }), userCssClasses.searchableLoadingIndicator), - loadingIcon: classnames(searchBoxSuit({ + loadingIcon: cx(searchBoxSuit({ descendantName: 'loadingIcon' }), userCssClasses.searchableLoadingIcon) } @@ -20755,7 +21355,7 @@ showMore: showMore }); var makeWidget = connectRefinementList(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -20771,17 +21371,15 @@ }); }; - /** @jsx h */ - var RelevantSort = function RelevantSort(_ref) { var cssClasses = _ref.cssClasses, templates = _ref.templates, isRelevantSorted = _ref.isRelevantSorted, isVirtualReplica = _ref.isVirtualReplica, refine = _ref.refine; - return isVirtualReplica ? v("div", { + return isVirtualReplica ? h("div", { className: cssClasses.root - }, v(Template, { + }, h(Template, { templateKey: "text", templates: templates, rootProps: { @@ -20790,7 +21388,7 @@ data: { isRelevantSorted: isRelevantSorted } - }), v("button", { + }), h("button", { type: "button", className: cssClasses.button, onClick: function onClick() { @@ -20800,7 +21398,7 @@ refine(undefined); } } - }, v(Template, { + }, h(Template, { rootTagName: "span", templateKey: "button", templates: templates, @@ -20811,7 +21409,9 @@ }; var defaultTemplates$f = { - text: '', + text: function text() { + return ''; + }, button: function button(_ref) { var isRelevantSorted = _ref.isRelevantSorted; return isRelevantSorted ? 'See all results' : 'See relevant results'; @@ -20831,7 +21431,7 @@ var isRelevantSorted = _ref2.isRelevantSorted, isVirtualReplica = _ref2.isVirtualReplica, refine = _ref2.refine; - S(v(RelevantSort, { + P(h(RelevantSort, { cssClasses: cssClasses, templates: templates, isRelevantSorted: isRelevantSorted, @@ -20854,11 +21454,11 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$p(), userCssClasses.root), - text: classnames(suit$p({ + root: cx(suit$p(), userCssClasses.root), + text: cx(suit$p({ descendantName: 'text' }), userCssClasses.text), - button: classnames(suit$p({ + button: cx(suit$p({ descendantName: 'button' }), userCssClasses.button) }; @@ -20872,7 +21472,7 @@ templates: templates }); var makeWidget = connectRelevantSort(specializedRenderer, function () { - S(null, containerNode); + P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({})), {}, { $$widgetType: 'ais.relevantSort' @@ -20898,7 +21498,7 @@ var refine = _ref2.refine, query = _ref2.query, isSearchStalled = _ref2.isSearchStalled; - S(v(SearchBox, { + P(h(SearchBox, { query: query, placeholder: placeholder, autofocus: autofocus, @@ -20950,29 +21550,29 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$q(), userCssClasses.root), - form: classnames(suit$q({ + root: cx(suit$q(), userCssClasses.root), + form: cx(suit$q({ descendantName: 'form' }), userCssClasses.form), - input: classnames(suit$q({ + input: cx(suit$q({ descendantName: 'input' }), userCssClasses.input), - submit: classnames(suit$q({ + submit: cx(suit$q({ descendantName: 'submit' }), userCssClasses.submit), - submitIcon: classnames(suit$q({ + submitIcon: cx(suit$q({ descendantName: 'submitIcon' }), userCssClasses.submitIcon), - reset: classnames(suit$q({ + reset: cx(suit$q({ descendantName: 'reset' }), userCssClasses.reset), - resetIcon: classnames(suit$q({ + resetIcon: cx(suit$q({ descendantName: 'resetIcon' }), userCssClasses.resetIcon), - loadingIndicator: classnames(suit$q({ + loadingIndicator: cx(suit$q({ descendantName: 'loadingIndicator' }), userCssClasses.loadingIndicator), - loadingIcon: classnames(suit$q({ + loadingIcon: cx(suit$q({ descendantName: 'loadingIcon' }), userCssClasses.loadingIcon) }; @@ -20991,7 +21591,7 @@ showLoadingIndicator: showLoadingIndicator }); var makeWidget = connectSearchBox(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ queryHook: queryHook @@ -21017,9 +21617,9 @@ return; } - S(v("div", { + P(h("div", { className: cssClasses.root - }, v(Selector, { + }, h(Selector, { cssClasses: cssClasses, currentValue: currentRefinement, options: options, @@ -21047,11 +21647,11 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$r(), userCssClasses.root), - select: classnames(suit$r({ + root: cx(suit$r(), userCssClasses.root), + select: cx(suit$r({ descendantName: 'select' }), userCssClasses.select), - option: classnames(suit$r({ + option: cx(suit$r({ descendantName: 'option' }), userCssClasses.option) }; @@ -21060,7 +21660,7 @@ cssClasses: cssClasses }); var makeWidget = connectSortBy(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ container: containerNode, @@ -21078,9 +21678,9 @@ templateProps = _ref.templateProps, rest = _objectWithoutProperties(_ref, ["nbHits", "nbSortedHits", "cssClasses", "templateProps"]); - return v("div", { - className: classnames(cssClasses.root) - }, v(Template, _extends({}, templateProps, { + return h("div", { + className: cx(cssClasses.root) + }, h(Template, _extends({}, templateProps, { templateKey: "text", rootTagName: "span", rootProps: { @@ -21105,24 +21705,70 @@ }); var suit$s = component('Stats'); var defaultTemplates$g = { - text: "\n {{#areHitsSorted}}\n {{#hasNoSortedResults}}No relevant results{{/hasNoSortedResults}}\n {{#hasOneSortedResults}}1 relevant result{{/hasOneSortedResults}}\n {{#hasManySortedResults}}{{#helpers.formatNumber}}{{nbSortedHits}}{{/helpers.formatNumber}} relevant results{{/hasManySortedResults}}\n sorted out of {{#helpers.formatNumber}}{{nbHits}}{{/helpers.formatNumber}}\n {{/areHitsSorted}}\n {{^areHitsSorted}}\n {{#hasNoResults}}No results{{/hasNoResults}}\n {{#hasOneResult}}1 result{{/hasOneResult}}\n {{#hasManyResults}}{{#helpers.formatNumber}}{{nbHits}}{{/helpers.formatNumber}} results{{/hasManyResults}}\n {{/areHitsSorted}}\n found in {{processingTimeMS}}ms" + text: function text(props) { + return "".concat(props.areHitsSorted ? getSortedResultsSentence(props) : getResultsSentence(props), " found in ").concat(props.processingTimeMS, "ms"); + } }; - var renderer$n = function renderer(_ref) { - var renderState = _ref.renderState, - cssClasses = _ref.cssClasses, - containerNode = _ref.containerNode, - templates = _ref.templates; - return function (_ref2, isFirstRendering) { - var hitsPerPage = _ref2.hitsPerPage, - nbHits = _ref2.nbHits, - nbSortedHits = _ref2.nbSortedHits, - areHitsSorted = _ref2.areHitsSorted, - nbPages = _ref2.nbPages, - page = _ref2.page, - processingTimeMS = _ref2.processingTimeMS, - query = _ref2.query, - instantSearchInstance = _ref2.instantSearchInstance; + function getSortedResultsSentence(_ref) { + var nbHits = _ref.nbHits, + hasNoSortedResults = _ref.hasNoSortedResults, + hasOneSortedResults = _ref.hasOneSortedResults, + hasManySortedResults = _ref.hasManySortedResults, + nbSortedHits = _ref.nbSortedHits; + var suffix = "sorted out of ".concat(formatNumber(nbHits)); + + if (hasNoSortedResults) { + return "No relevant results ".concat(suffix); + } + + if (hasOneSortedResults) { + return "1 relevant result ".concat(suffix); + } + + if (hasManySortedResults) { + return "".concat(formatNumber(nbSortedHits || 0), " relevant results ").concat(suffix); + } + + return ''; + } + + function getResultsSentence(_ref2) { + var nbHits = _ref2.nbHits, + hasNoResults = _ref2.hasNoResults, + hasOneResult = _ref2.hasOneResult, + hasManyResults = _ref2.hasManyResults; + + if (hasNoResults) { + return 'No results'; + } + + if (hasOneResult) { + return '1 result'; + } + + if (hasManyResults) { + return "".concat(formatNumber(nbHits), " results"); + } + + return ''; + } + + var renderer$n = function renderer(_ref3) { + var renderState = _ref3.renderState, + cssClasses = _ref3.cssClasses, + containerNode = _ref3.containerNode, + templates = _ref3.templates; + return function (_ref4, isFirstRendering) { + var hitsPerPage = _ref4.hitsPerPage, + nbHits = _ref4.nbHits, + nbSortedHits = _ref4.nbSortedHits, + areHitsSorted = _ref4.areHitsSorted, + nbPages = _ref4.nbPages, + page = _ref4.page, + processingTimeMS = _ref4.processingTimeMS, + query = _ref4.query, + instantSearchInstance = _ref4.instantSearchInstance; if (isFirstRendering) { renderState.templateProps = prepareTemplateProps({ @@ -21133,7 +21779,7 @@ return; } - S(v(Stats, { + P(h(Stats, { cssClasses: cssClasses, hitsPerPage: hitsPerPage, nbHits: nbHits, @@ -21156,12 +21802,12 @@ var stats = function stats(widgetParams) { - var _ref3 = widgetParams || {}, - container = _ref3.container, - _ref3$cssClasses = _ref3.cssClasses, - userCssClasses = _ref3$cssClasses === void 0 ? {} : _ref3$cssClasses, - _ref3$templates = _ref3.templates, - templates = _ref3$templates === void 0 ? {} : _ref3$templates; + var _ref5 = widgetParams || {}, + container = _ref5.container, + _ref5$cssClasses = _ref5.cssClasses, + userCssClasses = _ref5$cssClasses === void 0 ? {} : _ref5$cssClasses, + _ref5$templates = _ref5.templates, + templates = _ref5$templates === void 0 ? {} : _ref5$templates; if (!container) { throw new Error(withUsage$S('The `container` option is required.')); @@ -21169,8 +21815,8 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$s(), userCssClasses.root), - text: classnames(suit$s({ + root: cx(suit$s(), userCssClasses.root), + text: cx(suit$s({ descendantName: 'text' }), userCssClasses.text) }; @@ -21181,7 +21827,7 @@ renderState: {} }); var makeWidget = connectStats(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({})), {}, { $$widgetType: 'ais.stats' @@ -21193,11 +21839,11 @@ refine = _ref.refine, cssClasses = _ref.cssClasses, templateProps = _ref.templateProps; - return v("div", { + return h("div", { className: cssClasses.root - }, v("label", { + }, h("label", { className: cssClasses.label - }, v("input", { + }, h("input", { className: cssClasses.checkbox, type: "checkbox", checked: currentRefinement.isRefined, @@ -21206,7 +21852,7 @@ isRefined: !event.target.checked }); } - }), v(Template, _extends({}, templateProps, { + }), h(Template, _extends({}, templateProps, { rootTagName: "span", rootProps: { className: cssClasses.labelText @@ -21217,7 +21863,10 @@ }; var defaultTemplates$h = { - labelText: '{{name}}' + labelText: function labelText(_ref) { + var name = _ref.name; + return name; + } }; var withUsage$T = createDocumentationMessageGenerator({ @@ -21244,7 +21893,7 @@ return; } - S(v(ToggleRefinement, { + P(h(ToggleRefinement, { cssClasses: cssClasses, currentRefinement: value, templateProps: renderState.templateProps, @@ -21283,14 +21932,14 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$t(), userCssClasses.root), - label: classnames(suit$t({ + root: cx(suit$t(), userCssClasses.root), + label: cx(suit$t({ descendantName: 'label' }), userCssClasses.label), - checkbox: classnames(suit$t({ + checkbox: cx(suit$t({ descendantName: 'checkbox' }), userCssClasses.checkbox), - labelText: classnames(suit$t({ + labelText: cx(suit$t({ descendantName: 'labelText' }), userCssClasses.labelText) }; @@ -21301,7 +21950,7 @@ templates: templates }); var makeWidget = connectToggleRefinement(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ attribute: attribute, @@ -21312,8 +21961,6 @@ }); }; - /** @jsx h */ - var VoiceSearch = function VoiceSearch(_ref) { var cssClasses = _ref.cssClasses, isBrowserSupported = _ref.isBrowserSupported, @@ -21334,9 +21981,9 @@ transcript = voiceListeningState.transcript, isSpeechFinal = voiceListeningState.isSpeechFinal, errorCode = voiceListeningState.errorCode; - return v("div", { + return h("div", { className: cssClasses.root - }, v(Template, { + }, h(Template, { templateKey: "buttonText", rootTagName: "button", rootProps: { @@ -21355,7 +22002,7 @@ isBrowserSupported: isBrowserSupported }, templates: templates - }), v(Template, { + }), h(Template, { templateKey: "status", rootProps: { className: cssClasses.status @@ -21372,22 +22019,91 @@ })); }; - var getButtonInnerElement = function getButtonInnerElement(status, errorCode, isListening) { + var _ref2$1 = h(p, null, h("line", { + x1: "1", + y1: "1", + x2: "23", + y2: "23" + }), h("path", { + d: "M9 9v3a3 3 0 0 0 5.12 2.12M15 9.34V4a3 3 0 0 0-5.94-.6" + }), h("path", { + d: "M17 16.95A7 7 0 0 1 5 12v-2m14 0v2a7 7 0 0 1-.11 1.23" + }), h("line", { + x1: "12", + y1: "19", + x2: "12", + y2: "23" + }), h("line", { + x1: "8", + y1: "23", + x2: "16", + y2: "23" + })); + + var _ref3$1 = h("path", { + d: "M19 10v2a7 7 0 0 1-14 0v-2" + }); + + var _ref4$2 = h("line", { + x1: "12", + y1: "19", + x2: "12", + y2: "23" + }); + + var _ref5 = h("line", { + x1: "8", + y1: "23", + x2: "16", + y2: "23" + }); + + var ButtonInnerElement = function ButtonInnerElement(_ref) { + var status = _ref.status, + errorCode = _ref.errorCode, + isListening = _ref.isListening; + if (status === 'error' && errorCode === 'not-allowed') { - return "\n \n \n \n "; + return _ref2$1; } - return "\n \n \n \n "); + return h(p, null, h("path", { + d: "M12 1a3 3 0 0 0-3 3v8a3 3 0 0 0 6 0V4a3 3 0 0 0-3-3z", + fill: isListening ? 'currentColor' : 'none' + }), _ref3$1, _ref4$2, _ref5); }; var defaultTemplates$i = { - buttonText: function buttonText(_ref) { - var status = _ref.status, - errorCode = _ref.errorCode, - isListening = _ref.isListening; - return "\n ".concat(getButtonInnerElement(status, errorCode, isListening), "\n "); + buttonText: function buttonText(_ref6) { + var status = _ref6.status, + errorCode = _ref6.errorCode, + isListening = _ref6.isListening; + return h("svg", { + width: "16", + height: "16", + viewBox: "0 0 24 24", + fill: "none", + stroke: "currentColor" + /* eslint-disable react/no-unknown-property */ + // Preact supports kebab case attributes, and using camel case would + // require using `preact/compat`. + // @TODO: reconsider using the `react` ESLint preset + , + "stroke-width": "2", + "stroke-linecap": "round", + "stroke-linejoin": "round" + /* eslint-enable react/no-unknown-property */ + + }, h(ButtonInnerElement, { + status: status, + errorCode: errorCode, + isListening: isListening + })); }, - status: "

{{transcript}}

" + status: function status(_ref7) { + var transcript = _ref7.transcript; + return h("p", null, transcript); + } }; var withUsage$U = createDocumentationMessageGenerator({ @@ -21404,7 +22120,7 @@ isListening = _ref2.isListening, toggleListening = _ref2.toggleListening, voiceListeningState = _ref2.voiceListeningState; - S(v(VoiceSearch, { + P(h(VoiceSearch, { cssClasses: cssClasses, templates: templates, isBrowserSupported: isBrowserSupported, @@ -21434,11 +22150,11 @@ var containerNode = getContainerNode(container); var cssClasses = { - root: classnames(suit$u(), userCssClasses.root), - button: classnames(suit$u({ + root: cx(suit$u(), userCssClasses.root), + button: cx(suit$u({ descendantName: 'button' }), userCssClasses.button), - status: classnames(suit$u({ + status: cx(suit$u({ descendantName: 'status' }), userCssClasses.status) }; @@ -21451,7 +22167,7 @@ templates: templates }); var makeWidget = connectVoiceSearch(specializedRenderer, function () { - return S(null, containerNode); + return P(null, containerNode); }); return _objectSpread2(_objectSpread2({}, makeWidget({ container: containerNode, @@ -21508,7 +22224,7 @@ voiceSearch: voiceSearch }); - var createInsightsMiddleware = function createInsightsMiddleware(props) { + function createInsightsMiddleware(props) { var _ref = props || {}, _insightsClient = _ref.insightsClient, insightsInitParams = _ref.insightsInitParams, @@ -21573,31 +22289,28 @@ appId: appId, apiKey: apiKey }, insightsInitParams)); - var createWidget = connectConfigure(noop); - var configureClickAnalytics; - var configureUserToken; + var initialParameters; + var helper; return { onStateChange: function onStateChange() {}, - subscribe: function subscribe() { + subscribe: function subscribe() {}, + started: function started() { insightsClient('addAlgoliaAgent', 'insights-middleware'); - configureClickAnalytics = createWidget({ - searchParameters: { - clickAnalytics: true - } - }); - instantSearchInstance.addWidgets([configureClickAnalytics]); + helper = instantSearchInstance.helper; + initialParameters = { + userToken: helper.state.userToken, + clickAnalytics: helper.state.clickAnalytics + }; + helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread2(_objectSpread2({}, helper.state), {}, { + clickAnalytics: true + })); + instantSearchInstance.scheduleSearch(); var setUserTokenToSearch = function setUserTokenToSearch(userToken) { - if (configureUserToken) { - instantSearchInstance.removeWidgets([configureUserToken]); - } - - configureUserToken = createWidget({ - searchParameters: { - userToken: userToken - } - }); - instantSearchInstance.addWidgets([configureUserToken]); + helper.overrideStateWithoutTriggeringChangeEvent(_objectSpread2(_objectSpread2({}, helper.state), {}, { + userToken: userToken + })); + instantSearchInstance.scheduleSearch(); }; var anonymousUserToken = getInsightsAnonymousUserTokenInternal(); @@ -21625,9 +22338,7 @@ if (onEvent) { onEvent(event, _insightsClient); } else if (event.insightsMethod) { - // At this point, instantSearchInstance must be started and - // it means there is a configure widget (added above). - var hasUserToken = Boolean(instantSearchInstance.renderState[instantSearchInstance.indexName].configure.widgetParams.searchParameters.userToken); + var hasUserToken = Boolean(helper.state.userToken); if (hasUserToken) { insightsClient(event.insightsMethod, event.payload); @@ -21641,14 +22352,16 @@ }, unsubscribe: function unsubscribe() { insightsClient('onUserTokenChange', undefined); - instantSearchInstance.removeWidgets([configureClickAnalytics, configureUserToken]); - configureClickAnalytics = undefined; - configureUserToken = undefined; instantSearchInstance.sendEventToInsights = noop; + + if (helper && initialParameters) { + helper.setState(_objectSpread2(_objectSpread2({}, helper.state), initialParameters)); + instantSearchInstance.scheduleSearch(); + } } }; }; - }; + } diff --git a/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js.map b/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js.map index d3b1334..66e5a7e 100644 --- a/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js.map +++ b/wp-search-with-algolia/js/instantsearch.js/dist/instantsearch.development.js.map @@ -1 +1 @@ -{"version":3,"file":"instantsearch.development.js","sources":["../node_modules/algoliasearch-helper/src/functions/merge.js","../node_modules/algoliasearch-helper/src/functions/defaultsPure.js","../node_modules/algoliasearch-helper/src/functions/intersection.js","../node_modules/algoliasearch-helper/src/functions/find.js","../node_modules/algoliasearch-helper/src/functions/valToNumber.js","../node_modules/algoliasearch-helper/src/functions/omit.js","../node_modules/algoliasearch-helper/src/functions/objectHasKeys.js","../node_modules/algoliasearch-helper/src/utils/isValidUserToken.js","../node_modules/algoliasearch-helper/src/SearchParameters/RefinementList.js","../node_modules/algoliasearch-helper/src/SearchParameters/index.js","../node_modules/algoliasearch-helper/src/functions/orderBy.js","../node_modules/algoliasearch-helper/src/functions/compact.js","../node_modules/algoliasearch-helper/src/functions/findIndex.js","../node_modules/algoliasearch-helper/src/functions/formatSort.js","../node_modules/algoliasearch-helper/src/functions/escapeFacetValue.js","../node_modules/algoliasearch-helper/src/SearchResults/generate-hierarchical-tree.js","../node_modules/algoliasearch-helper/src/SearchResults/index.js","../node_modules/@algolia/events/events.js","../node_modules/algoliasearch-helper/src/functions/inherits.js","../node_modules/algoliasearch-helper/src/DerivedHelper/index.js","../node_modules/algoliasearch-helper/src/requestBuilder.js","../node_modules/algoliasearch-helper/src/version.js","../node_modules/algoliasearch-helper/src/algoliasearch.helper.js","../node_modules/algoliasearch-helper/index.js","../src/lib/utils/capitalize.ts","../src/lib/utils/defer.ts","../src/lib/utils/isDomElement.ts","../src/lib/utils/getContainerNode.ts","../src/lib/utils/isSpecialClick.ts","../src/lib/utils/uniq.ts","../src/lib/utils/prepareTemplateProps.ts","../node_modules/hogan.js/lib/compiler.js","../node_modules/hogan.js/lib/template.js","../node_modules/hogan.js/lib/hogan.js","../src/lib/utils/renderTemplate.ts","../src/lib/utils/find.ts","../src/lib/utils/escapeFacetValue.ts","../src/lib/utils/getRefinements.ts","../src/lib/utils/clearRefinements.ts","../src/lib/utils/getObjectType.ts","../src/lib/utils/checkRendering.ts","../src/lib/utils/noop.ts","../src/lib/utils/logger.ts","../src/lib/utils/typedObject.ts","../src/lib/utils/checkIndexUiState.ts","../src/lib/utils/getPropertyByPath.ts","../src/lib/utils/isFiniteNumber.ts","../src/lib/utils/isPlainObject.ts","../src/lib/utils/range.ts","../src/lib/utils/isEqual.ts","../src/lib/utils/escape.ts","../src/lib/utils/unescape.ts","../src/lib/utils/escape-highlight.ts","../src/lib/utils/concatHighlightedParts.ts","../src/lib/utils/getHighlightedParts.ts","../src/lib/utils/getHighlightFromSiblings.ts","../src/lib/utils/reverseHighlightedParts.ts","../src/lib/utils/findIndex.ts","../src/lib/utils/mergeSearchParameters.ts","../src/lib/utils/resolveSearchParameters.ts","../src/lib/utils/toArray.ts","../src/lib/utils/documentation.ts","../src/lib/utils/geo-search.ts","../src/lib/utils/hits-absolute-position.ts","../src/lib/utils/hits-query-id.ts","../src/lib/utils/isFacetRefined.ts","../src/lib/utils/createSendEventForFacet.ts","../src/lib/utils/serializer.ts","../src/lib/utils/createSendEventForHits.ts","../src/lib/utils/getAppIdAndApiKey.ts","../src/lib/utils/convertNumericRefinementsToFilters.ts","../src/lib/utils/createConcurrentSafePromise.ts","../src/lib/utils/debounce.ts","../src/lib/utils/getWidgetAttribute.ts","../src/lib/utils/safelyRunOnBrowser.ts","../src/widgets/index/index.ts","../src/lib/version.ts","../src/lib/suit.ts","../src/helpers/highlight.ts","../src/helpers/reverseHighlight.ts","../src/helpers/snippet.ts","../src/helpers/reverseSnippet.ts","../src/helpers/insights.ts","../src/helpers/get-insights-anonymous-user-token.ts","../src/lib/createHelpers.ts","../src/lib/stateMappings/simple.ts","../node_modules/qs/lib/formats.js","../node_modules/qs/lib/utils.js","../node_modules/qs/lib/stringify.js","../node_modules/qs/lib/parse.js","../node_modules/qs/lib/index.js","../src/lib/routers/history.ts","../src/middlewares/createRouterMiddleware.ts","../src/middlewares/createMetadataMiddleware.ts","../src/lib/InstantSearch.ts","../src/connectors/clear-refinements/connectClearRefinements.ts","../src/connectors/current-refinements/connectCurrentRefinements.ts","../src/connectors/hierarchical-menu/connectHierarchicalMenu.ts","../src/connectors/hits/connectHits.ts","../src/lib/insights/client.ts","../node_modules/preact/dist/preact.module.js","../src/lib/insights/listener.tsx","../src/connectors/hits/connectHitsWithInsights.ts","../src/connectors/hits-per-page/connectHitsPerPage.ts","../src/connectors/infinite-hits/connectInfiniteHits.ts","../src/connectors/infinite-hits/connectInfiniteHitsWithInsights.ts","../src/connectors/menu/connectMenu.ts","../src/connectors/numeric-menu/connectNumericMenu.ts","../src/connectors/pagination/Paginator.ts","../src/connectors/pagination/connectPagination.ts","../src/connectors/range/connectRange.ts","../src/connectors/refinement-list/connectRefinementList.ts","../src/connectors/search-box/connectSearchBox.ts","../src/connectors/sort-by/connectSortBy.ts","../src/connectors/rating-menu/connectRatingMenu.ts","../src/connectors/stats/connectStats.ts","../src/connectors/toggle-refinement/connectToggleRefinement.ts","../src/connectors/breadcrumb/connectBreadcrumb.ts","../src/connectors/geo-search/connectGeoSearch.ts","../src/connectors/powered-by/connectPoweredBy.ts","../src/connectors/configure/connectConfigure.ts","../src/connectors/configure-related-items/connectConfigureRelatedItems.ts","../src/connectors/autocomplete/connectAutocomplete.ts","../src/connectors/query-rules/connectQueryRules.ts","../src/lib/voiceSearchHelper/index.ts","../src/connectors/voice-search/connectVoiceSearch.ts","../src/connectors/answers/connectAnswers.ts","../src/connectors/relevant-sort/connectRelevantSort.ts","../src/connectors/dynamic-widgets/connectDynamicWidgets.ts","../src/connectors/index.ts","../src/widgets/analytics/analytics.ts","../node_modules/classnames/index.js","../src/components/Template/Template.tsx","../src/components/Breadcrumb/Breadcrumb.tsx","../src/widgets/breadcrumb/defaultTemplates.ts","../src/widgets/breadcrumb/breadcrumb.tsx","../src/components/ClearRefinements/ClearRefinements.tsx","../src/widgets/clear-refinements/defaultTemplates.ts","../src/widgets/clear-refinements/clear-refinements.tsx","../src/widgets/configure/configure.ts","../src/components/CurrentRefinements/CurrentRefinements.tsx","../src/widgets/current-refinements/current-refinements.tsx","../src/widgets/answers/defaultTemplates.ts","../src/components/Answers/Answers.tsx","../src/widgets/answers/answers.tsx","../src/widgets/configure-related-items/configure-related-items.ts","../src/widgets/dynamic-widgets/dynamic-widgets.ts","../src/components/GeoSearchControls/GeoSearchButton.tsx","../src/components/GeoSearchControls/GeoSearchToggle.tsx","../src/components/GeoSearchControls/GeoSearchControls.tsx","../src/widgets/geo-search/GeoSearchRenderer.js","../src/widgets/geo-search/defaultTemplates.ts","../src/widgets/geo-search/createHTMLMarker.ts","../src/widgets/geo-search/geo-search.ts","../src/components/RefinementList/RefinementListItem.tsx","../src/components/SearchBox/SearchBox.tsx","../src/components/RefinementList/RefinementList.tsx","../src/widgets/hierarchical-menu/defaultTemplates.ts","../src/widgets/hierarchical-menu/hierarchical-menu.tsx","../src/components/Hits/Hits.tsx","../src/widgets/hits/defaultTemplates.ts","../src/widgets/hits/hits.tsx","../src/components/Selector/Selector.tsx","../src/widgets/hits-per-page/hits-per-page.tsx","../src/components/InfiniteHits/InfiniteHits.tsx","../src/widgets/infinite-hits/defaultTemplates.ts","../src/widgets/infinite-hits/infinite-hits.tsx","../src/widgets/menu/defaultTemplates.ts","../src/widgets/menu/menu.tsx","../src/components/MenuSelect/MenuSelect.tsx","../src/widgets/menu-select/defaultTemplates.ts","../src/widgets/menu-select/menu-select.tsx","../src/widgets/numeric-menu/defaultTemplates.ts","../src/widgets/numeric-menu/numeric-menu.tsx","../src/components/Pagination/Pagination.tsx","../src/widgets/pagination/pagination.tsx","../node_modules/preact/hooks/dist/hooks.module.js","../src/components/Panel/Panel.tsx","../src/widgets/panel/panel.tsx","../src/widgets/places/places.ts","../src/components/PoweredBy/PoweredBy.tsx","../src/widgets/powered-by/powered-by.tsx","../src/widgets/query-rule-context/query-rule-context.tsx","../src/components/QueryRuleCustomData/QueryRuleCustomData.tsx","../src/widgets/query-rule-custom-data/query-rule-custom-data.tsx","../src/components/RangeInput/RangeInput.tsx","../src/widgets/range-input/range-input.tsx","../src/components/Slider/Rheostat.tsx","../src/components/Slider/Pit.tsx","../src/components/Slider/Slider.tsx","../src/widgets/range-slider/range-slider.tsx","../src/widgets/rating-menu/defaultTemplates.ts","../src/widgets/rating-menu/rating-menu.tsx","../src/widgets/search-box/defaultTemplates.ts","../src/widgets/refinement-list/defaultTemplates.ts","../src/widgets/refinement-list/refinement-list.tsx","../src/components/RelevantSort/RelevantSort.tsx","../src/widgets/relevant-sort/defaultTemplates.ts","../src/widgets/relevant-sort/relevant-sort.tsx","../src/widgets/search-box/search-box.tsx","../src/widgets/sort-by/sort-by.tsx","../src/components/Stats/Stats.tsx","../src/widgets/stats/stats.tsx","../src/components/ToggleRefinement/ToggleRefinement.tsx","../src/widgets/toggle-refinement/defaultTemplates.ts","../src/widgets/toggle-refinement/toggle-refinement.tsx","../src/components/VoiceSearch/VoiceSearch.tsx","../src/widgets/voice-search/defaultTemplates.ts","../src/widgets/voice-search/voice-search.tsx","../src/widgets/index.ts","../src/middlewares/createInsightsMiddleware.ts","../src/lib/stateMappings/singleIndex.ts","../src/lib/infiniteHitsCache/sessionStorage.ts","../src/index.ts"],"sourcesContent":["'use strict';\n\nfunction clone(value) {\n if (typeof value === 'object' && value !== null) {\n return _merge(Array.isArray(value) ? [] : {}, value);\n }\n return value;\n}\n\nfunction isObjectOrArrayOrFunction(value) {\n return (\n typeof value === 'function' ||\n Array.isArray(value) ||\n Object.prototype.toString.call(value) === '[object Object]'\n );\n}\n\nfunction _merge(target, source) {\n if (target === source) {\n return target;\n }\n\n for (var key in source) {\n if (\n !Object.prototype.hasOwnProperty.call(source, key) ||\n key === '__proto__'\n ) {\n continue;\n }\n\n var sourceVal = source[key];\n var targetVal = target[key];\n\n if (typeof targetVal !== 'undefined' && typeof sourceVal === 'undefined') {\n continue;\n }\n\n if (\n isObjectOrArrayOrFunction(targetVal) &&\n isObjectOrArrayOrFunction(sourceVal)\n ) {\n target[key] = _merge(targetVal, sourceVal);\n } else {\n target[key] = clone(sourceVal);\n }\n }\n return target;\n}\n\n/**\n * This method is like Object.assign, but recursively merges own and inherited\n * enumerable keyed properties of source objects into the destination object.\n *\n * NOTE: this behaves like lodash/merge, but:\n * - does mutate functions if they are a source\n * - treats non-plain objects as plain\n * - does not work for circular objects\n * - treats sparse arrays as sparse\n * - does not convert Array-like objects (Arguments, NodeLists, etc.) to arrays\n *\n * @param {Object} object The destination object.\n * @param {...Object} [sources] The source objects.\n * @returns {Object} Returns `object`.\n */\n\nfunction merge(target) {\n if (!isObjectOrArrayOrFunction(target)) {\n target = {};\n }\n\n for (var i = 1, l = arguments.length; i < l; i++) {\n var source = arguments[i];\n\n if (isObjectOrArrayOrFunction(source)) {\n _merge(target, source);\n }\n }\n return target;\n}\n\nmodule.exports = merge;\n","'use strict';\n\n// NOTE: this behaves like lodash/defaults, but doesn't mutate the target\n// it also preserve keys order\nmodule.exports = function defaultsPure() {\n var sources = Array.prototype.slice.call(arguments);\n\n return sources.reduceRight(function(acc, source) {\n Object.keys(Object(source)).forEach(function(key) {\n if (source[key] === undefined) {\n return;\n }\n if (acc[key] !== undefined) {\n // remove if already added, so that we can add it in correct order\n delete acc[key];\n }\n acc[key] = source[key];\n });\n return acc;\n }, {});\n};\n","'use strict';\n\nfunction intersection(arr1, arr2) {\n return arr1.filter(function(value, index) {\n return (\n arr2.indexOf(value) > -1 &&\n arr1.indexOf(value) === index /* skips duplicates */\n );\n });\n}\n\nmodule.exports = intersection;\n","'use strict';\n\n// @MAJOR can be replaced by native Array#find when we change support\nmodule.exports = function find(array, comparator) {\n if (!Array.isArray(array)) {\n return undefined;\n }\n\n for (var i = 0; i < array.length; i++) {\n if (comparator(array[i])) {\n return array[i];\n }\n }\n};\n","'use strict';\n\nfunction valToNumber(v) {\n if (typeof v === 'number') {\n return v;\n } else if (typeof v === 'string') {\n return parseFloat(v);\n } else if (Array.isArray(v)) {\n return v.map(valToNumber);\n }\n\n throw new Error('The value should be a number, a parsable string or an array of those.');\n}\n\nmodule.exports = valToNumber;\n","'use strict';\n\n// https://github.com/babel/babel/blob/3aaafae053fa75febb3aa45d45b6f00646e30ba4/packages/babel-helpers/src/helpers.js#L604-L620\nfunction _objectWithoutPropertiesLoose(source, excluded) {\n if (source === null) return {};\n var target = {};\n var sourceKeys = Object.keys(source);\n var key;\n var i;\n for (i = 0; i < sourceKeys.length; i++) {\n key = sourceKeys[i];\n if (excluded.indexOf(key) >= 0) continue;\n target[key] = source[key];\n }\n return target;\n}\n\nmodule.exports = _objectWithoutPropertiesLoose;\n","'use strict';\n\nfunction objectHasKeys(obj) {\n return obj && Object.keys(obj).length > 0;\n}\n\nmodule.exports = objectHasKeys;\n","'use strict';\n\nmodule.exports = function isValidUserToken(userToken) {\n if (userToken === null) {\n return false;\n }\n return /^[a-zA-Z0-9_-]{1,64}$/.test(userToken);\n};\n","'use strict';\n\n/**\n * Functions to manipulate refinement lists\n *\n * The RefinementList is not formally defined through a prototype but is based\n * on a specific structure.\n *\n * @module SearchParameters.refinementList\n *\n * @typedef {string[]} SearchParameters.refinementList.Refinements\n * @typedef {Object.} SearchParameters.refinementList.RefinementList\n */\n\nvar defaultsPure = require('../functions/defaultsPure');\nvar omit = require('../functions/omit');\nvar objectHasKeys = require('../functions/objectHasKeys');\n\nvar lib = {\n /**\n * Adds a refinement to a RefinementList\n * @param {RefinementList} refinementList the initial list\n * @param {string} attribute the attribute to refine\n * @param {string} value the value of the refinement, if the value is not a string it will be converted\n * @return {RefinementList} a new and updated refinement list\n */\n addRefinement: function addRefinement(refinementList, attribute, value) {\n if (lib.isRefined(refinementList, attribute, value)) {\n return refinementList;\n }\n\n var valueAsString = '' + value;\n\n var facetRefinement = !refinementList[attribute] ?\n [valueAsString] :\n refinementList[attribute].concat(valueAsString);\n\n var mod = {};\n\n mod[attribute] = facetRefinement;\n\n return defaultsPure({}, mod, refinementList);\n },\n /**\n * Removes refinement(s) for an attribute:\n * - if the value is specified removes the refinement for the value on the attribute\n * - if no value is specified removes all the refinements for this attribute\n * @param {RefinementList} refinementList the initial list\n * @param {string} attribute the attribute to refine\n * @param {string} [value] the value of the refinement\n * @return {RefinementList} a new and updated refinement lst\n */\n removeRefinement: function removeRefinement(refinementList, attribute, value) {\n if (value === undefined) {\n // we use the \"filter\" form of clearRefinement, since it leaves empty values as-is\n // the form with a string will remove the attribute completely\n return lib.clearRefinement(refinementList, function(v, f) {\n return attribute === f;\n });\n }\n\n var valueAsString = '' + value;\n\n return lib.clearRefinement(refinementList, function(v, f) {\n return attribute === f && valueAsString === v;\n });\n },\n /**\n * Toggles the refinement value for an attribute.\n * @param {RefinementList} refinementList the initial list\n * @param {string} attribute the attribute to refine\n * @param {string} value the value of the refinement\n * @return {RefinementList} a new and updated list\n */\n toggleRefinement: function toggleRefinement(refinementList, attribute, value) {\n if (value === undefined) throw new Error('toggleRefinement should be used with a value');\n\n if (lib.isRefined(refinementList, attribute, value)) {\n return lib.removeRefinement(refinementList, attribute, value);\n }\n\n return lib.addRefinement(refinementList, attribute, value);\n },\n /**\n * Clear all or parts of a RefinementList. Depending on the arguments, three\n * kinds of behavior can happen:\n * - if no attribute is provided: clears the whole list\n * - if an attribute is provided as a string: clears the list for the specific attribute\n * - if an attribute is provided as a function: discards the elements for which the function returns true\n * @param {RefinementList} refinementList the initial list\n * @param {string} [attribute] the attribute or function to discard\n * @param {string} [refinementType] optional parameter to give more context to the attribute function\n * @return {RefinementList} a new and updated refinement list\n */\n clearRefinement: function clearRefinement(refinementList, attribute, refinementType) {\n if (attribute === undefined) {\n if (!objectHasKeys(refinementList)) {\n return refinementList;\n }\n return {};\n } else if (typeof attribute === 'string') {\n return omit(refinementList, [attribute]);\n } else if (typeof attribute === 'function') {\n var hasChanged = false;\n\n var newRefinementList = Object.keys(refinementList).reduce(function(memo, key) {\n var values = refinementList[key] || [];\n var facetList = values.filter(function(value) {\n return !attribute(value, key, refinementType);\n });\n\n if (facetList.length !== values.length) {\n hasChanged = true;\n }\n memo[key] = facetList;\n\n return memo;\n }, {});\n\n if (hasChanged) return newRefinementList;\n return refinementList;\n }\n },\n /**\n * Test if the refinement value is used for the attribute. If no refinement value\n * is provided, test if the refinementList contains any refinement for the\n * given attribute.\n * @param {RefinementList} refinementList the list of refinement\n * @param {string} attribute name of the attribute\n * @param {string} [refinementValue] value of the filter/refinement\n * @return {boolean}\n */\n isRefined: function isRefined(refinementList, attribute, refinementValue) {\n var containsRefinements = !!refinementList[attribute] &&\n refinementList[attribute].length > 0;\n\n if (refinementValue === undefined || !containsRefinements) {\n return containsRefinements;\n }\n\n var refinementValueAsString = '' + refinementValue;\n\n return refinementList[attribute].indexOf(refinementValueAsString) !== -1;\n }\n};\n\nmodule.exports = lib;\n","'use strict';\n\nvar merge = require('../functions/merge');\nvar defaultsPure = require('../functions/defaultsPure');\nvar intersection = require('../functions/intersection');\nvar find = require('../functions/find');\nvar valToNumber = require('../functions/valToNumber');\nvar omit = require('../functions/omit');\nvar objectHasKeys = require('../functions/objectHasKeys');\nvar isValidUserToken = require('../utils/isValidUserToken');\n\nvar RefinementList = require('./RefinementList');\n\n/**\n * isEqual, but only for numeric refinement values, possible values:\n * - 5\n * - [5]\n * - [[5]]\n * - [[5,5],[4]]\n */\nfunction isEqualNumericRefinement(a, b) {\n if (Array.isArray(a) && Array.isArray(b)) {\n return (\n a.length === b.length &&\n a.every(function(el, i) {\n return isEqualNumericRefinement(b[i], el);\n })\n );\n }\n return a === b;\n}\n\n/**\n * like _.find but using deep equality to be able to use it\n * to find arrays.\n * @private\n * @param {any[]} array array to search into (elements are base or array of base)\n * @param {any} searchedValue the value we're looking for (base or array of base)\n * @return {any} the searched value or undefined\n */\nfunction findArray(array, searchedValue) {\n return find(array, function(currentValue) {\n return isEqualNumericRefinement(currentValue, searchedValue);\n });\n}\n\n/**\n * The facet list is the structure used to store the list of values used to\n * filter a single attribute.\n * @typedef {string[]} SearchParameters.FacetList\n */\n\n/**\n * Structure to store numeric filters with the operator as the key. The supported operators\n * are `=`, `>`, `<`, `>=`, `<=` and `!=`.\n * @typedef {Object.>} SearchParameters.OperatorList\n */\n\n/**\n * SearchParameters is the data structure that contains all the information\n * usable for making a search to Algolia API. It doesn't do the search itself,\n * nor does it contains logic about the parameters.\n * It is an immutable object, therefore it has been created in a way that each\n * changes does not change the object itself but returns a copy with the\n * modification.\n * This object should probably not be instantiated outside of the helper. It will\n * be provided when needed. This object is documented for reference as you'll\n * get it from events generated by the {@link AlgoliaSearchHelper}.\n * If need be, instantiate the Helper from the factory function {@link SearchParameters.make}\n * @constructor\n * @classdesc contains all the parameters of a search\n * @param {object|SearchParameters} newParameters existing parameters or partial object\n * for the properties of a new SearchParameters\n * @see SearchParameters.make\n * @example SearchParameters of the first query in\n * the instant search demo\n{\n \"query\": \"\",\n \"disjunctiveFacets\": [\n \"customerReviewCount\",\n \"category\",\n \"salePrice_range\",\n \"manufacturer\"\n ],\n \"maxValuesPerFacet\": 30,\n \"page\": 0,\n \"hitsPerPage\": 10,\n \"facets\": [\n \"type\",\n \"shipping\"\n ]\n}\n */\nfunction SearchParameters(newParameters) {\n var params = newParameters ? SearchParameters._parseNumbers(newParameters) : {};\n\n if (params.userToken !== undefined && !isValidUserToken(params.userToken)) {\n console.warn('[algoliasearch-helper] The `userToken` parameter is invalid. This can lead to wrong analytics.\\n - Format: [a-zA-Z0-9_-]{1,64}');\n }\n /**\n * This attribute contains the list of all the conjunctive facets\n * used. This list will be added to requested facets in the\n * [facets attribute](https://www.algolia.com/doc/rest-api/search#param-facets) sent to algolia.\n * @member {string[]}\n */\n this.facets = params.facets || [];\n /**\n * This attribute contains the list of all the disjunctive facets\n * used. This list will be added to requested facets in the\n * [facets attribute](https://www.algolia.com/doc/rest-api/search#param-facets) sent to algolia.\n * @member {string[]}\n */\n this.disjunctiveFacets = params.disjunctiveFacets || [];\n /**\n * This attribute contains the list of all the hierarchical facets\n * used. This list will be added to requested facets in the\n * [facets attribute](https://www.algolia.com/doc/rest-api/search#param-facets) sent to algolia.\n * Hierarchical facets are a sub type of disjunctive facets that\n * let you filter faceted attributes hierarchically.\n * @member {string[]|object[]}\n */\n this.hierarchicalFacets = params.hierarchicalFacets || [];\n\n // Refinements\n /**\n * This attribute contains all the filters that need to be\n * applied on the conjunctive facets. Each facet must be properly\n * defined in the `facets` attribute.\n *\n * The key is the name of the facet, and the `FacetList` contains all\n * filters selected for the associated facet name.\n *\n * When querying algolia, the values stored in this attribute will\n * be translated into the `facetFilters` attribute.\n * @member {Object.}\n */\n this.facetsRefinements = params.facetsRefinements || {};\n /**\n * This attribute contains all the filters that need to be\n * excluded from the conjunctive facets. Each facet must be properly\n * defined in the `facets` attribute.\n *\n * The key is the name of the facet, and the `FacetList` contains all\n * filters excluded for the associated facet name.\n *\n * When querying algolia, the values stored in this attribute will\n * be translated into the `facetFilters` attribute.\n * @member {Object.}\n */\n this.facetsExcludes = params.facetsExcludes || {};\n /**\n * This attribute contains all the filters that need to be\n * applied on the disjunctive facets. Each facet must be properly\n * defined in the `disjunctiveFacets` attribute.\n *\n * The key is the name of the facet, and the `FacetList` contains all\n * filters selected for the associated facet name.\n *\n * When querying algolia, the values stored in this attribute will\n * be translated into the `facetFilters` attribute.\n * @member {Object.}\n */\n this.disjunctiveFacetsRefinements = params.disjunctiveFacetsRefinements || {};\n /**\n * This attribute contains all the filters that need to be\n * applied on the numeric attributes.\n *\n * The key is the name of the attribute, and the value is the\n * filters to apply to this attribute.\n *\n * When querying algolia, the values stored in this attribute will\n * be translated into the `numericFilters` attribute.\n * @member {Object.}\n */\n this.numericRefinements = params.numericRefinements || {};\n /**\n * This attribute contains all the tags used to refine the query.\n *\n * When querying algolia, the values stored in this attribute will\n * be translated into the `tagFilters` attribute.\n * @member {string[]}\n */\n this.tagRefinements = params.tagRefinements || [];\n /**\n * This attribute contains all the filters that need to be\n * applied on the hierarchical facets. Each facet must be properly\n * defined in the `hierarchicalFacets` attribute.\n *\n * The key is the name of the facet, and the `FacetList` contains all\n * filters selected for the associated facet name. The FacetList values\n * are structured as a string that contain the values for each level\n * separated by the configured separator.\n *\n * When querying algolia, the values stored in this attribute will\n * be translated into the `facetFilters` attribute.\n * @member {Object.}\n */\n this.hierarchicalFacetsRefinements = params.hierarchicalFacetsRefinements || {};\n\n var self = this;\n Object.keys(params).forEach(function(paramName) {\n var isKeyKnown = SearchParameters.PARAMETERS.indexOf(paramName) !== -1;\n var isValueDefined = params[paramName] !== undefined;\n\n if (!isKeyKnown && isValueDefined) {\n self[paramName] = params[paramName];\n }\n });\n}\n\n/**\n * List all the properties in SearchParameters and therefore all the known Algolia properties\n * This doesn't contain any beta/hidden features.\n * @private\n */\nSearchParameters.PARAMETERS = Object.keys(new SearchParameters());\n\n/**\n * @private\n * @param {object} partialState full or part of a state\n * @return {object} a new object with the number keys as number\n */\nSearchParameters._parseNumbers = function(partialState) {\n // Do not reparse numbers in SearchParameters, they ought to be parsed already\n if (partialState instanceof SearchParameters) return partialState;\n\n var numbers = {};\n\n var numberKeys = [\n 'aroundPrecision',\n 'aroundRadius',\n 'getRankingInfo',\n 'minWordSizefor2Typos',\n 'minWordSizefor1Typo',\n 'page',\n 'maxValuesPerFacet',\n 'distinct',\n 'minimumAroundRadius',\n 'hitsPerPage',\n 'minProximity'\n ];\n\n numberKeys.forEach(function(k) {\n var value = partialState[k];\n if (typeof value === 'string') {\n var parsedValue = parseFloat(value);\n // global isNaN is ok to use here, value is only number or NaN\n numbers[k] = isNaN(parsedValue) ? value : parsedValue;\n }\n });\n\n // there's two formats of insideBoundingBox, we need to parse\n // the one which is an array of float geo rectangles\n if (Array.isArray(partialState.insideBoundingBox)) {\n numbers.insideBoundingBox = partialState.insideBoundingBox.map(function(geoRect) {\n if (Array.isArray(geoRect)) {\n return geoRect.map(function(value) {\n return parseFloat(value);\n });\n }\n return geoRect;\n });\n }\n\n if (partialState.numericRefinements) {\n var numericRefinements = {};\n Object.keys(partialState.numericRefinements).forEach(function(attribute) {\n var operators = partialState.numericRefinements[attribute] || {};\n numericRefinements[attribute] = {};\n Object.keys(operators).forEach(function(operator) {\n var values = operators[operator];\n var parsedValues = values.map(function(v) {\n if (Array.isArray(v)) {\n return v.map(function(vPrime) {\n if (typeof vPrime === 'string') {\n return parseFloat(vPrime);\n }\n return vPrime;\n });\n } else if (typeof v === 'string') {\n return parseFloat(v);\n }\n return v;\n });\n numericRefinements[attribute][operator] = parsedValues;\n });\n });\n numbers.numericRefinements = numericRefinements;\n }\n\n return merge({}, partialState, numbers);\n};\n\n/**\n * Factory for SearchParameters\n * @param {object|SearchParameters} newParameters existing parameters or partial\n * object for the properties of a new SearchParameters\n * @return {SearchParameters} frozen instance of SearchParameters\n */\nSearchParameters.make = function makeSearchParameters(newParameters) {\n var instance = new SearchParameters(newParameters);\n\n var hierarchicalFacets = newParameters.hierarchicalFacets || [];\n hierarchicalFacets.forEach(function(facet) {\n if (facet.rootPath) {\n var currentRefinement = instance.getHierarchicalRefinement(facet.name);\n\n if (currentRefinement.length > 0 && currentRefinement[0].indexOf(facet.rootPath) !== 0) {\n instance = instance.clearRefinements(facet.name);\n }\n\n // get it again in case it has been cleared\n currentRefinement = instance.getHierarchicalRefinement(facet.name);\n if (currentRefinement.length === 0) {\n instance = instance.toggleHierarchicalFacetRefinement(facet.name, facet.rootPath);\n }\n }\n });\n\n return instance;\n};\n\n/**\n * Validates the new parameters based on the previous state\n * @param {SearchParameters} currentState the current state\n * @param {object|SearchParameters} parameters the new parameters to set\n * @return {Error|null} Error if the modification is invalid, null otherwise\n */\nSearchParameters.validate = function(currentState, parameters) {\n var params = parameters || {};\n\n if (currentState.tagFilters && params.tagRefinements && params.tagRefinements.length > 0) {\n return new Error(\n '[Tags] Cannot switch from the managed tag API to the advanced API. It is probably ' +\n 'an error, if it is really what you want, you should first clear the tags with clearTags method.');\n }\n\n if (currentState.tagRefinements.length > 0 && params.tagFilters) {\n return new Error(\n '[Tags] Cannot switch from the advanced tag API to the managed API. It is probably ' +\n 'an error, if it is not, you should first clear the tags with clearTags method.');\n }\n\n if (\n currentState.numericFilters &&\n params.numericRefinements &&\n objectHasKeys(params.numericRefinements)\n ) {\n return new Error(\n \"[Numeric filters] Can't switch from the advanced to the managed API. It\" +\n ' is probably an error, if this is really what you want, you have to first' +\n ' clear the numeric filters.'\n );\n }\n\n if (objectHasKeys(currentState.numericRefinements) && params.numericFilters) {\n return new Error(\n \"[Numeric filters] Can't switch from the managed API to the advanced. It\" +\n ' is probably an error, if this is really what you want, you have to first' +\n ' clear the numeric filters.');\n }\n\n return null;\n};\n\nSearchParameters.prototype = {\n constructor: SearchParameters,\n\n /**\n * Remove all refinements (disjunctive + conjunctive + excludes + numeric filters)\n * @method\n * @param {undefined|string|SearchParameters.clearCallback} [attribute] optional string or function\n * - If not given, means to clear all the filters.\n * - If `string`, means to clear all refinements for the `attribute` named filter.\n * - If `function`, means to clear all the refinements that return truthy values.\n * @return {SearchParameters}\n */\n clearRefinements: function clearRefinements(attribute) {\n var patch = {\n numericRefinements: this._clearNumericRefinements(attribute),\n facetsRefinements: RefinementList.clearRefinement(\n this.facetsRefinements,\n attribute,\n 'conjunctiveFacet'\n ),\n facetsExcludes: RefinementList.clearRefinement(\n this.facetsExcludes,\n attribute,\n 'exclude'\n ),\n disjunctiveFacetsRefinements: RefinementList.clearRefinement(\n this.disjunctiveFacetsRefinements,\n attribute,\n 'disjunctiveFacet'\n ),\n hierarchicalFacetsRefinements: RefinementList.clearRefinement(\n this.hierarchicalFacetsRefinements,\n attribute,\n 'hierarchicalFacet'\n )\n };\n if (\n patch.numericRefinements === this.numericRefinements &&\n patch.facetsRefinements === this.facetsRefinements &&\n patch.facetsExcludes === this.facetsExcludes &&\n patch.disjunctiveFacetsRefinements === this.disjunctiveFacetsRefinements &&\n patch.hierarchicalFacetsRefinements === this.hierarchicalFacetsRefinements\n ) {\n return this;\n }\n return this.setQueryParameters(patch);\n },\n /**\n * Remove all the refined tags from the SearchParameters\n * @method\n * @return {SearchParameters}\n */\n clearTags: function clearTags() {\n if (this.tagFilters === undefined && this.tagRefinements.length === 0) return this;\n\n return this.setQueryParameters({\n tagFilters: undefined,\n tagRefinements: []\n });\n },\n /**\n * Set the index.\n * @method\n * @param {string} index the index name\n * @return {SearchParameters}\n */\n setIndex: function setIndex(index) {\n if (index === this.index) return this;\n\n return this.setQueryParameters({\n index: index\n });\n },\n /**\n * Query setter\n * @method\n * @param {string} newQuery value for the new query\n * @return {SearchParameters}\n */\n setQuery: function setQuery(newQuery) {\n if (newQuery === this.query) return this;\n\n return this.setQueryParameters({\n query: newQuery\n });\n },\n /**\n * Page setter\n * @method\n * @param {number} newPage new page number\n * @return {SearchParameters}\n */\n setPage: function setPage(newPage) {\n if (newPage === this.page) return this;\n\n return this.setQueryParameters({\n page: newPage\n });\n },\n /**\n * Facets setter\n * The facets are the simple facets, used for conjunctive (and) faceting.\n * @method\n * @param {string[]} facets all the attributes of the algolia records used for conjunctive faceting\n * @return {SearchParameters}\n */\n setFacets: function setFacets(facets) {\n return this.setQueryParameters({\n facets: facets\n });\n },\n /**\n * Disjunctive facets setter\n * Change the list of disjunctive (or) facets the helper chan handle.\n * @method\n * @param {string[]} facets all the attributes of the algolia records used for disjunctive faceting\n * @return {SearchParameters}\n */\n setDisjunctiveFacets: function setDisjunctiveFacets(facets) {\n return this.setQueryParameters({\n disjunctiveFacets: facets\n });\n },\n /**\n * HitsPerPage setter\n * Hits per page represents the number of hits retrieved for this query\n * @method\n * @param {number} n number of hits retrieved per page of results\n * @return {SearchParameters}\n */\n setHitsPerPage: function setHitsPerPage(n) {\n if (this.hitsPerPage === n) return this;\n\n return this.setQueryParameters({\n hitsPerPage: n\n });\n },\n /**\n * typoTolerance setter\n * Set the value of typoTolerance\n * @method\n * @param {string} typoTolerance new value of typoTolerance (\"true\", \"false\", \"min\" or \"strict\")\n * @return {SearchParameters}\n */\n setTypoTolerance: function setTypoTolerance(typoTolerance) {\n if (this.typoTolerance === typoTolerance) return this;\n\n return this.setQueryParameters({\n typoTolerance: typoTolerance\n });\n },\n /**\n * Add a numeric filter for a given attribute\n * When value is an array, they are combined with OR\n * When value is a single value, it will combined with AND\n * @method\n * @param {string} attribute attribute to set the filter on\n * @param {string} operator operator of the filter (possible values: =, >, >=, <, <=, !=)\n * @param {number | number[]} value value of the filter\n * @return {SearchParameters}\n * @example\n * // for price = 50 or 40\n * searchparameter.addNumericRefinement('price', '=', [50, 40]);\n * @example\n * // for size = 38 and 40\n * searchparameter.addNumericRefinement('size', '=', 38);\n * searchparameter.addNumericRefinement('size', '=', 40);\n */\n addNumericRefinement: function(attribute, operator, v) {\n var value = valToNumber(v);\n\n if (this.isNumericRefined(attribute, operator, value)) return this;\n\n var mod = merge({}, this.numericRefinements);\n\n mod[attribute] = merge({}, mod[attribute]);\n\n if (mod[attribute][operator]) {\n // Array copy\n mod[attribute][operator] = mod[attribute][operator].slice();\n // Add the element. Concat can't be used here because value can be an array.\n mod[attribute][operator].push(value);\n } else {\n mod[attribute][operator] = [value];\n }\n\n return this.setQueryParameters({\n numericRefinements: mod\n });\n },\n /**\n * Get the list of conjunctive refinements for a single facet\n * @param {string} facetName name of the attribute used for faceting\n * @return {string[]} list of refinements\n */\n getConjunctiveRefinements: function(facetName) {\n if (!this.isConjunctiveFacet(facetName)) {\n return [];\n }\n return this.facetsRefinements[facetName] || [];\n },\n /**\n * Get the list of disjunctive refinements for a single facet\n * @param {string} facetName name of the attribute used for faceting\n * @return {string[]} list of refinements\n */\n getDisjunctiveRefinements: function(facetName) {\n if (!this.isDisjunctiveFacet(facetName)) {\n return [];\n }\n return this.disjunctiveFacetsRefinements[facetName] || [];\n },\n /**\n * Get the list of hierarchical refinements for a single facet\n * @param {string} facetName name of the attribute used for faceting\n * @return {string[]} list of refinements\n */\n getHierarchicalRefinement: function(facetName) {\n // we send an array but we currently do not support multiple\n // hierarchicalRefinements for a hierarchicalFacet\n return this.hierarchicalFacetsRefinements[facetName] || [];\n },\n /**\n * Get the list of exclude refinements for a single facet\n * @param {string} facetName name of the attribute used for faceting\n * @return {string[]} list of refinements\n */\n getExcludeRefinements: function(facetName) {\n if (!this.isConjunctiveFacet(facetName)) {\n return [];\n }\n return this.facetsExcludes[facetName] || [];\n },\n\n /**\n * Remove all the numeric filter for a given (attribute, operator)\n * @method\n * @param {string} attribute attribute to set the filter on\n * @param {string} [operator] operator of the filter (possible values: =, >, >=, <, <=, !=)\n * @param {number} [number] the value to be removed\n * @return {SearchParameters}\n */\n removeNumericRefinement: function(attribute, operator, paramValue) {\n if (paramValue !== undefined) {\n if (!this.isNumericRefined(attribute, operator, paramValue)) {\n return this;\n }\n return this.setQueryParameters({\n numericRefinements: this._clearNumericRefinements(function(value, key) {\n return (\n key === attribute &&\n value.op === operator &&\n isEqualNumericRefinement(value.val, valToNumber(paramValue))\n );\n })\n });\n } else if (operator !== undefined) {\n if (!this.isNumericRefined(attribute, operator)) return this;\n return this.setQueryParameters({\n numericRefinements: this._clearNumericRefinements(function(value, key) {\n return key === attribute && value.op === operator;\n })\n });\n }\n\n if (!this.isNumericRefined(attribute)) return this;\n return this.setQueryParameters({\n numericRefinements: this._clearNumericRefinements(function(value, key) {\n return key === attribute;\n })\n });\n },\n /**\n * Get the list of numeric refinements for a single facet\n * @param {string} facetName name of the attribute used for faceting\n * @return {SearchParameters.OperatorList} list of refinements\n */\n getNumericRefinements: function(facetName) {\n return this.numericRefinements[facetName] || {};\n },\n /**\n * Return the current refinement for the (attribute, operator)\n * @param {string} attribute attribute in the record\n * @param {string} operator operator applied on the refined values\n * @return {Array.} refined values\n */\n getNumericRefinement: function(attribute, operator) {\n return this.numericRefinements[attribute] && this.numericRefinements[attribute][operator];\n },\n /**\n * Clear numeric filters.\n * @method\n * @private\n * @param {string|SearchParameters.clearCallback} [attribute] optional string or function\n * - If not given, means to clear all the filters.\n * - If `string`, means to clear all refinements for the `attribute` named filter.\n * - If `function`, means to clear all the refinements that return truthy values.\n * @return {Object.}\n */\n _clearNumericRefinements: function _clearNumericRefinements(attribute) {\n if (attribute === undefined) {\n if (!objectHasKeys(this.numericRefinements)) {\n return this.numericRefinements;\n }\n return {};\n } else if (typeof attribute === 'string') {\n return omit(this.numericRefinements, [attribute]);\n } else if (typeof attribute === 'function') {\n var hasChanged = false;\n var numericRefinements = this.numericRefinements;\n var newNumericRefinements = Object.keys(numericRefinements).reduce(function(memo, key) {\n var operators = numericRefinements[key];\n var operatorList = {};\n\n operators = operators || {};\n Object.keys(operators).forEach(function(operator) {\n var values = operators[operator] || [];\n var outValues = [];\n values.forEach(function(value) {\n var predicateResult = attribute({val: value, op: operator}, key, 'numeric');\n if (!predicateResult) outValues.push(value);\n });\n if (outValues.length !== values.length) {\n hasChanged = true;\n }\n operatorList[operator] = outValues;\n });\n\n memo[key] = operatorList;\n\n return memo;\n }, {});\n\n if (hasChanged) return newNumericRefinements;\n return this.numericRefinements;\n }\n },\n /**\n * Add a facet to the facets attribute of the helper configuration, if it\n * isn't already present.\n * @method\n * @param {string} facet facet name to add\n * @return {SearchParameters}\n */\n addFacet: function addFacet(facet) {\n if (this.isConjunctiveFacet(facet)) {\n return this;\n }\n\n return this.setQueryParameters({\n facets: this.facets.concat([facet])\n });\n },\n /**\n * Add a disjunctive facet to the disjunctiveFacets attribute of the helper\n * configuration, if it isn't already present.\n * @method\n * @param {string} facet disjunctive facet name to add\n * @return {SearchParameters}\n */\n addDisjunctiveFacet: function addDisjunctiveFacet(facet) {\n if (this.isDisjunctiveFacet(facet)) {\n return this;\n }\n\n return this.setQueryParameters({\n disjunctiveFacets: this.disjunctiveFacets.concat([facet])\n });\n },\n /**\n * Add a hierarchical facet to the hierarchicalFacets attribute of the helper\n * configuration.\n * @method\n * @param {object} hierarchicalFacet hierarchical facet to add\n * @return {SearchParameters}\n * @throws will throw an error if a hierarchical facet with the same name was already declared\n */\n addHierarchicalFacet: function addHierarchicalFacet(hierarchicalFacet) {\n if (this.isHierarchicalFacet(hierarchicalFacet.name)) {\n throw new Error(\n 'Cannot declare two hierarchical facets with the same name: `' + hierarchicalFacet.name + '`');\n }\n\n return this.setQueryParameters({\n hierarchicalFacets: this.hierarchicalFacets.concat([hierarchicalFacet])\n });\n },\n /**\n * Add a refinement on a \"normal\" facet\n * @method\n * @param {string} facet attribute to apply the faceting on\n * @param {string} value value of the attribute (will be converted to string)\n * @return {SearchParameters}\n */\n addFacetRefinement: function addFacetRefinement(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n throw new Error(facet + ' is not defined in the facets attribute of the helper configuration');\n }\n if (RefinementList.isRefined(this.facetsRefinements, facet, value)) return this;\n\n return this.setQueryParameters({\n facetsRefinements: RefinementList.addRefinement(this.facetsRefinements, facet, value)\n });\n },\n /**\n * Exclude a value from a \"normal\" facet\n * @method\n * @param {string} facet attribute to apply the exclusion on\n * @param {string} value value of the attribute (will be converted to string)\n * @return {SearchParameters}\n */\n addExcludeRefinement: function addExcludeRefinement(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n throw new Error(facet + ' is not defined in the facets attribute of the helper configuration');\n }\n if (RefinementList.isRefined(this.facetsExcludes, facet, value)) return this;\n\n return this.setQueryParameters({\n facetsExcludes: RefinementList.addRefinement(this.facetsExcludes, facet, value)\n });\n },\n /**\n * Adds a refinement on a disjunctive facet.\n * @method\n * @param {string} facet attribute to apply the faceting on\n * @param {string} value value of the attribute (will be converted to string)\n * @return {SearchParameters}\n */\n addDisjunctiveFacetRefinement: function addDisjunctiveFacetRefinement(facet, value) {\n if (!this.isDisjunctiveFacet(facet)) {\n throw new Error(\n facet + ' is not defined in the disjunctiveFacets attribute of the helper configuration');\n }\n\n if (RefinementList.isRefined(this.disjunctiveFacetsRefinements, facet, value)) return this;\n\n return this.setQueryParameters({\n disjunctiveFacetsRefinements: RefinementList.addRefinement(\n this.disjunctiveFacetsRefinements, facet, value)\n });\n },\n /**\n * addTagRefinement adds a tag to the list used to filter the results\n * @param {string} tag tag to be added\n * @return {SearchParameters}\n */\n addTagRefinement: function addTagRefinement(tag) {\n if (this.isTagRefined(tag)) return this;\n\n var modification = {\n tagRefinements: this.tagRefinements.concat(tag)\n };\n\n return this.setQueryParameters(modification);\n },\n /**\n * Remove a facet from the facets attribute of the helper configuration, if it\n * is present.\n * @method\n * @param {string} facet facet name to remove\n * @return {SearchParameters}\n */\n removeFacet: function removeFacet(facet) {\n if (!this.isConjunctiveFacet(facet)) {\n return this;\n }\n\n return this.clearRefinements(facet).setQueryParameters({\n facets: this.facets.filter(function(f) {\n return f !== facet;\n })\n });\n },\n /**\n * Remove a disjunctive facet from the disjunctiveFacets attribute of the\n * helper configuration, if it is present.\n * @method\n * @param {string} facet disjunctive facet name to remove\n * @return {SearchParameters}\n */\n removeDisjunctiveFacet: function removeDisjunctiveFacet(facet) {\n if (!this.isDisjunctiveFacet(facet)) {\n return this;\n }\n\n return this.clearRefinements(facet).setQueryParameters({\n disjunctiveFacets: this.disjunctiveFacets.filter(function(f) {\n return f !== facet;\n })\n });\n },\n /**\n * Remove a hierarchical facet from the hierarchicalFacets attribute of the\n * helper configuration, if it is present.\n * @method\n * @param {string} facet hierarchical facet name to remove\n * @return {SearchParameters}\n */\n removeHierarchicalFacet: function removeHierarchicalFacet(facet) {\n if (!this.isHierarchicalFacet(facet)) {\n return this;\n }\n\n return this.clearRefinements(facet).setQueryParameters({\n hierarchicalFacets: this.hierarchicalFacets.filter(function(f) {\n return f.name !== facet;\n })\n });\n },\n /**\n * Remove a refinement set on facet. If a value is provided, it will clear the\n * refinement for the given value, otherwise it will clear all the refinement\n * values for the faceted attribute.\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {string} [value] value used to filter\n * @return {SearchParameters}\n */\n removeFacetRefinement: function removeFacetRefinement(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n throw new Error(facet + ' is not defined in the facets attribute of the helper configuration');\n }\n if (!RefinementList.isRefined(this.facetsRefinements, facet, value)) return this;\n\n return this.setQueryParameters({\n facetsRefinements: RefinementList.removeRefinement(this.facetsRefinements, facet, value)\n });\n },\n /**\n * Remove a negative refinement on a facet\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {string} value value used to filter\n * @return {SearchParameters}\n */\n removeExcludeRefinement: function removeExcludeRefinement(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n throw new Error(facet + ' is not defined in the facets attribute of the helper configuration');\n }\n if (!RefinementList.isRefined(this.facetsExcludes, facet, value)) return this;\n\n return this.setQueryParameters({\n facetsExcludes: RefinementList.removeRefinement(this.facetsExcludes, facet, value)\n });\n },\n /**\n * Remove a refinement on a disjunctive facet\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {string} value value used to filter\n * @return {SearchParameters}\n */\n removeDisjunctiveFacetRefinement: function removeDisjunctiveFacetRefinement(facet, value) {\n if (!this.isDisjunctiveFacet(facet)) {\n throw new Error(\n facet + ' is not defined in the disjunctiveFacets attribute of the helper configuration');\n }\n if (!RefinementList.isRefined(this.disjunctiveFacetsRefinements, facet, value)) return this;\n\n return this.setQueryParameters({\n disjunctiveFacetsRefinements: RefinementList.removeRefinement(\n this.disjunctiveFacetsRefinements, facet, value)\n });\n },\n /**\n * Remove a tag from the list of tag refinements\n * @method\n * @param {string} tag the tag to remove\n * @return {SearchParameters}\n */\n removeTagRefinement: function removeTagRefinement(tag) {\n if (!this.isTagRefined(tag)) return this;\n\n var modification = {\n tagRefinements: this.tagRefinements.filter(function(t) {\n return t !== tag;\n })\n };\n\n return this.setQueryParameters(modification);\n },\n /**\n * Generic toggle refinement method to use with facet, disjunctive facets\n * and hierarchical facets\n * @param {string} facet the facet to refine\n * @param {string} value the associated value\n * @return {SearchParameters}\n * @throws will throw an error if the facet is not declared in the settings of the helper\n * @deprecated since version 2.19.0, see {@link SearchParameters#toggleFacetRefinement}\n */\n toggleRefinement: function toggleRefinement(facet, value) {\n return this.toggleFacetRefinement(facet, value);\n },\n /**\n * Generic toggle refinement method to use with facet, disjunctive facets\n * and hierarchical facets\n * @param {string} facet the facet to refine\n * @param {string} value the associated value\n * @return {SearchParameters}\n * @throws will throw an error if the facet is not declared in the settings of the helper\n */\n toggleFacetRefinement: function toggleFacetRefinement(facet, value) {\n if (this.isHierarchicalFacet(facet)) {\n return this.toggleHierarchicalFacetRefinement(facet, value);\n } else if (this.isConjunctiveFacet(facet)) {\n return this.toggleConjunctiveFacetRefinement(facet, value);\n } else if (this.isDisjunctiveFacet(facet)) {\n return this.toggleDisjunctiveFacetRefinement(facet, value);\n }\n\n throw new Error('Cannot refine the undeclared facet ' + facet +\n '; it should be added to the helper options facets, disjunctiveFacets or hierarchicalFacets');\n },\n /**\n * Switch the refinement applied over a facet/value\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {value} value value used for filtering\n * @return {SearchParameters}\n */\n toggleConjunctiveFacetRefinement: function toggleConjunctiveFacetRefinement(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n throw new Error(facet + ' is not defined in the facets attribute of the helper configuration');\n }\n\n return this.setQueryParameters({\n facetsRefinements: RefinementList.toggleRefinement(this.facetsRefinements, facet, value)\n });\n },\n /**\n * Switch the refinement applied over a facet/value\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {value} value value used for filtering\n * @return {SearchParameters}\n */\n toggleExcludeFacetRefinement: function toggleExcludeFacetRefinement(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n throw new Error(facet + ' is not defined in the facets attribute of the helper configuration');\n }\n\n return this.setQueryParameters({\n facetsExcludes: RefinementList.toggleRefinement(this.facetsExcludes, facet, value)\n });\n },\n /**\n * Switch the refinement applied over a facet/value\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {value} value value used for filtering\n * @return {SearchParameters}\n */\n toggleDisjunctiveFacetRefinement: function toggleDisjunctiveFacetRefinement(facet, value) {\n if (!this.isDisjunctiveFacet(facet)) {\n throw new Error(\n facet + ' is not defined in the disjunctiveFacets attribute of the helper configuration');\n }\n\n return this.setQueryParameters({\n disjunctiveFacetsRefinements: RefinementList.toggleRefinement(\n this.disjunctiveFacetsRefinements, facet, value)\n });\n },\n /**\n * Switch the refinement applied over a facet/value\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {value} value value used for filtering\n * @return {SearchParameters}\n */\n toggleHierarchicalFacetRefinement: function toggleHierarchicalFacetRefinement(facet, value) {\n if (!this.isHierarchicalFacet(facet)) {\n throw new Error(\n facet + ' is not defined in the hierarchicalFacets attribute of the helper configuration');\n }\n\n var separator = this._getHierarchicalFacetSeparator(this.getHierarchicalFacetByName(facet));\n\n var mod = {};\n\n var upOneOrMultipleLevel = this.hierarchicalFacetsRefinements[facet] !== undefined &&\n this.hierarchicalFacetsRefinements[facet].length > 0 && (\n // remove current refinement:\n // refinement was 'beer > IPA', call is toggleRefine('beer > IPA'), refinement should be `beer`\n this.hierarchicalFacetsRefinements[facet][0] === value ||\n // remove a parent refinement of the current refinement:\n // - refinement was 'beer > IPA > Flying dog'\n // - call is toggleRefine('beer > IPA')\n // - refinement should be `beer`\n this.hierarchicalFacetsRefinements[facet][0].indexOf(value + separator) === 0\n );\n\n if (upOneOrMultipleLevel) {\n if (value.indexOf(separator) === -1) {\n // go back to root level\n mod[facet] = [];\n } else {\n mod[facet] = [value.slice(0, value.lastIndexOf(separator))];\n }\n } else {\n mod[facet] = [value];\n }\n\n return this.setQueryParameters({\n hierarchicalFacetsRefinements: defaultsPure({}, mod, this.hierarchicalFacetsRefinements)\n });\n },\n\n /**\n * Adds a refinement on a hierarchical facet.\n * @param {string} facet the facet name\n * @param {string} path the hierarchical facet path\n * @return {SearchParameter} the new state\n * @throws Error if the facet is not defined or if the facet is refined\n */\n addHierarchicalFacetRefinement: function(facet, path) {\n if (this.isHierarchicalFacetRefined(facet)) {\n throw new Error(facet + ' is already refined.');\n }\n if (!this.isHierarchicalFacet(facet)) {\n throw new Error(facet + ' is not defined in the hierarchicalFacets attribute of the helper configuration.');\n }\n var mod = {};\n mod[facet] = [path];\n return this.setQueryParameters({\n hierarchicalFacetsRefinements: defaultsPure({}, mod, this.hierarchicalFacetsRefinements)\n });\n },\n\n /**\n * Removes the refinement set on a hierarchical facet.\n * @param {string} facet the facet name\n * @return {SearchParameter} the new state\n * @throws Error if the facet is not defined or if the facet is not refined\n */\n removeHierarchicalFacetRefinement: function(facet) {\n if (!this.isHierarchicalFacetRefined(facet)) {\n return this;\n }\n var mod = {};\n mod[facet] = [];\n return this.setQueryParameters({\n hierarchicalFacetsRefinements: defaultsPure({}, mod, this.hierarchicalFacetsRefinements)\n });\n },\n /**\n * Switch the tag refinement\n * @method\n * @param {string} tag the tag to remove or add\n * @return {SearchParameters}\n */\n toggleTagRefinement: function toggleTagRefinement(tag) {\n if (this.isTagRefined(tag)) {\n return this.removeTagRefinement(tag);\n }\n\n return this.addTagRefinement(tag);\n },\n /**\n * Test if the facet name is from one of the disjunctive facets\n * @method\n * @param {string} facet facet name to test\n * @return {boolean}\n */\n isDisjunctiveFacet: function(facet) {\n return this.disjunctiveFacets.indexOf(facet) > -1;\n },\n /**\n * Test if the facet name is from one of the hierarchical facets\n * @method\n * @param {string} facetName facet name to test\n * @return {boolean}\n */\n isHierarchicalFacet: function(facetName) {\n return this.getHierarchicalFacetByName(facetName) !== undefined;\n },\n /**\n * Test if the facet name is from one of the conjunctive/normal facets\n * @method\n * @param {string} facet facet name to test\n * @return {boolean}\n */\n isConjunctiveFacet: function(facet) {\n return this.facets.indexOf(facet) > -1;\n },\n /**\n * Returns true if the facet is refined, either for a specific value or in\n * general.\n * @method\n * @param {string} facet name of the attribute for used for faceting\n * @param {string} value, optional value. If passed will test that this value\n * is filtering the given facet.\n * @return {boolean} returns true if refined\n */\n isFacetRefined: function isFacetRefined(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n return false;\n }\n return RefinementList.isRefined(this.facetsRefinements, facet, value);\n },\n /**\n * Returns true if the facet contains exclusions or if a specific value is\n * excluded.\n *\n * @method\n * @param {string} facet name of the attribute for used for faceting\n * @param {string} [value] optional value. If passed will test that this value\n * is filtering the given facet.\n * @return {boolean} returns true if refined\n */\n isExcludeRefined: function isExcludeRefined(facet, value) {\n if (!this.isConjunctiveFacet(facet)) {\n return false;\n }\n return RefinementList.isRefined(this.facetsExcludes, facet, value);\n },\n /**\n * Returns true if the facet contains a refinement, or if a value passed is a\n * refinement for the facet.\n * @method\n * @param {string} facet name of the attribute for used for faceting\n * @param {string} value optional, will test if the value is used for refinement\n * if there is one, otherwise will test if the facet contains any refinement\n * @return {boolean}\n */\n isDisjunctiveFacetRefined: function isDisjunctiveFacetRefined(facet, value) {\n if (!this.isDisjunctiveFacet(facet)) {\n return false;\n }\n return RefinementList.isRefined(this.disjunctiveFacetsRefinements, facet, value);\n },\n /**\n * Returns true if the facet contains a refinement, or if a value passed is a\n * refinement for the facet.\n * @method\n * @param {string} facet name of the attribute for used for faceting\n * @param {string} value optional, will test if the value is used for refinement\n * if there is one, otherwise will test if the facet contains any refinement\n * @return {boolean}\n */\n isHierarchicalFacetRefined: function isHierarchicalFacetRefined(facet, value) {\n if (!this.isHierarchicalFacet(facet)) {\n return false;\n }\n\n var refinements = this.getHierarchicalRefinement(facet);\n\n if (!value) {\n return refinements.length > 0;\n }\n\n return refinements.indexOf(value) !== -1;\n },\n /**\n * Test if the triple (attribute, operator, value) is already refined.\n * If only the attribute and the operator are provided, it tests if the\n * contains any refinement value.\n * @method\n * @param {string} attribute attribute for which the refinement is applied\n * @param {string} [operator] operator of the refinement\n * @param {string} [value] value of the refinement\n * @return {boolean} true if it is refined\n */\n isNumericRefined: function isNumericRefined(attribute, operator, value) {\n if (value === undefined && operator === undefined) {\n return !!this.numericRefinements[attribute];\n }\n\n var isOperatorDefined =\n this.numericRefinements[attribute] &&\n this.numericRefinements[attribute][operator] !== undefined;\n\n if (value === undefined || !isOperatorDefined) {\n return isOperatorDefined;\n }\n\n var parsedValue = valToNumber(value);\n var isAttributeValueDefined =\n findArray(this.numericRefinements[attribute][operator], parsedValue) !==\n undefined;\n\n return isOperatorDefined && isAttributeValueDefined;\n },\n /**\n * Returns true if the tag refined, false otherwise\n * @method\n * @param {string} tag the tag to check\n * @return {boolean}\n */\n isTagRefined: function isTagRefined(tag) {\n return this.tagRefinements.indexOf(tag) !== -1;\n },\n /**\n * Returns the list of all disjunctive facets refined\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {value} value value used for filtering\n * @return {string[]}\n */\n getRefinedDisjunctiveFacets: function getRefinedDisjunctiveFacets() {\n var self = this;\n\n // attributes used for numeric filter can also be disjunctive\n var disjunctiveNumericRefinedFacets = intersection(\n Object.keys(this.numericRefinements).filter(function(facet) {\n return Object.keys(self.numericRefinements[facet]).length > 0;\n }),\n this.disjunctiveFacets\n );\n\n return Object.keys(this.disjunctiveFacetsRefinements).filter(function(facet) {\n return self.disjunctiveFacetsRefinements[facet].length > 0;\n })\n .concat(disjunctiveNumericRefinedFacets)\n .concat(this.getRefinedHierarchicalFacets());\n },\n /**\n * Returns the list of all disjunctive facets refined\n * @method\n * @param {string} facet name of the attribute used for faceting\n * @param {value} value value used for filtering\n * @return {string[]}\n */\n getRefinedHierarchicalFacets: function getRefinedHierarchicalFacets() {\n var self = this;\n return intersection(\n // enforce the order between the two arrays,\n // so that refinement name index === hierarchical facet index\n this.hierarchicalFacets.map(function(facet) { return facet.name; }),\n Object.keys(this.hierarchicalFacetsRefinements).filter(function(facet) {\n return self.hierarchicalFacetsRefinements[facet].length > 0;\n })\n );\n },\n /**\n * Returned the list of all disjunctive facets not refined\n * @method\n * @return {string[]}\n */\n getUnrefinedDisjunctiveFacets: function() {\n var refinedFacets = this.getRefinedDisjunctiveFacets();\n\n return this.disjunctiveFacets.filter(function(f) {\n return refinedFacets.indexOf(f) === -1;\n });\n },\n\n managedParameters: [\n 'index',\n\n 'facets',\n 'disjunctiveFacets',\n 'facetsRefinements',\n 'hierarchicalFacets',\n 'facetsExcludes',\n\n 'disjunctiveFacetsRefinements',\n 'numericRefinements',\n 'tagRefinements',\n 'hierarchicalFacetsRefinements'\n ],\n getQueryParams: function getQueryParams() {\n var managedParameters = this.managedParameters;\n\n var queryParams = {};\n\n var self = this;\n Object.keys(this).forEach(function(paramName) {\n var paramValue = self[paramName];\n if (managedParameters.indexOf(paramName) === -1 && paramValue !== undefined) {\n queryParams[paramName] = paramValue;\n }\n });\n\n return queryParams;\n },\n /**\n * Let the user set a specific value for a given parameter. Will return the\n * same instance if the parameter is invalid or if the value is the same as the\n * previous one.\n * @method\n * @param {string} parameter the parameter name\n * @param {any} value the value to be set, must be compliant with the definition\n * of the attribute on the object\n * @return {SearchParameters} the updated state\n */\n setQueryParameter: function setParameter(parameter, value) {\n if (this[parameter] === value) return this;\n\n var modification = {};\n\n modification[parameter] = value;\n\n return this.setQueryParameters(modification);\n },\n /**\n * Let the user set any of the parameters with a plain object.\n * @method\n * @param {object} params all the keys and the values to be updated\n * @return {SearchParameters} a new updated instance\n */\n setQueryParameters: function setQueryParameters(params) {\n if (!params) return this;\n\n var error = SearchParameters.validate(this, params);\n\n if (error) {\n throw error;\n }\n\n var self = this;\n var nextWithNumbers = SearchParameters._parseNumbers(params);\n var previousPlainObject = Object.keys(this).reduce(function(acc, key) {\n acc[key] = self[key];\n return acc;\n }, {});\n\n var nextPlainObject = Object.keys(nextWithNumbers).reduce(\n function(previous, key) {\n var isPreviousValueDefined = previous[key] !== undefined;\n var isNextValueDefined = nextWithNumbers[key] !== undefined;\n\n if (isPreviousValueDefined && !isNextValueDefined) {\n return omit(previous, [key]);\n }\n\n if (isNextValueDefined) {\n previous[key] = nextWithNumbers[key];\n }\n\n return previous;\n },\n previousPlainObject\n );\n\n return new this.constructor(nextPlainObject);\n },\n\n /**\n * Returns a new instance with the page reset. Two scenarios possible:\n * the page is omitted -> return the given instance\n * the page is set -> return a new instance with a page of 0\n * @return {SearchParameters} a new updated instance\n */\n resetPage: function() {\n if (this.page === undefined) {\n return this;\n }\n\n return this.setPage(0);\n },\n\n /**\n * Helper function to get the hierarchicalFacet separator or the default one (`>`)\n * @param {object} hierarchicalFacet\n * @return {string} returns the hierarchicalFacet.separator or `>` as default\n */\n _getHierarchicalFacetSortBy: function(hierarchicalFacet) {\n return hierarchicalFacet.sortBy || ['isRefined:desc', 'name:asc'];\n },\n\n /**\n * Helper function to get the hierarchicalFacet separator or the default one (`>`)\n * @private\n * @param {object} hierarchicalFacet\n * @return {string} returns the hierarchicalFacet.separator or `>` as default\n */\n _getHierarchicalFacetSeparator: function(hierarchicalFacet) {\n return hierarchicalFacet.separator || ' > ';\n },\n\n /**\n * Helper function to get the hierarchicalFacet prefix path or null\n * @private\n * @param {object} hierarchicalFacet\n * @return {string} returns the hierarchicalFacet.rootPath or null as default\n */\n _getHierarchicalRootPath: function(hierarchicalFacet) {\n return hierarchicalFacet.rootPath || null;\n },\n\n /**\n * Helper function to check if we show the parent level of the hierarchicalFacet\n * @private\n * @param {object} hierarchicalFacet\n * @return {string} returns the hierarchicalFacet.showParentLevel or true as default\n */\n _getHierarchicalShowParentLevel: function(hierarchicalFacet) {\n if (typeof hierarchicalFacet.showParentLevel === 'boolean') {\n return hierarchicalFacet.showParentLevel;\n }\n return true;\n },\n\n /**\n * Helper function to get the hierarchicalFacet by it's name\n * @param {string} hierarchicalFacetName\n * @return {object} a hierarchicalFacet\n */\n getHierarchicalFacetByName: function(hierarchicalFacetName) {\n return find(\n this.hierarchicalFacets,\n function(f) {\n return f.name === hierarchicalFacetName;\n }\n );\n },\n\n /**\n * Get the current breadcrumb for a hierarchical facet, as an array\n * @param {string} facetName Hierarchical facet name\n * @return {array.} the path as an array of string\n */\n getHierarchicalFacetBreadcrumb: function(facetName) {\n if (!this.isHierarchicalFacet(facetName)) {\n return [];\n }\n\n var refinement = this.getHierarchicalRefinement(facetName)[0];\n if (!refinement) return [];\n\n var separator = this._getHierarchicalFacetSeparator(\n this.getHierarchicalFacetByName(facetName)\n );\n var path = refinement.split(separator);\n return path.map(function(part) {\n return part.trim();\n });\n },\n\n toString: function() {\n return JSON.stringify(this, null, 2);\n }\n};\n\n/**\n * Callback used for clearRefinement method\n * @callback SearchParameters.clearCallback\n * @param {OperatorList|FacetList} value the value of the filter\n * @param {string} key the current attribute name\n * @param {string} type `numeric`, `disjunctiveFacet`, `conjunctiveFacet`, `hierarchicalFacet` or `exclude`\n * depending on the type of facet\n * @return {boolean} `true` if the element should be removed. `false` otherwise.\n */\nmodule.exports = SearchParameters;\n","'use strict';\n\nfunction compareAscending(value, other) {\n if (value !== other) {\n var valIsDefined = value !== undefined;\n var valIsNull = value === null;\n\n var othIsDefined = other !== undefined;\n var othIsNull = other === null;\n\n if (\n (!othIsNull && value > other) ||\n (valIsNull && othIsDefined) ||\n !valIsDefined\n ) {\n return 1;\n }\n if (\n (!valIsNull && value < other) ||\n (othIsNull && valIsDefined) ||\n !othIsDefined\n ) {\n return -1;\n }\n }\n return 0;\n}\n\n/**\n * @param {Array} collection object with keys in attributes\n * @param {Array} iteratees attributes\n * @param {Array} orders asc | desc\n */\nfunction orderBy(collection, iteratees, orders) {\n if (!Array.isArray(collection)) {\n return [];\n }\n\n if (!Array.isArray(orders)) {\n orders = [];\n }\n\n var result = collection.map(function(value, index) {\n return {\n criteria: iteratees.map(function(iteratee) {\n return value[iteratee];\n }),\n index: index,\n value: value\n };\n });\n\n result.sort(function comparer(object, other) {\n var index = -1;\n\n while (++index < object.criteria.length) {\n var res = compareAscending(object.criteria[index], other.criteria[index]);\n if (res) {\n if (index >= orders.length) {\n return res;\n }\n if (orders[index] === 'desc') {\n return -res;\n }\n return res;\n }\n }\n\n // This ensures a stable sort in V8 and other engines.\n // See https://bugs.chromium.org/p/v8/issues/detail?id=90 for more details.\n return object.index - other.index;\n });\n\n return result.map(function(res) {\n return res.value;\n });\n}\n\nmodule.exports = orderBy;\n","'use strict';\n\nmodule.exports = function compact(array) {\n if (!Array.isArray(array)) {\n return [];\n }\n\n return array.filter(Boolean);\n};\n","'use strict';\n\n// @MAJOR can be replaced by native Array#findIndex when we change support\nmodule.exports = function find(array, comparator) {\n if (!Array.isArray(array)) {\n return -1;\n }\n\n for (var i = 0; i < array.length; i++) {\n if (comparator(array[i])) {\n return i;\n }\n }\n return -1;\n};\n","'use strict';\n\nvar find = require('./find');\n\n/**\n * Transform sort format from user friendly notation to lodash format\n * @param {string[]} sortBy array of predicate of the form \"attribute:order\"\n * @param {string[]} [defaults] array of predicate of the form \"attribute:order\"\n * @return {array.} array containing 2 elements : attributes, orders\n */\nmodule.exports = function formatSort(sortBy, defaults) {\n var defaultInstructions = (defaults || []).map(function(sort) {\n return sort.split(':');\n });\n\n return sortBy.reduce(\n function preparePredicate(out, sort) {\n var sortInstruction = sort.split(':');\n\n var matchingDefault = find(defaultInstructions, function(\n defaultInstruction\n ) {\n return defaultInstruction[0] === sortInstruction[0];\n });\n\n if (sortInstruction.length > 1 || !matchingDefault) {\n out[0].push(sortInstruction[0]);\n out[1].push(sortInstruction[1]);\n return out;\n }\n\n out[0].push(matchingDefault[0]);\n out[1].push(matchingDefault[1]);\n return out;\n },\n [[], []]\n );\n};\n","'use strict';\n\n/**\n * Replaces a leading - with \\-\n * @private\n * @param {any} value the facet value to replace\n * @returns any\n */\nfunction escapeFacetValue(value) {\n if (typeof value !== 'string') return value;\n\n return String(value).replace(/^-/, '\\\\-');\n}\n\n/**\n * Replaces a leading \\- with -\n * @private\n * @param {any} value the escaped facet value\n * @returns any\n */\nfunction unescapeFacetValue(value) {\n if (typeof value !== 'string') return value;\n\n return value.replace(/^\\\\-/, '-');\n}\n\nmodule.exports = {\n escapeFacetValue: escapeFacetValue,\n unescapeFacetValue: unescapeFacetValue\n};\n","'use strict';\n\nmodule.exports = generateTrees;\n\nvar orderBy = require('../functions/orderBy');\nvar find = require('../functions/find');\nvar prepareHierarchicalFacetSortBy = require('../functions/formatSort');\nvar fv = require('../functions/escapeFacetValue');\nvar escapeFacetValue = fv.escapeFacetValue;\nvar unescapeFacetValue = fv.unescapeFacetValue;\n\nfunction generateTrees(state) {\n return function generate(hierarchicalFacetResult, hierarchicalFacetIndex) {\n var hierarchicalFacet = state.hierarchicalFacets[hierarchicalFacetIndex];\n var hierarchicalFacetRefinement =\n (state.hierarchicalFacetsRefinements[hierarchicalFacet.name] &&\n state.hierarchicalFacetsRefinements[hierarchicalFacet.name][0]) ||\n '';\n var hierarchicalSeparator = state._getHierarchicalFacetSeparator(\n hierarchicalFacet\n );\n var hierarchicalRootPath = state._getHierarchicalRootPath(\n hierarchicalFacet\n );\n var hierarchicalShowParentLevel = state._getHierarchicalShowParentLevel(\n hierarchicalFacet\n );\n var sortBy = prepareHierarchicalFacetSortBy(\n state._getHierarchicalFacetSortBy(hierarchicalFacet)\n );\n\n var rootExhaustive = hierarchicalFacetResult.every(function(facetResult) {\n return facetResult.exhaustive;\n });\n\n var generateTreeFn = generateHierarchicalTree(\n sortBy,\n hierarchicalSeparator,\n hierarchicalRootPath,\n hierarchicalShowParentLevel,\n hierarchicalFacetRefinement\n );\n\n var results = hierarchicalFacetResult;\n\n if (hierarchicalRootPath) {\n results = hierarchicalFacetResult.slice(\n hierarchicalRootPath.split(hierarchicalSeparator).length\n );\n }\n\n return results.reduce(generateTreeFn, {\n name: state.hierarchicalFacets[hierarchicalFacetIndex].name,\n count: null, // root level, no count\n isRefined: true, // root level, always refined\n path: null, // root level, no path\n escapedValue: null,\n exhaustive: rootExhaustive,\n data: null\n });\n };\n}\n\nfunction generateHierarchicalTree(\n sortBy,\n hierarchicalSeparator,\n hierarchicalRootPath,\n hierarchicalShowParentLevel,\n currentRefinement\n) {\n return function generateTree(\n hierarchicalTree,\n hierarchicalFacetResult,\n currentHierarchicalLevel\n ) {\n var parent = hierarchicalTree;\n\n if (currentHierarchicalLevel > 0) {\n var level = 0;\n\n parent = hierarchicalTree;\n\n while (level < currentHierarchicalLevel) {\n /**\n * @type {object[]]} hierarchical data\n */\n var data = parent && Array.isArray(parent.data) ? parent.data : [];\n parent = find(data, function(subtree) {\n return subtree.isRefined;\n });\n level++;\n }\n }\n\n // we found a refined parent, let's add current level data under it\n if (parent) {\n // filter values in case an object has multiple categories:\n // {\n // categories: {\n // level0: ['beers', 'bières'],\n // level1: ['beers > IPA', 'bières > Belges']\n // }\n // }\n //\n // If parent refinement is `beers`, then we do not want to have `bières > Belges`\n // showing up\n\n var picked = Object.keys(hierarchicalFacetResult.data)\n .map(function(facetValue) {\n return [facetValue, hierarchicalFacetResult.data[facetValue]];\n })\n .filter(function(tuple) {\n var facetValue = tuple[0];\n return onlyMatchingTree(\n facetValue,\n parent.path || hierarchicalRootPath,\n currentRefinement,\n hierarchicalSeparator,\n hierarchicalRootPath,\n hierarchicalShowParentLevel\n );\n });\n\n parent.data = orderBy(\n picked.map(function(tuple) {\n var facetValue = tuple[0];\n var facetCount = tuple[1];\n\n return format(\n facetCount,\n facetValue,\n hierarchicalSeparator,\n unescapeFacetValue(currentRefinement),\n hierarchicalFacetResult.exhaustive\n );\n }),\n sortBy[0],\n sortBy[1]\n );\n }\n\n return hierarchicalTree;\n };\n}\n\nfunction onlyMatchingTree(\n facetValue,\n parentPath,\n currentRefinement,\n hierarchicalSeparator,\n hierarchicalRootPath,\n hierarchicalShowParentLevel\n) {\n // we want the facetValue is a child of hierarchicalRootPath\n if (\n hierarchicalRootPath &&\n (facetValue.indexOf(hierarchicalRootPath) !== 0 ||\n hierarchicalRootPath === facetValue)\n ) {\n return false;\n }\n\n // we always want root levels (only when there is no prefix path)\n return (\n (!hierarchicalRootPath &&\n facetValue.indexOf(hierarchicalSeparator) === -1) ||\n // if there is a rootPath, being root level mean 1 level under rootPath\n (hierarchicalRootPath &&\n facetValue.split(hierarchicalSeparator).length -\n hierarchicalRootPath.split(hierarchicalSeparator).length ===\n 1) ||\n // if current refinement is a root level and current facetValue is a root level,\n // keep the facetValue\n (facetValue.indexOf(hierarchicalSeparator) === -1 &&\n currentRefinement.indexOf(hierarchicalSeparator) === -1) ||\n // currentRefinement is a child of the facet value\n currentRefinement.indexOf(facetValue) === 0 ||\n // facetValue is a child of the current parent, add it\n (facetValue.indexOf(parentPath + hierarchicalSeparator) === 0 &&\n (hierarchicalShowParentLevel ||\n facetValue.indexOf(currentRefinement) === 0))\n );\n}\n\nfunction format(\n facetCount,\n facetValue,\n hierarchicalSeparator,\n currentRefinement,\n exhaustive\n) {\n var parts = facetValue.split(hierarchicalSeparator);\n return {\n name: parts[parts.length - 1].trim(),\n path: facetValue,\n escapedValue: escapeFacetValue(facetValue),\n count: facetCount,\n isRefined:\n currentRefinement === facetValue ||\n currentRefinement.indexOf(facetValue + hierarchicalSeparator) === 0,\n exhaustive: exhaustive,\n data: null\n };\n}\n","'use strict';\n\nvar merge = require('../functions/merge');\nvar defaultsPure = require('../functions/defaultsPure');\nvar orderBy = require('../functions/orderBy');\nvar compact = require('../functions/compact');\nvar find = require('../functions/find');\nvar findIndex = require('../functions/findIndex');\nvar formatSort = require('../functions/formatSort');\nvar fv = require('../functions/escapeFacetValue');\nvar escapeFacetValue = fv.escapeFacetValue;\nvar unescapeFacetValue = fv.unescapeFacetValue;\n\nvar generateHierarchicalTree = require('./generate-hierarchical-tree');\n\n/**\n * @typedef SearchResults.Facet\n * @type {object}\n * @property {string} name name of the attribute in the record\n * @property {object} data the faceting data: value, number of entries\n * @property {object} stats undefined unless facet_stats is retrieved from algolia\n */\n\n/**\n * @typedef SearchResults.HierarchicalFacet\n * @type {object}\n * @property {string} name name of the current value given the hierarchical level, trimmed.\n * If root node, you get the facet name\n * @property {number} count number of objects matching this hierarchical value\n * @property {string} path the current hierarchical value full path\n * @property {boolean} isRefined `true` if the current value was refined, `false` otherwise\n * @property {HierarchicalFacet[]} data sub values for the current level\n */\n\n/**\n * @typedef SearchResults.FacetValue\n * @type {object}\n * @property {string} name the facet value itself\n * @property {number} count times this facet appears in the results\n * @property {boolean} isRefined is the facet currently selected\n * @property {boolean} isExcluded is the facet currently excluded (only for conjunctive facets)\n */\n\n/**\n * @typedef Refinement\n * @type {object}\n * @property {string} type the type of filter used:\n * `numeric`, `facet`, `exclude`, `disjunctive`, `hierarchical`\n * @property {string} attributeName name of the attribute used for filtering\n * @property {string} name the value of the filter\n * @property {number} numericValue the value as a number. Only for numeric filters.\n * @property {string} operator the operator used. Only for numeric filters.\n * @property {number} count the number of computed hits for this filter. Only on facets.\n * @property {boolean} exhaustive if the count is exhaustive\n */\n\n/**\n * @param {string[]} attributes\n */\nfunction getIndices(attributes) {\n var indices = {};\n\n attributes.forEach(function(val, idx) {\n indices[val] = idx;\n });\n\n return indices;\n}\n\nfunction assignFacetStats(dest, facetStats, key) {\n if (facetStats && facetStats[key]) {\n dest.stats = facetStats[key];\n }\n}\n\n/**\n * @typedef {Object} HierarchicalFacet\n * @property {string} name\n * @property {string[]} attributes\n */\n\n/**\n * @param {HierarchicalFacet[]} hierarchicalFacets\n * @param {string} hierarchicalAttributeName\n */\nfunction findMatchingHierarchicalFacetFromAttributeName(\n hierarchicalFacets,\n hierarchicalAttributeName\n) {\n return find(hierarchicalFacets, function facetKeyMatchesAttribute(\n hierarchicalFacet\n ) {\n var facetNames = hierarchicalFacet.attributes || [];\n return facetNames.indexOf(hierarchicalAttributeName) > -1;\n });\n}\n\n/*eslint-disable */\n/**\n * Constructor for SearchResults\n * @class\n * @classdesc SearchResults contains the results of a query to Algolia using the\n * {@link AlgoliaSearchHelper}.\n * @param {SearchParameters} state state that led to the response\n * @param {array.} results the results from algolia client\n * @example SearchResults of the first query in\n * the instant search demo\n{\n \"hitsPerPage\": 10,\n \"processingTimeMS\": 2,\n \"facets\": [\n {\n \"name\": \"type\",\n \"data\": {\n \"HardGood\": 6627,\n \"BlackTie\": 550,\n \"Music\": 665,\n \"Software\": 131,\n \"Game\": 456,\n \"Movie\": 1571\n },\n \"exhaustive\": false\n },\n {\n \"exhaustive\": false,\n \"data\": {\n \"Free shipping\": 5507\n },\n \"name\": \"shipping\"\n }\n ],\n \"hits\": [\n {\n \"thumbnailImage\": \"http://img.bbystatic.com/BestBuy_US/images/products/1688/1688832_54x108_s.gif\",\n \"_highlightResult\": {\n \"shortDescription\": {\n \"matchLevel\": \"none\",\n \"value\": \"Safeguard your PC, Mac, Android and iOS devices with comprehensive Internet protection\",\n \"matchedWords\": []\n },\n \"category\": {\n \"matchLevel\": \"none\",\n \"value\": \"Computer Security Software\",\n \"matchedWords\": []\n },\n \"manufacturer\": {\n \"matchedWords\": [],\n \"value\": \"Webroot\",\n \"matchLevel\": \"none\"\n },\n \"name\": {\n \"value\": \"Webroot SecureAnywhere Internet Security (3-Device) (1-Year Subscription) - Mac/Windows\",\n \"matchedWords\": [],\n \"matchLevel\": \"none\"\n }\n },\n \"image\": \"http://img.bbystatic.com/BestBuy_US/images/products/1688/1688832_105x210_sc.jpg\",\n \"shipping\": \"Free shipping\",\n \"bestSellingRank\": 4,\n \"shortDescription\": \"Safeguard your PC, Mac, Android and iOS devices with comprehensive Internet protection\",\n \"url\": \"http://www.bestbuy.com/site/webroot-secureanywhere-internet-security-3-devi…d=1219060687969&skuId=1688832&cmp=RMX&ky=2d3GfEmNIzjA0vkzveHdZEBgpPCyMnLTJ\",\n \"name\": \"Webroot SecureAnywhere Internet Security (3-Device) (1-Year Subscription) - Mac/Windows\",\n \"category\": \"Computer Security Software\",\n \"salePrice_range\": \"1 - 50\",\n \"objectID\": \"1688832\",\n \"type\": \"Software\",\n \"customerReviewCount\": 5980,\n \"salePrice\": 49.99,\n \"manufacturer\": \"Webroot\"\n },\n ....\n ],\n \"nbHits\": 10000,\n \"disjunctiveFacets\": [\n {\n \"exhaustive\": false,\n \"data\": {\n \"5\": 183,\n \"12\": 112,\n \"7\": 149,\n ...\n },\n \"name\": \"customerReviewCount\",\n \"stats\": {\n \"max\": 7461,\n \"avg\": 157.939,\n \"min\": 1\n }\n },\n {\n \"data\": {\n \"Printer Ink\": 142,\n \"Wireless Speakers\": 60,\n \"Point & Shoot Cameras\": 48,\n ...\n },\n \"name\": \"category\",\n \"exhaustive\": false\n },\n {\n \"exhaustive\": false,\n \"data\": {\n \"> 5000\": 2,\n \"1 - 50\": 6524,\n \"501 - 2000\": 566,\n \"201 - 500\": 1501,\n \"101 - 200\": 1360,\n \"2001 - 5000\": 47\n },\n \"name\": \"salePrice_range\"\n },\n {\n \"data\": {\n \"Dynex™\": 202,\n \"Insignia™\": 230,\n \"PNY\": 72,\n ...\n },\n \"name\": \"manufacturer\",\n \"exhaustive\": false\n }\n ],\n \"query\": \"\",\n \"nbPages\": 100,\n \"page\": 0,\n \"index\": \"bestbuy\"\n}\n **/\n/*eslint-enable */\nfunction SearchResults(state, results, options) {\n var mainSubResponse = results[0];\n\n this._rawResults = results;\n\n var self = this;\n\n // https://www.algolia.com/doc/api-reference/api-methods/search/#response\n Object.keys(mainSubResponse).forEach(function(key) {\n self[key] = mainSubResponse[key];\n });\n\n // Make every key of the result options reachable from the instance\n Object.keys(options || {}).forEach(function(key) {\n self[key] = options[key];\n });\n\n /**\n * query used to generate the results\n * @name query\n * @member {string}\n * @memberof SearchResults\n * @instance\n */\n /**\n * The query as parsed by the engine given all the rules.\n * @name parsedQuery\n * @member {string}\n * @memberof SearchResults\n * @instance\n */\n /**\n * all the records that match the search parameters. Each record is\n * augmented with a new attribute `_highlightResult`\n * which is an object keyed by attribute and with the following properties:\n * - `value` : the value of the facet highlighted (html)\n * - `matchLevel`: full, partial or none depending on how the query terms match\n * @name hits\n * @member {object[]}\n * @memberof SearchResults\n * @instance\n */\n /**\n * index where the results come from\n * @name index\n * @member {string}\n * @memberof SearchResults\n * @instance\n */\n /**\n * number of hits per page requested\n * @name hitsPerPage\n * @member {number}\n * @memberof SearchResults\n * @instance\n */\n /**\n * total number of hits of this query on the index\n * @name nbHits\n * @member {number}\n * @memberof SearchResults\n * @instance\n */\n /**\n * total number of pages with respect to the number of hits per page and the total number of hits\n * @name nbPages\n * @member {number}\n * @memberof SearchResults\n * @instance\n */\n /**\n * current page\n * @name page\n * @member {number}\n * @memberof SearchResults\n * @instance\n */\n /**\n * The position if the position was guessed by IP.\n * @name aroundLatLng\n * @member {string}\n * @memberof SearchResults\n * @instance\n * @example \"48.8637,2.3615\",\n */\n /**\n * The radius computed by Algolia.\n * @name automaticRadius\n * @member {string}\n * @memberof SearchResults\n * @instance\n * @example \"126792922\",\n */\n /**\n * String identifying the server used to serve this request.\n *\n * getRankingInfo needs to be set to `true` for this to be returned\n *\n * @name serverUsed\n * @member {string}\n * @memberof SearchResults\n * @instance\n * @example \"c7-use-2.algolia.net\",\n */\n /**\n * Boolean that indicates if the computation of the counts did time out.\n * @deprecated\n * @name timeoutCounts\n * @member {boolean}\n * @memberof SearchResults\n * @instance\n */\n /**\n * Boolean that indicates if the computation of the hits did time out.\n * @deprecated\n * @name timeoutHits\n * @member {boolean}\n * @memberof SearchResults\n * @instance\n */\n /**\n * True if the counts of the facets is exhaustive\n * @name exhaustiveFacetsCount\n * @member {boolean}\n * @memberof SearchResults\n * @instance\n */\n /**\n * True if the number of hits is exhaustive\n * @name exhaustiveNbHits\n * @member {boolean}\n * @memberof SearchResults\n * @instance\n */\n /**\n * Contains the userData if they are set by a [query rule](https://www.algolia.com/doc/guides/query-rules/query-rules-overview/).\n * @name userData\n * @member {object[]}\n * @memberof SearchResults\n * @instance\n */\n /**\n * queryID is the unique identifier of the query used to generate the current search results.\n * This value is only available if the `clickAnalytics` search parameter is set to `true`.\n * @name queryID\n * @member {string}\n * @memberof SearchResults\n * @instance\n */\n\n /**\n * sum of the processing time of all the queries\n * @member {number}\n */\n this.processingTimeMS = results.reduce(function(sum, result) {\n return result.processingTimeMS === undefined\n ? sum\n : sum + result.processingTimeMS;\n }, 0);\n\n /**\n * disjunctive facets results\n * @member {SearchResults.Facet[]}\n */\n this.disjunctiveFacets = [];\n /**\n * disjunctive facets results\n * @member {SearchResults.HierarchicalFacet[]}\n */\n this.hierarchicalFacets = state.hierarchicalFacets.map(function initFutureTree() {\n return [];\n });\n /**\n * other facets results\n * @member {SearchResults.Facet[]}\n */\n this.facets = [];\n\n var disjunctiveFacets = state.getRefinedDisjunctiveFacets();\n\n var facetsIndices = getIndices(state.facets);\n var disjunctiveFacetsIndices = getIndices(state.disjunctiveFacets);\n var nextDisjunctiveResult = 1;\n\n // Since we send request only for disjunctive facets that have been refined,\n // we get the facets information from the first, general, response.\n\n var mainFacets = mainSubResponse.facets || {};\n\n Object.keys(mainFacets).forEach(function(facetKey) {\n var facetValueObject = mainFacets[facetKey];\n\n var hierarchicalFacet = findMatchingHierarchicalFacetFromAttributeName(\n state.hierarchicalFacets,\n facetKey\n );\n\n if (hierarchicalFacet) {\n // Place the hierarchicalFacet data at the correct index depending on\n // the attributes order that was defined at the helper initialization\n var facetIndex = hierarchicalFacet.attributes.indexOf(facetKey);\n var idxAttributeName = findIndex(state.hierarchicalFacets, function(f) {\n return f.name === hierarchicalFacet.name;\n });\n self.hierarchicalFacets[idxAttributeName][facetIndex] = {\n attribute: facetKey,\n data: facetValueObject,\n exhaustive: mainSubResponse.exhaustiveFacetsCount\n };\n } else {\n var isFacetDisjunctive = state.disjunctiveFacets.indexOf(facetKey) !== -1;\n var isFacetConjunctive = state.facets.indexOf(facetKey) !== -1;\n var position;\n\n if (isFacetDisjunctive) {\n position = disjunctiveFacetsIndices[facetKey];\n self.disjunctiveFacets[position] = {\n name: facetKey,\n data: facetValueObject,\n exhaustive: mainSubResponse.exhaustiveFacetsCount\n };\n assignFacetStats(self.disjunctiveFacets[position], mainSubResponse.facets_stats, facetKey);\n }\n if (isFacetConjunctive) {\n position = facetsIndices[facetKey];\n self.facets[position] = {\n name: facetKey,\n data: facetValueObject,\n exhaustive: mainSubResponse.exhaustiveFacetsCount\n };\n assignFacetStats(self.facets[position], mainSubResponse.facets_stats, facetKey);\n }\n }\n });\n\n // Make sure we do not keep holes within the hierarchical facets\n this.hierarchicalFacets = compact(this.hierarchicalFacets);\n\n // aggregate the refined disjunctive facets\n disjunctiveFacets.forEach(function(disjunctiveFacet) {\n var result = results[nextDisjunctiveResult];\n var facets = result && result.facets ? result.facets : {};\n var hierarchicalFacet = state.getHierarchicalFacetByName(disjunctiveFacet);\n\n // There should be only item in facets.\n Object.keys(facets).forEach(function(dfacet) {\n var facetResults = facets[dfacet];\n\n var position;\n\n if (hierarchicalFacet) {\n position = findIndex(state.hierarchicalFacets, function(f) {\n return f.name === hierarchicalFacet.name;\n });\n var attributeIndex = findIndex(self.hierarchicalFacets[position], function(f) {\n return f.attribute === dfacet;\n });\n\n // previous refinements and no results so not able to find it\n if (attributeIndex === -1) {\n return;\n }\n\n self.hierarchicalFacets[position][attributeIndex].data = merge(\n {},\n self.hierarchicalFacets[position][attributeIndex].data,\n facetResults\n );\n } else {\n position = disjunctiveFacetsIndices[dfacet];\n\n var dataFromMainRequest = mainSubResponse.facets && mainSubResponse.facets[dfacet] || {};\n\n self.disjunctiveFacets[position] = {\n name: dfacet,\n data: defaultsPure({}, facetResults, dataFromMainRequest),\n exhaustive: result.exhaustiveFacetsCount\n };\n assignFacetStats(self.disjunctiveFacets[position], result.facets_stats, dfacet);\n\n if (state.disjunctiveFacetsRefinements[dfacet]) {\n state.disjunctiveFacetsRefinements[dfacet].forEach(function(refinementValue) {\n // add the disjunctive refinements if it is no more retrieved\n if (!self.disjunctiveFacets[position].data[refinementValue] &&\n state.disjunctiveFacetsRefinements[dfacet].indexOf(unescapeFacetValue(refinementValue)) > -1) {\n self.disjunctiveFacets[position].data[refinementValue] = 0;\n }\n });\n }\n }\n });\n nextDisjunctiveResult++;\n });\n\n // if we have some root level values for hierarchical facets, merge them\n state.getRefinedHierarchicalFacets().forEach(function(refinedFacet) {\n var hierarchicalFacet = state.getHierarchicalFacetByName(refinedFacet);\n var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet);\n\n var currentRefinement = state.getHierarchicalRefinement(refinedFacet);\n // if we are already at a root refinement (or no refinement at all), there is no\n // root level values request\n if (currentRefinement.length === 0 || currentRefinement[0].split(separator).length < 2) {\n return;\n }\n\n var result = results[nextDisjunctiveResult];\n var facets = result && result.facets\n ? result.facets\n : {};\n Object.keys(facets).forEach(function(dfacet) {\n var facetResults = facets[dfacet];\n var position = findIndex(state.hierarchicalFacets, function(f) {\n return f.name === hierarchicalFacet.name;\n });\n var attributeIndex = findIndex(self.hierarchicalFacets[position], function(f) {\n return f.attribute === dfacet;\n });\n\n // previous refinements and no results so not able to find it\n if (attributeIndex === -1) {\n return;\n }\n\n // when we always get root levels, if the hits refinement is `beers > IPA` (count: 5),\n // then the disjunctive values will be `beers` (count: 100),\n // but we do not want to display\n // | beers (100)\n // > IPA (5)\n // We want\n // | beers (5)\n // > IPA (5)\n var defaultData = {};\n\n if (currentRefinement.length > 0) {\n var root = currentRefinement[0].split(separator)[0];\n defaultData[root] = self.hierarchicalFacets[position][attributeIndex].data[root];\n }\n\n self.hierarchicalFacets[position][attributeIndex].data = defaultsPure(\n defaultData,\n facetResults,\n self.hierarchicalFacets[position][attributeIndex].data\n );\n });\n\n nextDisjunctiveResult++;\n });\n\n // add the excludes\n Object.keys(state.facetsExcludes).forEach(function(facetName) {\n var excludes = state.facetsExcludes[facetName];\n var position = facetsIndices[facetName];\n\n self.facets[position] = {\n name: facetName,\n data: mainSubResponse.facets[facetName],\n exhaustive: mainSubResponse.exhaustiveFacetsCount\n };\n excludes.forEach(function(facetValue) {\n self.facets[position] = self.facets[position] || {name: facetName};\n self.facets[position].data = self.facets[position].data || {};\n self.facets[position].data[facetValue] = 0;\n });\n });\n\n /**\n * @type {Array}\n */\n this.hierarchicalFacets = this.hierarchicalFacets.map(generateHierarchicalTree(state));\n\n /**\n * @type {Array}\n */\n this.facets = compact(this.facets);\n /**\n * @type {Array}\n */\n this.disjunctiveFacets = compact(this.disjunctiveFacets);\n\n this._state = state;\n}\n\n/**\n * Get a facet object with its name\n * @deprecated\n * @param {string} name name of the faceted attribute\n * @return {SearchResults.Facet} the facet object\n */\nSearchResults.prototype.getFacetByName = function(name) {\n function predicate(facet) {\n return facet.name === name;\n }\n\n return find(this.facets, predicate) ||\n find(this.disjunctiveFacets, predicate) ||\n find(this.hierarchicalFacets, predicate);\n};\n\n/**\n * Get the facet values of a specified attribute from a SearchResults object.\n * @private\n * @param {SearchResults} results the search results to search in\n * @param {string} attribute name of the faceted attribute to search for\n * @return {array|object} facet values. For the hierarchical facets it is an object.\n */\nfunction extractNormalizedFacetValues(results, attribute) {\n function predicate(facet) {\n return facet.name === attribute;\n }\n\n if (results._state.isConjunctiveFacet(attribute)) {\n var facet = find(results.facets, predicate);\n if (!facet) return [];\n\n return Object.keys(facet.data).map(function(name) {\n var value = escapeFacetValue(name);\n return {\n name: name,\n escapedValue: value,\n count: facet.data[name],\n isRefined: results._state.isFacetRefined(attribute, value),\n isExcluded: results._state.isExcludeRefined(attribute, name)\n };\n });\n } else if (results._state.isDisjunctiveFacet(attribute)) {\n var disjunctiveFacet = find(results.disjunctiveFacets, predicate);\n if (!disjunctiveFacet) return [];\n\n return Object.keys(disjunctiveFacet.data).map(function(name) {\n var value = escapeFacetValue(name);\n return {\n name: name,\n escapedValue: value,\n count: disjunctiveFacet.data[name],\n isRefined: results._state.isDisjunctiveFacetRefined(attribute, value)\n };\n });\n } else if (results._state.isHierarchicalFacet(attribute)) {\n return find(results.hierarchicalFacets, predicate);\n }\n}\n\n/**\n * Sort nodes of a hierarchical or disjunctive facet results\n * @private\n * @param {function} sortFn\n * @param {HierarchicalFacet|Array} node node upon which we want to apply the sort\n * @param {string[]} names attribute names\n * @param {number} [level=0] current index in the names array\n */\nfunction recSort(sortFn, node, names, level) {\n level = level || 0;\n\n if (Array.isArray(node)) {\n return sortFn(node, names[level]);\n }\n\n if (!node.data || node.data.length === 0) {\n return node;\n }\n\n var children = node.data.map(function(childNode) {\n return recSort(sortFn, childNode, names, level + 1);\n });\n var sortedChildren = sortFn(children, names[level]);\n var newNode = defaultsPure({data: sortedChildren}, node);\n return newNode;\n}\n\nSearchResults.DEFAULT_SORT = ['isRefined:desc', 'count:desc', 'name:asc'];\n\nfunction vanillaSortFn(order, data) {\n return data.sort(order);\n}\n\n/**\n * @typedef FacetOrdering\n * @type {Object}\n * @property {string[]} [order]\n * @property {'count' | 'alpha' | 'hidden'} [sortRemainingBy]\n */\n\n/**\n * Sorts facet arrays via their facet ordering\n * @param {Array} facetValues the values\n * @param {FacetOrdering} facetOrdering the ordering\n * @returns {Array}\n */\nfunction sortViaFacetOrdering(facetValues, facetOrdering) {\n var orderedFacets = [];\n var remainingFacets = [];\n\n var order = facetOrdering.order || [];\n /**\n * an object with the keys being the values in order, the values their index:\n * ['one', 'two'] -> { one: 0, two: 1 }\n */\n var reverseOrder = order.reduce(function(acc, name, i) {\n acc[name] = i;\n return acc;\n }, {});\n\n facetValues.forEach(function(item) {\n // hierarchical facets get sorted using their raw name\n var name = item.path || item.name;\n if (reverseOrder[name] !== undefined) {\n orderedFacets[reverseOrder[name]] = item;\n } else {\n remainingFacets.push(item);\n }\n });\n\n orderedFacets = orderedFacets.filter(function(facet) {\n return facet;\n });\n\n var sortRemainingBy = facetOrdering.sortRemainingBy;\n var ordering;\n if (sortRemainingBy === 'hidden') {\n return orderedFacets;\n } else if (sortRemainingBy === 'alpha') {\n ordering = [['path', 'name'], ['asc', 'asc']];\n } else {\n ordering = [['count'], ['desc']];\n }\n\n return orderedFacets.concat(\n orderBy(remainingFacets, ordering[0], ordering[1])\n );\n}\n\n/**\n * @param {SearchResults} results the search results class\n * @param {string} attribute the attribute to retrieve ordering of\n * @returns {FacetOrdering=}\n */\nfunction getFacetOrdering(results, attribute) {\n return (\n results.renderingContent &&\n results.renderingContent.facetOrdering &&\n results.renderingContent.facetOrdering.values &&\n results.renderingContent.facetOrdering.values[attribute]\n );\n}\n\n/**\n * Get a the list of values for a given facet attribute. Those values are sorted\n * refinement first, descending count (bigger value on top), and name ascending\n * (alphabetical order). The sort formula can overridden using either string based\n * predicates or a function.\n *\n * This method will return all the values returned by the Algolia engine plus all\n * the values already refined. This means that it can happen that the\n * `maxValuesPerFacet` [configuration](https://www.algolia.com/doc/rest-api/search#param-maxValuesPerFacet)\n * might not be respected if you have facet values that are already refined.\n * @param {string} attribute attribute name\n * @param {object} opts configuration options.\n * @param {boolean} [opts.facetOrdering]\n * Force the use of facetOrdering from the result if a sortBy is present. If\n * sortBy isn't present, facetOrdering will be used automatically.\n * @param {Array. | function} opts.sortBy\n * When using strings, it consists of\n * the name of the [FacetValue](#SearchResults.FacetValue) or the\n * [HierarchicalFacet](#SearchResults.HierarchicalFacet) attributes with the\n * order (`asc` or `desc`). For example to order the value by count, the\n * argument would be `['count:asc']`.\n *\n * If only the attribute name is specified, the ordering defaults to the one\n * specified in the default value for this attribute.\n *\n * When not specified, the order is\n * ascending. This parameter can also be a function which takes two facet\n * values and should return a number, 0 if equal, 1 if the first argument is\n * bigger or -1 otherwise.\n *\n * The default value for this attribute `['isRefined:desc', 'count:desc', 'name:asc']`\n * @return {FacetValue[]|HierarchicalFacet|undefined} depending on the type of facet of\n * the attribute requested (hierarchical, disjunctive or conjunctive)\n * @example\n * helper.on('result', function(event){\n * //get values ordered only by name ascending using the string predicate\n * event.results.getFacetValues('city', {sortBy: ['name:asc']});\n * //get values ordered only by count ascending using a function\n * event.results.getFacetValues('city', {\n * // this is equivalent to ['count:asc']\n * sortBy: function(a, b) {\n * if (a.count === b.count) return 0;\n * if (a.count > b.count) return 1;\n * if (b.count > a.count) return -1;\n * }\n * });\n * });\n */\nSearchResults.prototype.getFacetValues = function(attribute, opts) {\n var facetValues = extractNormalizedFacetValues(this, attribute);\n if (!facetValues) {\n return undefined;\n }\n\n var options = defaultsPure({}, opts, {\n sortBy: SearchResults.DEFAULT_SORT,\n // if no sortBy is given, attempt to sort based on facetOrdering\n // if it is given, we still allow to sort via facet ordering first\n facetOrdering: !(opts && opts.sortBy)\n });\n\n var results = this;\n var attributes;\n if (Array.isArray(facetValues)) {\n attributes = [attribute];\n } else {\n var config = results._state.getHierarchicalFacetByName(facetValues.name);\n attributes = config.attributes;\n }\n\n return recSort(function(data, facetName) {\n if (options.facetOrdering) {\n var facetOrdering = getFacetOrdering(results, facetName);\n if (Boolean(facetOrdering)) {\n return sortViaFacetOrdering(data, facetOrdering);\n }\n }\n\n if (Array.isArray(options.sortBy)) {\n var order = formatSort(options.sortBy, SearchResults.DEFAULT_SORT);\n return orderBy(data, order[0], order[1]);\n } else if (typeof options.sortBy === 'function') {\n return vanillaSortFn(options.sortBy, data);\n }\n throw new Error(\n 'options.sortBy is optional but if defined it must be ' +\n 'either an array of string (predicates) or a sorting function'\n );\n }, facetValues, attributes);\n};\n\n/**\n * Returns the facet stats if attribute is defined and the facet contains some.\n * Otherwise returns undefined.\n * @param {string} attribute name of the faceted attribute\n * @return {object} The stats of the facet\n */\nSearchResults.prototype.getFacetStats = function(attribute) {\n if (this._state.isConjunctiveFacet(attribute)) {\n return getFacetStatsIfAvailable(this.facets, attribute);\n } else if (this._state.isDisjunctiveFacet(attribute)) {\n return getFacetStatsIfAvailable(this.disjunctiveFacets, attribute);\n }\n\n return undefined;\n};\n\n/**\n * @typedef {Object} FacetListItem\n * @property {string} name\n */\n\n/**\n * @param {FacetListItem[]} facetList (has more items, but enough for here)\n * @param {string} facetName\n */\nfunction getFacetStatsIfAvailable(facetList, facetName) {\n var data = find(facetList, function(facet) {\n return facet.name === facetName;\n });\n return data && data.stats;\n}\n\n/**\n * Returns all refinements for all filters + tags. It also provides\n * additional information: count and exhaustiveness for each filter.\n *\n * See the [refinement type](#Refinement) for an exhaustive view of the available\n * data.\n *\n * Note that for a numeric refinement, results are grouped per operator, this\n * means that it will return responses for operators which are empty.\n *\n * @return {Array.} all the refinements\n */\nSearchResults.prototype.getRefinements = function() {\n var state = this._state;\n var results = this;\n var res = [];\n\n Object.keys(state.facetsRefinements).forEach(function(attributeName) {\n state.facetsRefinements[attributeName].forEach(function(name) {\n res.push(getRefinement(state, 'facet', attributeName, name, results.facets));\n });\n });\n\n Object.keys(state.facetsExcludes).forEach(function(attributeName) {\n state.facetsExcludes[attributeName].forEach(function(name) {\n res.push(getRefinement(state, 'exclude', attributeName, name, results.facets));\n });\n });\n\n Object.keys(state.disjunctiveFacetsRefinements).forEach(function(attributeName) {\n state.disjunctiveFacetsRefinements[attributeName].forEach(function(name) {\n res.push(getRefinement(state, 'disjunctive', attributeName, name, results.disjunctiveFacets));\n });\n });\n\n Object.keys(state.hierarchicalFacetsRefinements).forEach(function(attributeName) {\n state.hierarchicalFacetsRefinements[attributeName].forEach(function(name) {\n res.push(getHierarchicalRefinement(state, attributeName, name, results.hierarchicalFacets));\n });\n });\n\n\n Object.keys(state.numericRefinements).forEach(function(attributeName) {\n var operators = state.numericRefinements[attributeName];\n Object.keys(operators).forEach(function(operator) {\n operators[operator].forEach(function(value) {\n res.push({\n type: 'numeric',\n attributeName: attributeName,\n name: value,\n numericValue: value,\n operator: operator\n });\n });\n });\n });\n\n state.tagRefinements.forEach(function(name) {\n res.push({type: 'tag', attributeName: '_tags', name: name});\n });\n\n return res;\n};\n\n/**\n * @typedef {Object} Facet\n * @property {string} name\n * @property {Object} data\n * @property {boolean} exhaustive\n */\n\n/**\n * @param {*} state\n * @param {*} type\n * @param {string} attributeName\n * @param {*} name\n * @param {Facet[]} resultsFacets\n */\nfunction getRefinement(state, type, attributeName, name, resultsFacets) {\n var facet = find(resultsFacets, function(f) {\n return f.name === attributeName;\n });\n var count = facet && facet.data && facet.data[name] ? facet.data[name] : 0;\n var exhaustive = (facet && facet.exhaustive) || false;\n\n return {\n type: type,\n attributeName: attributeName,\n name: name,\n count: count,\n exhaustive: exhaustive\n };\n}\n\n/**\n * @param {*} state\n * @param {string} attributeName\n * @param {*} name\n * @param {Facet[]} resultsFacets\n */\nfunction getHierarchicalRefinement(state, attributeName, name, resultsFacets) {\n var facetDeclaration = state.getHierarchicalFacetByName(attributeName);\n var separator = state._getHierarchicalFacetSeparator(facetDeclaration);\n var split = name.split(separator);\n var rootFacet = find(resultsFacets, function(facet) {\n return facet.name === attributeName;\n });\n\n var facet = split.reduce(function(intermediateFacet, part) {\n var newFacet =\n intermediateFacet && find(intermediateFacet.data, function(f) {\n return f.name === part;\n });\n return newFacet !== undefined ? newFacet : intermediateFacet;\n }, rootFacet);\n\n var count = (facet && facet.count) || 0;\n var exhaustive = (facet && facet.exhaustive) || false;\n var path = (facet && facet.path) || '';\n\n return {\n type: 'hierarchical',\n attributeName: attributeName,\n name: path,\n count: count,\n exhaustive: exhaustive\n };\n}\n\nmodule.exports = SearchResults;\n","// Copyright Joyent, Inc. and other Node contributors.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a\n// copy of this software and associated documentation files (the\n// \"Software\"), to deal in the Software without restriction, including\n// without limitation the rights to use, copy, modify, merge, publish,\n// distribute, sublicense, and/or sell copies of the Software, and to permit\n// persons to whom the Software is furnished to do so, subject to the\n// following conditions:\n//\n// The above copyright notice and this permission notice shall be included\n// in all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS\n// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\n// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN\n// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,\n// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR\n// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE\n// USE OR OTHER DEALINGS IN THE SOFTWARE.\n\nfunction EventEmitter() {\n this._events = this._events || {};\n this._maxListeners = this._maxListeners || undefined;\n}\nmodule.exports = EventEmitter;\n\n// Backwards-compat with node 0.10.x\n// EventEmitter.EventEmitter = EventEmitter;\n\nEventEmitter.prototype._events = undefined;\nEventEmitter.prototype._maxListeners = undefined;\n\n// By default EventEmitters will print a warning if more than 10 listeners are\n// added to it. This is a useful default which helps finding memory leaks.\nEventEmitter.defaultMaxListeners = 10;\n\n// Obviously not all Emitters should be limited to 10. This function allows\n// that to be increased. Set to zero for unlimited.\nEventEmitter.prototype.setMaxListeners = function(n) {\n if (!isNumber(n) || n < 0 || isNaN(n))\n throw TypeError('n must be a positive number');\n this._maxListeners = n;\n return this;\n};\n\nEventEmitter.prototype.emit = function(type) {\n var er, handler, len, args, i, listeners;\n\n if (!this._events)\n this._events = {};\n\n // If there is no 'error' event listener then throw.\n if (type === 'error') {\n if (!this._events.error ||\n (isObject(this._events.error) && !this._events.error.length)) {\n er = arguments[1];\n if (er instanceof Error) {\n throw er; // Unhandled 'error' event\n } else {\n // At least give some kind of context to the user\n var err = new Error('Uncaught, unspecified \"error\" event. (' + er + ')');\n err.context = er;\n throw err;\n }\n }\n }\n\n handler = this._events[type];\n\n if (isUndefined(handler))\n return false;\n\n if (isFunction(handler)) {\n switch (arguments.length) {\n // fast cases\n case 1:\n handler.call(this);\n break;\n case 2:\n handler.call(this, arguments[1]);\n break;\n case 3:\n handler.call(this, arguments[1], arguments[2]);\n break;\n // slower\n default:\n args = Array.prototype.slice.call(arguments, 1);\n handler.apply(this, args);\n }\n } else if (isObject(handler)) {\n args = Array.prototype.slice.call(arguments, 1);\n listeners = handler.slice();\n len = listeners.length;\n for (i = 0; i < len; i++)\n listeners[i].apply(this, args);\n }\n\n return true;\n};\n\nEventEmitter.prototype.addListener = function(type, listener) {\n var m;\n\n if (!isFunction(listener))\n throw TypeError('listener must be a function');\n\n if (!this._events)\n this._events = {};\n\n // To avoid recursion in the case that type === \"newListener\"! Before\n // adding it to the listeners, first emit \"newListener\".\n if (this._events.newListener)\n this.emit('newListener', type,\n isFunction(listener.listener) ?\n listener.listener : listener);\n\n if (!this._events[type])\n // Optimize the case of one listener. Don't need the extra array object.\n this._events[type] = listener;\n else if (isObject(this._events[type]))\n // If we've already got an array, just append.\n this._events[type].push(listener);\n else\n // Adding the second element, need to change to array.\n this._events[type] = [this._events[type], listener];\n\n // Check for listener leak\n if (isObject(this._events[type]) && !this._events[type].warned) {\n if (!isUndefined(this._maxListeners)) {\n m = this._maxListeners;\n } else {\n m = EventEmitter.defaultMaxListeners;\n }\n\n if (m && m > 0 && this._events[type].length > m) {\n this._events[type].warned = true;\n console.error('(node) warning: possible EventEmitter memory ' +\n 'leak detected. %d listeners added. ' +\n 'Use emitter.setMaxListeners() to increase limit.',\n this._events[type].length);\n if (typeof console.trace === 'function') {\n // not supported in IE 10\n console.trace();\n }\n }\n }\n\n return this;\n};\n\nEventEmitter.prototype.on = EventEmitter.prototype.addListener;\n\nEventEmitter.prototype.once = function(type, listener) {\n if (!isFunction(listener))\n throw TypeError('listener must be a function');\n\n var fired = false;\n\n function g() {\n this.removeListener(type, g);\n\n if (!fired) {\n fired = true;\n listener.apply(this, arguments);\n }\n }\n\n g.listener = listener;\n this.on(type, g);\n\n return this;\n};\n\n// emits a 'removeListener' event iff the listener was removed\nEventEmitter.prototype.removeListener = function(type, listener) {\n var list, position, length, i;\n\n if (!isFunction(listener))\n throw TypeError('listener must be a function');\n\n if (!this._events || !this._events[type])\n return this;\n\n list = this._events[type];\n length = list.length;\n position = -1;\n\n if (list === listener ||\n (isFunction(list.listener) && list.listener === listener)) {\n delete this._events[type];\n if (this._events.removeListener)\n this.emit('removeListener', type, listener);\n\n } else if (isObject(list)) {\n for (i = length; i-- > 0;) {\n if (list[i] === listener ||\n (list[i].listener && list[i].listener === listener)) {\n position = i;\n break;\n }\n }\n\n if (position < 0)\n return this;\n\n if (list.length === 1) {\n list.length = 0;\n delete this._events[type];\n } else {\n list.splice(position, 1);\n }\n\n if (this._events.removeListener)\n this.emit('removeListener', type, listener);\n }\n\n return this;\n};\n\nEventEmitter.prototype.removeAllListeners = function(type) {\n var key, listeners;\n\n if (!this._events)\n return this;\n\n // not listening for removeListener, no need to emit\n if (!this._events.removeListener) {\n if (arguments.length === 0)\n this._events = {};\n else if (this._events[type])\n delete this._events[type];\n return this;\n }\n\n // emit removeListener for all listeners on all events\n if (arguments.length === 0) {\n for (key in this._events) {\n if (key === 'removeListener') continue;\n this.removeAllListeners(key);\n }\n this.removeAllListeners('removeListener');\n this._events = {};\n return this;\n }\n\n listeners = this._events[type];\n\n if (isFunction(listeners)) {\n this.removeListener(type, listeners);\n } else if (listeners) {\n // LIFO order\n while (listeners.length)\n this.removeListener(type, listeners[listeners.length - 1]);\n }\n delete this._events[type];\n\n return this;\n};\n\nEventEmitter.prototype.listeners = function(type) {\n var ret;\n if (!this._events || !this._events[type])\n ret = [];\n else if (isFunction(this._events[type]))\n ret = [this._events[type]];\n else\n ret = this._events[type].slice();\n return ret;\n};\n\nEventEmitter.prototype.listenerCount = function(type) {\n if (this._events) {\n var evlistener = this._events[type];\n\n if (isFunction(evlistener))\n return 1;\n else if (evlistener)\n return evlistener.length;\n }\n return 0;\n};\n\nEventEmitter.listenerCount = function(emitter, type) {\n return emitter.listenerCount(type);\n};\n\nfunction isFunction(arg) {\n return typeof arg === 'function';\n}\n\nfunction isNumber(arg) {\n return typeof arg === 'number';\n}\n\nfunction isObject(arg) {\n return typeof arg === 'object' && arg !== null;\n}\n\nfunction isUndefined(arg) {\n return arg === void 0;\n}\n","'use strict';\n\nfunction inherits(ctor, superCtor) {\n ctor.prototype = Object.create(superCtor.prototype, {\n constructor: {\n value: ctor,\n enumerable: false,\n writable: true,\n configurable: true\n }\n });\n}\n\nmodule.exports = inherits;\n","'use strict';\n\nvar EventEmitter = require('@algolia/events');\nvar inherits = require('../functions/inherits');\n\n/**\n * A DerivedHelper is a way to create sub requests to\n * Algolia from a main helper.\n * @class\n * @classdesc The DerivedHelper provides an event based interface for search callbacks:\n * - search: when a search is triggered using the `search()` method.\n * - result: when the response is retrieved from Algolia and is processed.\n * This event contains a {@link SearchResults} object and the\n * {@link SearchParameters} corresponding to this answer.\n */\nfunction DerivedHelper(mainHelper, fn) {\n this.main = mainHelper;\n this.fn = fn;\n this.lastResults = null;\n}\n\ninherits(DerivedHelper, EventEmitter);\n\n/**\n * Detach this helper from the main helper\n * @return {undefined}\n * @throws Error if the derived helper is already detached\n */\nDerivedHelper.prototype.detach = function() {\n this.removeAllListeners();\n this.main.detachDerivedHelper(this);\n};\n\nDerivedHelper.prototype.getModifiedState = function(parameters) {\n return this.fn(parameters);\n};\n\nmodule.exports = DerivedHelper;\n","'use strict';\n\nvar merge = require('./functions/merge');\n\nvar requestBuilder = {\n /**\n * Get all the queries to send to the client, those queries can used directly\n * with the Algolia client.\n * @private\n * @return {object[]} The queries\n */\n _getQueries: function getQueries(index, state) {\n var queries = [];\n\n // One query for the hits\n queries.push({\n indexName: index,\n params: requestBuilder._getHitsSearchParams(state)\n });\n\n // One for each disjunctive facets\n state.getRefinedDisjunctiveFacets().forEach(function(refinedFacet) {\n queries.push({\n indexName: index,\n params: requestBuilder._getDisjunctiveFacetSearchParams(state, refinedFacet)\n });\n });\n\n // maybe more to get the root level of hierarchical facets when activated\n state.getRefinedHierarchicalFacets().forEach(function(refinedFacet) {\n var hierarchicalFacet = state.getHierarchicalFacetByName(refinedFacet);\n\n var currentRefinement = state.getHierarchicalRefinement(refinedFacet);\n // if we are deeper than level 0 (starting from `beer > IPA`)\n // we want to get the root values\n var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet);\n if (currentRefinement.length > 0 && currentRefinement[0].split(separator).length > 1) {\n queries.push({\n indexName: index,\n params: requestBuilder._getDisjunctiveFacetSearchParams(state, refinedFacet, true)\n });\n }\n });\n\n return queries;\n },\n\n /**\n * Build search parameters used to fetch hits\n * @private\n * @return {object.}\n */\n _getHitsSearchParams: function(state) {\n var facets = state.facets\n .concat(state.disjunctiveFacets)\n .concat(requestBuilder._getHitsHierarchicalFacetsAttributes(state));\n\n\n var facetFilters = requestBuilder._getFacetFilters(state);\n var numericFilters = requestBuilder._getNumericFilters(state);\n var tagFilters = requestBuilder._getTagFilters(state);\n var additionalParams = {\n facets: facets.indexOf('*') > -1 ? ['*'] : facets,\n tagFilters: tagFilters\n };\n\n if (facetFilters.length > 0) {\n additionalParams.facetFilters = facetFilters;\n }\n\n if (numericFilters.length > 0) {\n additionalParams.numericFilters = numericFilters;\n }\n\n return merge({}, state.getQueryParams(), additionalParams);\n },\n\n /**\n * Build search parameters used to fetch a disjunctive facet\n * @private\n * @param {string} facet the associated facet name\n * @param {boolean} hierarchicalRootLevel ?? FIXME\n * @return {object}\n */\n _getDisjunctiveFacetSearchParams: function(state, facet, hierarchicalRootLevel) {\n var facetFilters = requestBuilder._getFacetFilters(state, facet, hierarchicalRootLevel);\n var numericFilters = requestBuilder._getNumericFilters(state, facet);\n var tagFilters = requestBuilder._getTagFilters(state);\n var additionalParams = {\n hitsPerPage: 1,\n page: 0,\n attributesToRetrieve: [],\n attributesToHighlight: [],\n attributesToSnippet: [],\n tagFilters: tagFilters,\n analytics: false,\n clickAnalytics: false\n };\n\n var hierarchicalFacet = state.getHierarchicalFacetByName(facet);\n\n if (hierarchicalFacet) {\n additionalParams.facets = requestBuilder._getDisjunctiveHierarchicalFacetAttribute(\n state,\n hierarchicalFacet,\n hierarchicalRootLevel\n );\n } else {\n additionalParams.facets = facet;\n }\n\n if (numericFilters.length > 0) {\n additionalParams.numericFilters = numericFilters;\n }\n\n if (facetFilters.length > 0) {\n additionalParams.facetFilters = facetFilters;\n }\n\n return merge({}, state.getQueryParams(), additionalParams);\n },\n\n /**\n * Return the numeric filters in an algolia request fashion\n * @private\n * @param {string} [facetName] the name of the attribute for which the filters should be excluded\n * @return {string[]} the numeric filters in the algolia format\n */\n _getNumericFilters: function(state, facetName) {\n if (state.numericFilters) {\n return state.numericFilters;\n }\n\n var numericFilters = [];\n\n Object.keys(state.numericRefinements).forEach(function(attribute) {\n var operators = state.numericRefinements[attribute] || {};\n Object.keys(operators).forEach(function(operator) {\n var values = operators[operator] || [];\n if (facetName !== attribute) {\n values.forEach(function(value) {\n if (Array.isArray(value)) {\n var vs = value.map(function(v) {\n return attribute + operator + v;\n });\n numericFilters.push(vs);\n } else {\n numericFilters.push(attribute + operator + value);\n }\n });\n }\n });\n });\n\n return numericFilters;\n },\n\n /**\n * Return the tags filters depending\n * @private\n * @return {string}\n */\n _getTagFilters: function(state) {\n if (state.tagFilters) {\n return state.tagFilters;\n }\n\n return state.tagRefinements.join(',');\n },\n\n\n /**\n * Build facetFilters parameter based on current refinements. The array returned\n * contains strings representing the facet filters in the algolia format.\n * @private\n * @param {string} [facet] if set, the current disjunctive facet\n * @return {array.}\n */\n _getFacetFilters: function(state, facet, hierarchicalRootLevel) {\n var facetFilters = [];\n\n var facetsRefinements = state.facetsRefinements || {};\n Object.keys(facetsRefinements).forEach(function(facetName) {\n var facetValues = facetsRefinements[facetName] || [];\n facetValues.forEach(function(facetValue) {\n facetFilters.push(facetName + ':' + facetValue);\n });\n });\n\n var facetsExcludes = state.facetsExcludes || {};\n Object.keys(facetsExcludes).forEach(function(facetName) {\n var facetValues = facetsExcludes[facetName] || [];\n facetValues.forEach(function(facetValue) {\n facetFilters.push(facetName + ':-' + facetValue);\n });\n });\n\n var disjunctiveFacetsRefinements = state.disjunctiveFacetsRefinements || {};\n Object.keys(disjunctiveFacetsRefinements).forEach(function(facetName) {\n var facetValues = disjunctiveFacetsRefinements[facetName] || [];\n if (facetName === facet || !facetValues || facetValues.length === 0) {\n return;\n }\n var orFilters = [];\n\n facetValues.forEach(function(facetValue) {\n orFilters.push(facetName + ':' + facetValue);\n });\n\n facetFilters.push(orFilters);\n });\n\n var hierarchicalFacetsRefinements = state.hierarchicalFacetsRefinements || {};\n Object.keys(hierarchicalFacetsRefinements).forEach(function(facetName) {\n var facetValues = hierarchicalFacetsRefinements[facetName] || [];\n var facetValue = facetValues[0];\n\n if (facetValue === undefined) {\n return;\n }\n\n var hierarchicalFacet = state.getHierarchicalFacetByName(facetName);\n var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet);\n var rootPath = state._getHierarchicalRootPath(hierarchicalFacet);\n var attributeToRefine;\n var attributesIndex;\n\n // we ask for parent facet values only when the `facet` is the current hierarchical facet\n if (facet === facetName) {\n // if we are at the root level already, no need to ask for facet values, we get them from\n // the hits query\n if (facetValue.indexOf(separator) === -1 || (!rootPath && hierarchicalRootLevel === true) ||\n (rootPath && rootPath.split(separator).length === facetValue.split(separator).length)) {\n return;\n }\n\n if (!rootPath) {\n attributesIndex = facetValue.split(separator).length - 2;\n facetValue = facetValue.slice(0, facetValue.lastIndexOf(separator));\n } else {\n attributesIndex = rootPath.split(separator).length - 1;\n facetValue = rootPath;\n }\n\n attributeToRefine = hierarchicalFacet.attributes[attributesIndex];\n } else {\n attributesIndex = facetValue.split(separator).length - 1;\n\n attributeToRefine = hierarchicalFacet.attributes[attributesIndex];\n }\n\n if (attributeToRefine) {\n facetFilters.push([attributeToRefine + ':' + facetValue]);\n }\n });\n\n return facetFilters;\n },\n\n _getHitsHierarchicalFacetsAttributes: function(state) {\n var out = [];\n\n return state.hierarchicalFacets.reduce(\n // ask for as much levels as there's hierarchical refinements\n function getHitsAttributesForHierarchicalFacet(allAttributes, hierarchicalFacet) {\n var hierarchicalRefinement = state.getHierarchicalRefinement(hierarchicalFacet.name)[0];\n\n // if no refinement, ask for root level\n if (!hierarchicalRefinement) {\n allAttributes.push(hierarchicalFacet.attributes[0]);\n return allAttributes;\n }\n\n var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet);\n var level = hierarchicalRefinement.split(separator).length;\n var newAttributes = hierarchicalFacet.attributes.slice(0, level + 1);\n\n return allAttributes.concat(newAttributes);\n }, out);\n },\n\n _getDisjunctiveHierarchicalFacetAttribute: function(state, hierarchicalFacet, rootLevel) {\n var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet);\n if (rootLevel === true) {\n var rootPath = state._getHierarchicalRootPath(hierarchicalFacet);\n var attributeIndex = 0;\n\n if (rootPath) {\n attributeIndex = rootPath.split(separator).length;\n }\n return [hierarchicalFacet.attributes[attributeIndex]];\n }\n\n var hierarchicalRefinement = state.getHierarchicalRefinement(hierarchicalFacet.name)[0] || '';\n // if refinement is 'beers > IPA > Flying dog',\n // then we want `facets: ['beers > IPA']` as disjunctive facet (parent level values)\n\n var parentLevel = hierarchicalRefinement.split(separator).length - 1;\n return hierarchicalFacet.attributes.slice(0, parentLevel + 1);\n },\n\n getSearchForFacetQuery: function(facetName, query, maxFacetHits, state) {\n var stateForSearchForFacetValues = state.isDisjunctiveFacet(facetName) ?\n state.clearRefinements(facetName) :\n state;\n var searchForFacetSearchParameters = {\n facetQuery: query,\n facetName: facetName\n };\n if (typeof maxFacetHits === 'number') {\n searchForFacetSearchParameters.maxFacetHits = maxFacetHits;\n }\n return merge(\n {},\n requestBuilder._getHitsSearchParams(stateForSearchForFacetValues),\n searchForFacetSearchParameters\n );\n }\n};\n\nmodule.exports = requestBuilder;\n","'use strict';\n\nmodule.exports = '3.8.2';\n","'use strict';\n\nvar SearchParameters = require('./SearchParameters');\nvar SearchResults = require('./SearchResults');\nvar DerivedHelper = require('./DerivedHelper');\nvar requestBuilder = require('./requestBuilder');\n\nvar EventEmitter = require('@algolia/events');\nvar inherits = require('./functions/inherits');\nvar objectHasKeys = require('./functions/objectHasKeys');\nvar omit = require('./functions/omit');\nvar merge = require('./functions/merge');\n\nvar version = require('./version');\nvar escapeFacetValue = require('./functions/escapeFacetValue').escapeFacetValue;\n\n/**\n * Event triggered when a parameter is set or updated\n * @event AlgoliaSearchHelper#event:change\n * @property {object} event\n * @property {SearchParameters} event.state the current parameters with the latest changes applied\n * @property {SearchResults} event.results the previous results received from Algolia. `null` before the first request\n * @example\n * helper.on('change', function(event) {\n * console.log('The parameters have changed');\n * });\n */\n\n/**\n * Event triggered when a main search is sent to Algolia\n * @event AlgoliaSearchHelper#event:search\n * @property {object} event\n * @property {SearchParameters} event.state the parameters used for this search\n * @property {SearchResults} event.results the results from the previous search. `null` if it is the first search.\n * @example\n * helper.on('search', function(event) {\n * console.log('Search sent');\n * });\n */\n\n/**\n * Event triggered when a search using `searchForFacetValues` is sent to Algolia\n * @event AlgoliaSearchHelper#event:searchForFacetValues\n * @property {object} event\n * @property {SearchParameters} event.state the parameters used for this search it is the first search.\n * @property {string} event.facet the facet searched into\n * @property {string} event.query the query used to search in the facets\n * @example\n * helper.on('searchForFacetValues', function(event) {\n * console.log('searchForFacetValues sent');\n * });\n */\n\n/**\n * Event triggered when a search using `searchOnce` is sent to Algolia\n * @event AlgoliaSearchHelper#event:searchOnce\n * @property {object} event\n * @property {SearchParameters} event.state the parameters used for this search it is the first search.\n * @example\n * helper.on('searchOnce', function(event) {\n * console.log('searchOnce sent');\n * });\n */\n\n/**\n * Event triggered when the results are retrieved from Algolia\n * @event AlgoliaSearchHelper#event:result\n * @property {object} event\n * @property {SearchResults} event.results the results received from Algolia\n * @property {SearchParameters} event.state the parameters used to query Algolia. Those might be different from the one in the helper instance (for example if the network is unreliable).\n * @example\n * helper.on('result', function(event) {\n * console.log('Search results received');\n * });\n */\n\n/**\n * Event triggered when Algolia sends back an error. For example, if an unknown parameter is\n * used, the error can be caught using this event.\n * @event AlgoliaSearchHelper#event:error\n * @property {object} event\n * @property {Error} event.error the error returned by the Algolia.\n * @example\n * helper.on('error', function(event) {\n * console.log('Houston we got a problem.');\n * });\n */\n\n/**\n * Event triggered when the queue of queries have been depleted (with any result or outdated queries)\n * @event AlgoliaSearchHelper#event:searchQueueEmpty\n * @example\n * helper.on('searchQueueEmpty', function() {\n * console.log('No more search pending');\n * // This is received before the result event if we're not expecting new results\n * });\n *\n * helper.search();\n */\n\n/**\n * Initialize a new AlgoliaSearchHelper\n * @class\n * @classdesc The AlgoliaSearchHelper is a class that ease the management of the\n * search. It provides an event based interface for search callbacks:\n * - change: when the internal search state is changed.\n * This event contains a {@link SearchParameters} object and the\n * {@link SearchResults} of the last result if any.\n * - search: when a search is triggered using the `search()` method.\n * - result: when the response is retrieved from Algolia and is processed.\n * This event contains a {@link SearchResults} object and the\n * {@link SearchParameters} corresponding to this answer.\n * - error: when the response is an error. This event contains the error returned by the server.\n * @param {AlgoliaSearch} client an AlgoliaSearch client\n * @param {string} index the index name to query\n * @param {SearchParameters | object} options an object defining the initial\n * config of the search. It doesn't have to be a {SearchParameters},\n * just an object containing the properties you need from it.\n */\nfunction AlgoliaSearchHelper(client, index, options) {\n if (typeof client.addAlgoliaAgent === 'function') {\n client.addAlgoliaAgent('JS Helper (' + version + ')');\n }\n\n this.setClient(client);\n var opts = options || {};\n opts.index = index;\n this.state = SearchParameters.make(opts);\n this.lastResults = null;\n this._queryId = 0;\n this._lastQueryIdReceived = -1;\n this.derivedHelpers = [];\n this._currentNbQueries = 0;\n}\n\ninherits(AlgoliaSearchHelper, EventEmitter);\n\n/**\n * Start the search with the parameters set in the state. When the\n * method is called, it triggers a `search` event. The results will\n * be available through the `result` event. If an error occurs, an\n * `error` will be fired instead.\n * @return {AlgoliaSearchHelper}\n * @fires search\n * @fires result\n * @fires error\n * @chainable\n */\nAlgoliaSearchHelper.prototype.search = function() {\n this._search({onlyWithDerivedHelpers: false});\n return this;\n};\n\nAlgoliaSearchHelper.prototype.searchOnlyWithDerivedHelpers = function() {\n this._search({onlyWithDerivedHelpers: true});\n return this;\n};\n\n/**\n * Gets the search query parameters that would be sent to the Algolia Client\n * for the hits\n * @return {object} Query Parameters\n */\nAlgoliaSearchHelper.prototype.getQuery = function() {\n var state = this.state;\n return requestBuilder._getHitsSearchParams(state);\n};\n\n/**\n * Start a search using a modified version of the current state. This method does\n * not trigger the helper lifecycle and does not modify the state kept internally\n * by the helper. This second aspect means that the next search call will be the\n * same as a search call before calling searchOnce.\n * @param {object} options can contain all the parameters that can be set to SearchParameters\n * plus the index\n * @param {function} [callback] optional callback executed when the response from the\n * server is back.\n * @return {promise|undefined} if a callback is passed the method returns undefined\n * otherwise it returns a promise containing an object with two keys :\n * - content with a SearchResults\n * - state with the state used for the query as a SearchParameters\n * @example\n * // Changing the number of records returned per page to 1\n * // This example uses the callback API\n * var state = helper.searchOnce({hitsPerPage: 1},\n * function(error, content, state) {\n * // if an error occurred it will be passed in error, otherwise its value is null\n * // content contains the results formatted as a SearchResults\n * // state is the instance of SearchParameters used for this search\n * });\n * @example\n * // Changing the number of records returned per page to 1\n * // This example uses the promise API\n * var state1 = helper.searchOnce({hitsPerPage: 1})\n * .then(promiseHandler);\n *\n * function promiseHandler(res) {\n * // res contains\n * // {\n * // content : SearchResults\n * // state : SearchParameters (the one used for this specific search)\n * // }\n * }\n */\nAlgoliaSearchHelper.prototype.searchOnce = function(options, cb) {\n var tempState = !options ? this.state : this.state.setQueryParameters(options);\n var queries = requestBuilder._getQueries(tempState.index, tempState);\n var self = this;\n\n this._currentNbQueries++;\n\n this.emit('searchOnce', {\n state: tempState\n });\n\n if (cb) {\n this.client\n .search(queries)\n .then(function(content) {\n self._currentNbQueries--;\n if (self._currentNbQueries === 0) {\n self.emit('searchQueueEmpty');\n }\n\n cb(null, new SearchResults(tempState, content.results), tempState);\n })\n .catch(function(err) {\n self._currentNbQueries--;\n if (self._currentNbQueries === 0) {\n self.emit('searchQueueEmpty');\n }\n\n cb(err, null, tempState);\n });\n\n return undefined;\n }\n\n return this.client.search(queries).then(function(content) {\n self._currentNbQueries--;\n if (self._currentNbQueries === 0) self.emit('searchQueueEmpty');\n return {\n content: new SearchResults(tempState, content.results),\n state: tempState,\n _originalResponse: content\n };\n }, function(e) {\n self._currentNbQueries--;\n if (self._currentNbQueries === 0) self.emit('searchQueueEmpty');\n throw e;\n });\n};\n\n /**\n * Start the search for answers with the parameters set in the state.\n * This method returns a promise.\n * @param {Object} options - the options for answers API call\n * @param {string[]} options.attributesForPrediction - Attributes to use for predictions. If empty, `searchableAttributes` is used instead.\n * @param {string[]} options.queryLanguages - The languages in the query. Currently only supports ['en'].\n * @param {number} options.nbHits - Maximum number of answers to retrieve from the Answers Engine. Cannot be greater than 1000.\n *\n * @return {promise} the answer results\n */\nAlgoliaSearchHelper.prototype.findAnswers = function(options) {\n var state = this.state;\n var derivedHelper = this.derivedHelpers[0];\n if (!derivedHelper) {\n return Promise.resolve([]);\n }\n var derivedState = derivedHelper.getModifiedState(state);\n var data = merge(\n {\n attributesForPrediction: options.attributesForPrediction,\n nbHits: options.nbHits\n },\n {\n params: omit(requestBuilder._getHitsSearchParams(derivedState), [\n 'attributesToSnippet',\n 'hitsPerPage',\n 'restrictSearchableAttributes',\n 'snippetEllipsisText' // FIXME remove this line once the engine is fixed.\n ])\n }\n );\n\n var errorMessage = 'search for answers was called, but this client does not have a function client.initIndex(index).findAnswers';\n if (typeof this.client.initIndex !== 'function') {\n throw new Error(errorMessage);\n }\n var index = this.client.initIndex(derivedState.index);\n if (typeof index.findAnswers !== 'function') {\n throw new Error(errorMessage);\n }\n return index.findAnswers(derivedState.query, options.queryLanguages, data);\n};\n\n/**\n * Structure of each result when using\n * [`searchForFacetValues()`](reference.html#AlgoliaSearchHelper#searchForFacetValues)\n * @typedef FacetSearchHit\n * @type {object}\n * @property {string} value the facet value\n * @property {string} highlighted the facet value highlighted with the query string\n * @property {number} count number of occurrence of this facet value\n * @property {boolean} isRefined true if the value is already refined\n */\n\n/**\n * Structure of the data resolved by the\n * [`searchForFacetValues()`](reference.html#AlgoliaSearchHelper#searchForFacetValues)\n * promise.\n * @typedef FacetSearchResult\n * @type {object}\n * @property {FacetSearchHit} facetHits the results for this search for facet values\n * @property {number} processingTimeMS time taken by the query inside the engine\n */\n\n/**\n * Search for facet values based on an query and the name of a faceted attribute. This\n * triggers a search and will return a promise. On top of using the query, it also sends\n * the parameters from the state so that the search is narrowed down to only the possible values.\n *\n * See the description of [FacetSearchResult](reference.html#FacetSearchResult)\n * @param {string} facet the name of the faceted attribute\n * @param {string} query the string query for the search\n * @param {number} [maxFacetHits] the maximum number values returned. Should be > 0 and <= 100\n * @param {object} [userState] the set of custom parameters to use on top of the current state. Setting a property to `undefined` removes\n * it in the generated query.\n * @return {promise.} the results of the search\n */\nAlgoliaSearchHelper.prototype.searchForFacetValues = function(facet, query, maxFacetHits, userState) {\n var clientHasSFFV = typeof this.client.searchForFacetValues === 'function';\n if (\n !clientHasSFFV &&\n typeof this.client.initIndex !== 'function'\n ) {\n throw new Error(\n 'search for facet values (searchable) was called, but this client does not have a function client.searchForFacetValues or client.initIndex(index).searchForFacetValues'\n );\n }\n var state = this.state.setQueryParameters(userState || {});\n var isDisjunctive = state.isDisjunctiveFacet(facet);\n var algoliaQuery = requestBuilder.getSearchForFacetQuery(facet, query, maxFacetHits, state);\n\n this._currentNbQueries++;\n var self = this;\n\n this.emit('searchForFacetValues', {\n state: state,\n facet: facet,\n query: query\n });\n\n var searchForFacetValuesPromise = clientHasSFFV\n ? this.client.searchForFacetValues([{indexName: state.index, params: algoliaQuery}])\n : this.client.initIndex(state.index).searchForFacetValues(algoliaQuery);\n\n return searchForFacetValuesPromise.then(function addIsRefined(content) {\n self._currentNbQueries--;\n if (self._currentNbQueries === 0) self.emit('searchQueueEmpty');\n\n content = Array.isArray(content) ? content[0] : content;\n\n content.facetHits.forEach(function(f) {\n f.escapedValue = escapeFacetValue(f.value);\n f.isRefined = isDisjunctive\n ? state.isDisjunctiveFacetRefined(facet, f.escapedValue)\n : state.isFacetRefined(facet, f.escapedValue);\n });\n\n return content;\n }, function(e) {\n self._currentNbQueries--;\n if (self._currentNbQueries === 0) self.emit('searchQueueEmpty');\n throw e;\n });\n};\n\n/**\n * Sets the text query used for the search.\n *\n * This method resets the current page to 0.\n * @param {string} q the user query\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.setQuery = function(q) {\n this._change({\n state: this.state.resetPage().setQuery(q),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Remove all the types of refinements except tags. A string can be provided to remove\n * only the refinements of a specific attribute. For more advanced use case, you can\n * provide a function instead. This function should follow the\n * [clearCallback definition](#SearchParameters.clearCallback).\n *\n * This method resets the current page to 0.\n * @param {string} [name] optional name of the facet / attribute on which we want to remove all refinements\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n * @example\n * // Removing all the refinements\n * helper.clearRefinements().search();\n * @example\n * // Removing all the filters on a the category attribute.\n * helper.clearRefinements('category').search();\n * @example\n * // Removing only the exclude filters on the category facet.\n * helper.clearRefinements(function(value, attribute, type) {\n * return type === 'exclude' && attribute === 'category';\n * }).search();\n */\nAlgoliaSearchHelper.prototype.clearRefinements = function(name) {\n this._change({\n state: this.state.resetPage().clearRefinements(name),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Remove all the tag filters.\n *\n * This method resets the current page to 0.\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.clearTags = function() {\n this._change({\n state: this.state.resetPage().clearTags(),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Adds a disjunctive filter to a faceted attribute with the `value` provided. If the\n * filter is already set, it doesn't change the filters.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} value the associated value (will be converted to string)\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.addDisjunctiveFacetRefinement = function(facet, value) {\n this._change({\n state: this.state.resetPage().addDisjunctiveFacetRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#addDisjunctiveFacetRefinement}\n */\nAlgoliaSearchHelper.prototype.addDisjunctiveRefine = function() {\n return this.addDisjunctiveFacetRefinement.apply(this, arguments);\n};\n\n/**\n * Adds a refinement on a hierarchical facet. It will throw\n * an exception if the facet is not defined or if the facet\n * is already refined.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet name\n * @param {string} path the hierarchical facet path\n * @return {AlgoliaSearchHelper}\n * @throws Error if the facet is not defined or if the facet is refined\n * @chainable\n * @fires change\n */\nAlgoliaSearchHelper.prototype.addHierarchicalFacetRefinement = function(facet, value) {\n this._change({\n state: this.state.resetPage().addHierarchicalFacetRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Adds a an numeric filter to an attribute with the `operator` and `value` provided. If the\n * filter is already set, it doesn't change the filters.\n *\n * This method resets the current page to 0.\n * @param {string} attribute the attribute on which the numeric filter applies\n * @param {string} operator the operator of the filter\n * @param {number} value the value of the filter\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.addNumericRefinement = function(attribute, operator, value) {\n this._change({\n state: this.state.resetPage().addNumericRefinement(attribute, operator, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Adds a filter to a faceted attribute with the `value` provided. If the\n * filter is already set, it doesn't change the filters.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} value the associated value (will be converted to string)\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.addFacetRefinement = function(facet, value) {\n this._change({\n state: this.state.resetPage().addFacetRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#addFacetRefinement}\n */\nAlgoliaSearchHelper.prototype.addRefine = function() {\n return this.addFacetRefinement.apply(this, arguments);\n};\n\n\n/**\n * Adds a an exclusion filter to a faceted attribute with the `value` provided. If the\n * filter is already set, it doesn't change the filters.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} value the associated value (will be converted to string)\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.addFacetExclusion = function(facet, value) {\n this._change({\n state: this.state.resetPage().addExcludeRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#addFacetExclusion}\n */\nAlgoliaSearchHelper.prototype.addExclude = function() {\n return this.addFacetExclusion.apply(this, arguments);\n};\n\n/**\n * Adds a tag filter with the `tag` provided. If the\n * filter is already set, it doesn't change the filters.\n *\n * This method resets the current page to 0.\n * @param {string} tag the tag to add to the filter\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.addTag = function(tag) {\n this._change({\n state: this.state.resetPage().addTagRefinement(tag),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Removes an numeric filter to an attribute with the `operator` and `value` provided. If the\n * filter is not set, it doesn't change the filters.\n *\n * Some parameters are optional, triggering different behavior:\n * - if the value is not provided, then all the numeric value will be removed for the\n * specified attribute/operator couple.\n * - if the operator is not provided either, then all the numeric filter on this attribute\n * will be removed.\n *\n * This method resets the current page to 0.\n * @param {string} attribute the attribute on which the numeric filter applies\n * @param {string} [operator] the operator of the filter\n * @param {number} [value] the value of the filter\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.removeNumericRefinement = function(attribute, operator, value) {\n this._change({\n state: this.state.resetPage().removeNumericRefinement(attribute, operator, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Removes a disjunctive filter to a faceted attribute with the `value` provided. If the\n * filter is not set, it doesn't change the filters.\n *\n * If the value is omitted, then this method will remove all the filters for the\n * attribute.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} [value] the associated value\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.removeDisjunctiveFacetRefinement = function(facet, value) {\n this._change({\n state: this.state.resetPage().removeDisjunctiveFacetRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#removeDisjunctiveFacetRefinement}\n */\nAlgoliaSearchHelper.prototype.removeDisjunctiveRefine = function() {\n return this.removeDisjunctiveFacetRefinement.apply(this, arguments);\n};\n\n/**\n * Removes the refinement set on a hierarchical facet.\n * @param {string} facet the facet name\n * @return {AlgoliaSearchHelper}\n * @throws Error if the facet is not defined or if the facet is not refined\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.removeHierarchicalFacetRefinement = function(facet) {\n this._change({\n state: this.state.resetPage().removeHierarchicalFacetRefinement(facet),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Removes a filter to a faceted attribute with the `value` provided. If the\n * filter is not set, it doesn't change the filters.\n *\n * If the value is omitted, then this method will remove all the filters for the\n * attribute.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} [value] the associated value\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.removeFacetRefinement = function(facet, value) {\n this._change({\n state: this.state.resetPage().removeFacetRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#removeFacetRefinement}\n */\nAlgoliaSearchHelper.prototype.removeRefine = function() {\n return this.removeFacetRefinement.apply(this, arguments);\n};\n\n/**\n * Removes an exclusion filter to a faceted attribute with the `value` provided. If the\n * filter is not set, it doesn't change the filters.\n *\n * If the value is omitted, then this method will remove all the filters for the\n * attribute.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} [value] the associated value\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.removeFacetExclusion = function(facet, value) {\n this._change({\n state: this.state.resetPage().removeExcludeRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#removeFacetExclusion}\n */\nAlgoliaSearchHelper.prototype.removeExclude = function() {\n return this.removeFacetExclusion.apply(this, arguments);\n};\n\n/**\n * Removes a tag filter with the `tag` provided. If the\n * filter is not set, it doesn't change the filters.\n *\n * This method resets the current page to 0.\n * @param {string} tag tag to remove from the filter\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.removeTag = function(tag) {\n this._change({\n state: this.state.resetPage().removeTagRefinement(tag),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Adds or removes an exclusion filter to a faceted attribute with the `value` provided. If\n * the value is set then it removes it, otherwise it adds the filter.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} value the associated value\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.toggleFacetExclusion = function(facet, value) {\n this._change({\n state: this.state.resetPage().toggleExcludeFacetRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#toggleFacetExclusion}\n */\nAlgoliaSearchHelper.prototype.toggleExclude = function() {\n return this.toggleFacetExclusion.apply(this, arguments);\n};\n\n/**\n * Adds or removes a filter to a faceted attribute with the `value` provided. If\n * the value is set then it removes it, otherwise it adds the filter.\n *\n * This method can be used for conjunctive, disjunctive and hierarchical filters.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} value the associated value\n * @return {AlgoliaSearchHelper}\n * @throws Error will throw an error if the facet is not declared in the settings of the helper\n * @fires change\n * @chainable\n * @deprecated since version 2.19.0, see {@link AlgoliaSearchHelper#toggleFacetRefinement}\n */\nAlgoliaSearchHelper.prototype.toggleRefinement = function(facet, value) {\n return this.toggleFacetRefinement(facet, value);\n};\n\n/**\n * Adds or removes a filter to a faceted attribute with the `value` provided. If\n * the value is set then it removes it, otherwise it adds the filter.\n *\n * This method can be used for conjunctive, disjunctive and hierarchical filters.\n *\n * This method resets the current page to 0.\n * @param {string} facet the facet to refine\n * @param {string} value the associated value\n * @return {AlgoliaSearchHelper}\n * @throws Error will throw an error if the facet is not declared in the settings of the helper\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.toggleFacetRefinement = function(facet, value) {\n this._change({\n state: this.state.resetPage().toggleFacetRefinement(facet, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * @deprecated since version 2.4.0, see {@link AlgoliaSearchHelper#toggleFacetRefinement}\n */\nAlgoliaSearchHelper.prototype.toggleRefine = function() {\n return this.toggleFacetRefinement.apply(this, arguments);\n};\n\n/**\n * Adds or removes a tag filter with the `value` provided. If\n * the value is set then it removes it, otherwise it adds the filter.\n *\n * This method resets the current page to 0.\n * @param {string} tag tag to remove or add\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.toggleTag = function(tag) {\n this._change({\n state: this.state.resetPage().toggleTagRefinement(tag),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Increments the page number by one.\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n * @example\n * helper.setPage(0).nextPage().getPage();\n * // returns 1\n */\nAlgoliaSearchHelper.prototype.nextPage = function() {\n var page = this.state.page || 0;\n return this.setPage(page + 1);\n};\n\n/**\n * Decrements the page number by one.\n * @fires change\n * @return {AlgoliaSearchHelper}\n * @chainable\n * @example\n * helper.setPage(1).previousPage().getPage();\n * // returns 0\n */\nAlgoliaSearchHelper.prototype.previousPage = function() {\n var page = this.state.page || 0;\n return this.setPage(page - 1);\n};\n\n/**\n * @private\n */\nfunction setCurrentPage(page) {\n if (page < 0) throw new Error('Page requested below 0.');\n\n this._change({\n state: this.state.setPage(page),\n isPageReset: false\n });\n\n return this;\n}\n\n/**\n * Change the current page\n * @deprecated\n * @param {number} page The page number\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.setCurrentPage = setCurrentPage;\n\n/**\n * Updates the current page.\n * @function\n * @param {number} page The page number\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.setPage = setCurrentPage;\n\n/**\n * Updates the name of the index that will be targeted by the query.\n *\n * This method resets the current page to 0.\n * @param {string} name the index name\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.setIndex = function(name) {\n this._change({\n state: this.state.resetPage().setIndex(name),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Update a parameter of the search. This method reset the page\n *\n * The complete list of parameters is available on the\n * [Algolia website](https://www.algolia.com/doc/rest#query-an-index).\n * The most commonly used parameters have their own [shortcuts](#query-parameters-shortcuts)\n * or benefit from higher-level APIs (all the kind of filters and facets have their own API)\n *\n * This method resets the current page to 0.\n * @param {string} parameter name of the parameter to update\n * @param {any} value new value of the parameter\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n * @example\n * helper.setQueryParameter('hitsPerPage', 20).search();\n */\nAlgoliaSearchHelper.prototype.setQueryParameter = function(parameter, value) {\n this._change({\n state: this.state.resetPage().setQueryParameter(parameter, value),\n isPageReset: true\n });\n\n return this;\n};\n\n/**\n * Set the whole state (warning: will erase previous state)\n * @param {SearchParameters} newState the whole new state\n * @return {AlgoliaSearchHelper}\n * @fires change\n * @chainable\n */\nAlgoliaSearchHelper.prototype.setState = function(newState) {\n this._change({\n state: SearchParameters.make(newState),\n isPageReset: false\n });\n\n return this;\n};\n\n/**\n * Override the current state without triggering a change event.\n * Do not use this method unless you know what you are doing. (see the example\n * for a legit use case)\n * @param {SearchParameters} newState the whole new state\n * @return {AlgoliaSearchHelper}\n * @example\n * helper.on('change', function(state){\n * // In this function you might want to find a way to store the state in the url/history\n * updateYourURL(state)\n * })\n * window.onpopstate = function(event){\n * // This is naive though as you should check if the state is really defined etc.\n * helper.overrideStateWithoutTriggeringChangeEvent(event.state).search()\n * }\n * @chainable\n */\nAlgoliaSearchHelper.prototype.overrideStateWithoutTriggeringChangeEvent = function(newState) {\n this.state = new SearchParameters(newState);\n return this;\n};\n\n/**\n * Check if an attribute has any numeric, conjunctive, disjunctive or hierarchical filters.\n * @param {string} attribute the name of the attribute\n * @return {boolean} true if the attribute is filtered by at least one value\n * @example\n * // hasRefinements works with numeric, conjunctive, disjunctive and hierarchical filters\n * helper.hasRefinements('price'); // false\n * helper.addNumericRefinement('price', '>', 100);\n * helper.hasRefinements('price'); // true\n *\n * helper.hasRefinements('color'); // false\n * helper.addFacetRefinement('color', 'blue');\n * helper.hasRefinements('color'); // true\n *\n * helper.hasRefinements('material'); // false\n * helper.addDisjunctiveFacetRefinement('material', 'plastic');\n * helper.hasRefinements('material'); // true\n *\n * helper.hasRefinements('categories'); // false\n * helper.toggleFacetRefinement('categories', 'kitchen > knife');\n * helper.hasRefinements('categories'); // true\n *\n */\nAlgoliaSearchHelper.prototype.hasRefinements = function(attribute) {\n if (objectHasKeys(this.state.getNumericRefinements(attribute))) {\n return true;\n } else if (this.state.isConjunctiveFacet(attribute)) {\n return this.state.isFacetRefined(attribute);\n } else if (this.state.isDisjunctiveFacet(attribute)) {\n return this.state.isDisjunctiveFacetRefined(attribute);\n } else if (this.state.isHierarchicalFacet(attribute)) {\n return this.state.isHierarchicalFacetRefined(attribute);\n }\n\n // there's currently no way to know that the user did call `addNumericRefinement` at some point\n // thus we cannot distinguish if there once was a numeric refinement that was cleared\n // so we will return false in every other situations to be consistent\n // while what we should do here is throw because we did not find the attribute in any type\n // of refinement\n return false;\n};\n\n/**\n * Check if a value is excluded for a specific faceted attribute. If the value\n * is omitted then the function checks if there is any excluding refinements.\n *\n * @param {string} facet name of the attribute for used for faceting\n * @param {string} [value] optional value. If passed will test that this value\n * is filtering the given facet.\n * @return {boolean} true if refined\n * @example\n * helper.isExcludeRefined('color'); // false\n * helper.isExcludeRefined('color', 'blue') // false\n * helper.isExcludeRefined('color', 'red') // false\n *\n * helper.addFacetExclusion('color', 'red');\n *\n * helper.isExcludeRefined('color'); // true\n * helper.isExcludeRefined('color', 'blue') // false\n * helper.isExcludeRefined('color', 'red') // true\n */\nAlgoliaSearchHelper.prototype.isExcluded = function(facet, value) {\n return this.state.isExcludeRefined(facet, value);\n};\n\n/**\n * @deprecated since 2.4.0, see {@link AlgoliaSearchHelper#hasRefinements}\n */\nAlgoliaSearchHelper.prototype.isDisjunctiveRefined = function(facet, value) {\n return this.state.isDisjunctiveFacetRefined(facet, value);\n};\n\n/**\n * Check if the string is a currently filtering tag.\n * @param {string} tag tag to check\n * @return {boolean}\n */\nAlgoliaSearchHelper.prototype.hasTag = function(tag) {\n return this.state.isTagRefined(tag);\n};\n\n/**\n * @deprecated since 2.4.0, see {@link AlgoliaSearchHelper#hasTag}\n */\nAlgoliaSearchHelper.prototype.isTagRefined = function() {\n return this.hasTagRefinements.apply(this, arguments);\n};\n\n\n/**\n * Get the name of the currently used index.\n * @return {string}\n * @example\n * helper.setIndex('highestPrice_products').getIndex();\n * // returns 'highestPrice_products'\n */\nAlgoliaSearchHelper.prototype.getIndex = function() {\n return this.state.index;\n};\n\nfunction getCurrentPage() {\n return this.state.page;\n}\n\n/**\n * Get the currently selected page\n * @deprecated\n * @return {number} the current page\n */\nAlgoliaSearchHelper.prototype.getCurrentPage = getCurrentPage;\n/**\n * Get the currently selected page\n * @function\n * @return {number} the current page\n */\nAlgoliaSearchHelper.prototype.getPage = getCurrentPage;\n\n/**\n * Get all the tags currently set to filters the results.\n *\n * @return {string[]} The list of tags currently set.\n */\nAlgoliaSearchHelper.prototype.getTags = function() {\n return this.state.tagRefinements;\n};\n\n/**\n * Get the list of refinements for a given attribute. This method works with\n * conjunctive, disjunctive, excluding and numerical filters.\n *\n * See also SearchResults#getRefinements\n *\n * @param {string} facetName attribute name used for faceting\n * @return {Array.} All Refinement are objects that contain a value, and\n * a type. Numeric also contains an operator.\n * @example\n * helper.addNumericRefinement('price', '>', 100);\n * helper.getRefinements('price');\n * // [\n * // {\n * // \"value\": [\n * // 100\n * // ],\n * // \"operator\": \">\",\n * // \"type\": \"numeric\"\n * // }\n * // ]\n * @example\n * helper.addFacetRefinement('color', 'blue');\n * helper.addFacetExclusion('color', 'red');\n * helper.getRefinements('color');\n * // [\n * // {\n * // \"value\": \"blue\",\n * // \"type\": \"conjunctive\"\n * // },\n * // {\n * // \"value\": \"red\",\n * // \"type\": \"exclude\"\n * // }\n * // ]\n * @example\n * helper.addDisjunctiveFacetRefinement('material', 'plastic');\n * // [\n * // {\n * // \"value\": \"plastic\",\n * // \"type\": \"disjunctive\"\n * // }\n * // ]\n */\nAlgoliaSearchHelper.prototype.getRefinements = function(facetName) {\n var refinements = [];\n\n if (this.state.isConjunctiveFacet(facetName)) {\n var conjRefinements = this.state.getConjunctiveRefinements(facetName);\n\n conjRefinements.forEach(function(r) {\n refinements.push({\n value: r,\n type: 'conjunctive'\n });\n });\n\n var excludeRefinements = this.state.getExcludeRefinements(facetName);\n\n excludeRefinements.forEach(function(r) {\n refinements.push({\n value: r,\n type: 'exclude'\n });\n });\n } else if (this.state.isDisjunctiveFacet(facetName)) {\n var disjRefinements = this.state.getDisjunctiveRefinements(facetName);\n\n disjRefinements.forEach(function(r) {\n refinements.push({\n value: r,\n type: 'disjunctive'\n });\n });\n }\n\n var numericRefinements = this.state.getNumericRefinements(facetName);\n\n Object.keys(numericRefinements).forEach(function(operator) {\n var value = numericRefinements[operator];\n\n refinements.push({\n value: value,\n operator: operator,\n type: 'numeric'\n });\n });\n\n return refinements;\n};\n\n/**\n * Return the current refinement for the (attribute, operator)\n * @param {string} attribute attribute in the record\n * @param {string} operator operator applied on the refined values\n * @return {Array.} refined values\n */\nAlgoliaSearchHelper.prototype.getNumericRefinement = function(attribute, operator) {\n return this.state.getNumericRefinement(attribute, operator);\n};\n\n/**\n * Get the current breadcrumb for a hierarchical facet, as an array\n * @param {string} facetName Hierarchical facet name\n * @return {array.} the path as an array of string\n */\nAlgoliaSearchHelper.prototype.getHierarchicalFacetBreadcrumb = function(facetName) {\n return this.state.getHierarchicalFacetBreadcrumb(facetName);\n};\n\n// /////////// PRIVATE\n\n/**\n * Perform the underlying queries\n * @private\n * @return {undefined}\n * @fires search\n * @fires result\n * @fires error\n */\nAlgoliaSearchHelper.prototype._search = function(options) {\n var state = this.state;\n var states = [];\n var mainQueries = [];\n\n if (!options.onlyWithDerivedHelpers) {\n mainQueries = requestBuilder._getQueries(state.index, state);\n\n states.push({\n state: state,\n queriesCount: mainQueries.length,\n helper: this\n });\n\n this.emit('search', {\n state: state,\n results: this.lastResults\n });\n }\n\n var derivedQueries = this.derivedHelpers.map(function(derivedHelper) {\n var derivedState = derivedHelper.getModifiedState(state);\n var derivedStateQueries = requestBuilder._getQueries(derivedState.index, derivedState);\n\n states.push({\n state: derivedState,\n queriesCount: derivedStateQueries.length,\n helper: derivedHelper\n });\n\n derivedHelper.emit('search', {\n state: derivedState,\n results: derivedHelper.lastResults\n });\n\n return derivedStateQueries;\n });\n\n var queries = Array.prototype.concat.apply(mainQueries, derivedQueries);\n var queryId = this._queryId++;\n\n this._currentNbQueries++;\n\n try {\n this.client.search(queries)\n .then(this._dispatchAlgoliaResponse.bind(this, states, queryId))\n .catch(this._dispatchAlgoliaError.bind(this, queryId));\n } catch (error) {\n // If we reach this part, we're in an internal error state\n this.emit('error', {\n error: error\n });\n }\n};\n\n/**\n * Transform the responses as sent by the server and transform them into a user\n * usable object that merge the results of all the batch requests. It will dispatch\n * over the different helper + derived helpers (when there are some).\n * @private\n * @param {array.<{SearchParameters, AlgoliaQueries, AlgoliaSearchHelper}>}\n * state state used for to generate the request\n * @param {number} queryId id of the current request\n * @param {object} content content of the response\n * @return {undefined}\n */\nAlgoliaSearchHelper.prototype._dispatchAlgoliaResponse = function(states, queryId, content) {\n // FIXME remove the number of outdated queries discarded instead of just one\n\n if (queryId < this._lastQueryIdReceived) {\n // Outdated answer\n return;\n }\n\n this._currentNbQueries -= (queryId - this._lastQueryIdReceived);\n this._lastQueryIdReceived = queryId;\n\n if (this._currentNbQueries === 0) this.emit('searchQueueEmpty');\n\n var results = content.results.slice();\n\n states.forEach(function(s) {\n var state = s.state;\n var queriesCount = s.queriesCount;\n var helper = s.helper;\n var specificResults = results.splice(0, queriesCount);\n\n var formattedResponse = helper.lastResults = new SearchResults(state, specificResults);\n\n helper.emit('result', {\n results: formattedResponse,\n state: state\n });\n });\n};\n\nAlgoliaSearchHelper.prototype._dispatchAlgoliaError = function(queryId, error) {\n if (queryId < this._lastQueryIdReceived) {\n // Outdated answer\n return;\n }\n\n this._currentNbQueries -= queryId - this._lastQueryIdReceived;\n this._lastQueryIdReceived = queryId;\n\n this.emit('error', {\n error: error\n });\n\n if (this._currentNbQueries === 0) this.emit('searchQueueEmpty');\n};\n\nAlgoliaSearchHelper.prototype.containsRefinement = function(query, facetFilters, numericFilters, tagFilters) {\n return query ||\n facetFilters.length !== 0 ||\n numericFilters.length !== 0 ||\n tagFilters.length !== 0;\n};\n\n/**\n * Test if there are some disjunctive refinements on the facet\n * @private\n * @param {string} facet the attribute to test\n * @return {boolean}\n */\nAlgoliaSearchHelper.prototype._hasDisjunctiveRefinements = function(facet) {\n return this.state.disjunctiveRefinements[facet] &&\n this.state.disjunctiveRefinements[facet].length > 0;\n};\n\nAlgoliaSearchHelper.prototype._change = function(event) {\n var state = event.state;\n var isPageReset = event.isPageReset;\n\n if (state !== this.state) {\n this.state = state;\n\n this.emit('change', {\n state: this.state,\n results: this.lastResults,\n isPageReset: isPageReset\n });\n }\n};\n\n/**\n * Clears the cache of the underlying Algolia client.\n * @return {AlgoliaSearchHelper}\n */\nAlgoliaSearchHelper.prototype.clearCache = function() {\n this.client.clearCache && this.client.clearCache();\n return this;\n};\n\n/**\n * Updates the internal client instance. If the reference of the clients\n * are equal then no update is actually done.\n * @param {AlgoliaSearch} newClient an AlgoliaSearch client\n * @return {AlgoliaSearchHelper}\n */\nAlgoliaSearchHelper.prototype.setClient = function(newClient) {\n if (this.client === newClient) return this;\n\n if (typeof newClient.addAlgoliaAgent === 'function') {\n newClient.addAlgoliaAgent('JS Helper (' + version + ')');\n }\n this.client = newClient;\n\n return this;\n};\n\n/**\n * Gets the instance of the currently used client.\n * @return {AlgoliaSearch}\n */\nAlgoliaSearchHelper.prototype.getClient = function() {\n return this.client;\n};\n\n/**\n * Creates an derived instance of the Helper. A derived helper\n * is a way to request other indices synchronised with the lifecycle\n * of the main Helper. This mechanism uses the multiqueries feature\n * of Algolia to aggregate all the requests in a single network call.\n *\n * This method takes a function that is used to create a new SearchParameter\n * that will be used to create requests to Algolia. Those new requests\n * are created just before the `search` event. The signature of the function\n * is `SearchParameters -> SearchParameters`.\n *\n * This method returns a new DerivedHelper which is an EventEmitter\n * that fires the same `search`, `result` and `error` events. Those\n * events, however, will receive data specific to this DerivedHelper\n * and the SearchParameters that is returned by the call of the\n * parameter function.\n * @param {function} fn SearchParameters -> SearchParameters\n * @return {DerivedHelper}\n */\nAlgoliaSearchHelper.prototype.derive = function(fn) {\n var derivedHelper = new DerivedHelper(this, fn);\n this.derivedHelpers.push(derivedHelper);\n return derivedHelper;\n};\n\n/**\n * This method detaches a derived Helper from the main one. Prefer using the one from the\n * derived helper itself, to remove the event listeners too.\n * @private\n * @return {undefined}\n * @throws Error\n */\nAlgoliaSearchHelper.prototype.detachDerivedHelper = function(derivedHelper) {\n var pos = this.derivedHelpers.indexOf(derivedHelper);\n if (pos === -1) throw new Error('Derived helper already detached');\n this.derivedHelpers.splice(pos, 1);\n};\n\n/**\n * This method returns true if there is currently at least one on-going search.\n * @return {boolean} true if there is a search pending\n */\nAlgoliaSearchHelper.prototype.hasPendingRequests = function() {\n return this._currentNbQueries > 0;\n};\n\n/**\n * @typedef AlgoliaSearchHelper.NumericRefinement\n * @type {object}\n * @property {number[]} value the numbers that are used for filtering this attribute with\n * the operator specified.\n * @property {string} operator the faceting data: value, number of entries\n * @property {string} type will be 'numeric'\n */\n\n/**\n * @typedef AlgoliaSearchHelper.FacetRefinement\n * @type {object}\n * @property {string} value the string use to filter the attribute\n * @property {string} type the type of filter: 'conjunctive', 'disjunctive', 'exclude'\n */\n\nmodule.exports = AlgoliaSearchHelper;\n","'use strict';\n\nvar AlgoliaSearchHelper = require('./src/algoliasearch.helper');\n\nvar SearchParameters = require('./src/SearchParameters');\nvar SearchResults = require('./src/SearchResults');\n\n/**\n * The algoliasearchHelper module is the function that will let its\n * contains everything needed to use the Algoliasearch\n * Helper. It is a also a function that instanciate the helper.\n * To use the helper, you also need the Algolia JS client v3.\n * @example\n * //using the UMD build\n * var client = algoliasearch('latency', '6be0576ff61c053d5f9a3225e2a90f76');\n * var helper = algoliasearchHelper(client, 'bestbuy', {\n * facets: ['shipping'],\n * disjunctiveFacets: ['category']\n * });\n * helper.on('result', function(event) {\n * console.log(event.results);\n * });\n * helper\n * .toggleFacetRefinement('category', 'Movies & TV Shows')\n * .toggleFacetRefinement('shipping', 'Free shipping')\n * .search();\n * @example\n * // The helper is an event emitter using the node API\n * helper.on('result', updateTheResults);\n * helper.once('result', updateTheResults);\n * helper.removeListener('result', updateTheResults);\n * helper.removeAllListeners('result');\n * @module algoliasearchHelper\n * @param {AlgoliaSearch} client an AlgoliaSearch client\n * @param {string} index the name of the index to query\n * @param {SearchParameters|object} opts an object defining the initial config of the search. It doesn't have to be a {SearchParameters}, just an object containing the properties you need from it.\n * @return {AlgoliaSearchHelper}\n */\nfunction algoliasearchHelper(client, index, opts) {\n return new AlgoliaSearchHelper(client, index, opts);\n}\n\n/**\n * The version currently used\n * @member module:algoliasearchHelper.version\n * @type {number}\n */\nalgoliasearchHelper.version = require('./src/version.js');\n\n/**\n * Constructor for the Helper.\n * @member module:algoliasearchHelper.AlgoliaSearchHelper\n * @type {AlgoliaSearchHelper}\n */\nalgoliasearchHelper.AlgoliaSearchHelper = AlgoliaSearchHelper;\n\n/**\n * Constructor for the object containing all the parameters of the search.\n * @member module:algoliasearchHelper.SearchParameters\n * @type {SearchParameters}\n */\nalgoliasearchHelper.SearchParameters = SearchParameters;\n\n/**\n * Constructor for the object containing the results of the search.\n * @member module:algoliasearchHelper.SearchResults\n * @type {SearchResults}\n */\nalgoliasearchHelper.SearchResults = SearchResults;\n\nmodule.exports = algoliasearchHelper;\n","function capitalize(text: string): string {\n return text.toString().charAt(0).toUpperCase() + text.toString().slice(1);\n}\n\nexport default capitalize;\n","const nextMicroTask = Promise.resolve();\n\ntype Callback = (...args: any[]) => void;\ntype Defer = Callback & {\n wait(): Promise;\n cancel(): void;\n};\n\nconst defer = (callback: Callback): Defer => {\n let progress: Promise | null = null;\n let cancelled = false;\n\n const fn: Defer = (...args) => {\n if (progress !== null) {\n return;\n }\n\n progress = nextMicroTask.then(() => {\n progress = null;\n\n if (cancelled) {\n cancelled = false;\n return;\n }\n\n callback(...args);\n });\n };\n\n fn.wait = () => {\n if (progress === null) {\n throw new Error(\n 'The deferred function should be called before calling `wait()`'\n );\n }\n\n return progress;\n };\n\n fn.cancel = () => {\n if (progress === null) {\n return;\n }\n\n cancelled = true;\n };\n\n return fn;\n};\n\nexport default defer;\n","function isDomElement(object: any): object is HTMLElement {\n return (\n object instanceof HTMLElement || (Boolean(object) && object.nodeType > 0)\n );\n}\n\nexport default isDomElement;\n","import isDomElement from './isDomElement';\n\n/**\n * Return the container. If it's a string, it is considered a\n * css selector and retrieves the first matching element. Otherwise\n * test if it validates that it's a correct DOMElement.\n *\n * @param {string|HTMLElement} selectorOrHTMLElement CSS Selector or container node.\n * @return {HTMLElement} Container node\n * @throws Error when the type is not correct\n */\nfunction getContainerNode(\n selectorOrHTMLElement: string | HTMLElement\n): HTMLElement {\n const isSelectorString = typeof selectorOrHTMLElement === 'string';\n const domElement = isSelectorString\n ? document.querySelector(selectorOrHTMLElement as string)\n : selectorOrHTMLElement;\n\n if (!isDomElement(domElement)) {\n let errorMessage = 'Container must be `string` or `HTMLElement`.';\n\n if (isSelectorString) {\n errorMessage += ` Unable to find ${selectorOrHTMLElement}`;\n }\n\n throw new Error(errorMessage);\n }\n\n return domElement;\n}\n\nexport default getContainerNode;\n","function isSpecialClick(event: MouseEvent): boolean {\n const isMiddleClick = event.button === 1;\n\n return (\n isMiddleClick ||\n event.altKey ||\n event.ctrlKey ||\n event.metaKey ||\n event.shiftKey\n );\n}\n\nexport default isSpecialClick;\n","function uniq(array: TItem[]): TItem[] {\n return array.filter((value, index, self) => self.indexOf(value) === index);\n}\n\nexport default uniq;\n","import uniq from './uniq';\nimport type { HoganHelpers, Templates } from '../../types';\nimport type { HoganOptions } from 'hogan.js';\n\ntype TemplatesConfig = {\n helpers?: HoganHelpers;\n compileOptions?: HoganOptions;\n};\n\nexport type PreparedTemplateProps = {\n templatesConfig: TemplatesConfig;\n templates: TTemplates;\n useCustomCompileOptions: {\n [TKey in keyof Partial]: boolean;\n };\n};\n\nfunction prepareTemplates(\n // can not use = {} here, since the template could have different constraints\n defaultTemplates?: TTemplates,\n templates: Partial = {}\n) {\n const allKeys = uniq([\n ...Object.keys(defaultTemplates || {}),\n ...Object.keys(templates),\n ]);\n\n return allKeys.reduce(\n (config, key: keyof TTemplates) => {\n const defaultTemplate = defaultTemplates\n ? defaultTemplates[key]\n : undefined;\n const customTemplate = templates[key];\n const isCustomTemplate =\n customTemplate !== undefined && customTemplate !== defaultTemplate;\n\n config.templates[key] = isCustomTemplate\n ? customTemplate! // typescript doesn't recognize that this condition asserts customTemplate is defined\n : defaultTemplate!;\n\n config.useCustomCompileOptions[key] = isCustomTemplate;\n\n return config;\n },\n {\n // eslint-disable-next-line @typescript-eslint/consistent-type-assertions\n templates: {} as TTemplates,\n // eslint-disable-next-line @typescript-eslint/consistent-type-assertions\n useCustomCompileOptions: {} as {\n [TKey in keyof TTemplates]: boolean;\n },\n }\n );\n}\n\n/**\n * Prepares an object to be passed to the Template widget\n */\nfunction prepareTemplateProps({\n defaultTemplates,\n templates,\n templatesConfig,\n}: {\n defaultTemplates: TTemplates;\n templates?: Partial;\n templatesConfig: TemplatesConfig;\n}): PreparedTemplateProps {\n const preparedTemplates = prepareTemplates(defaultTemplates, templates);\n\n return {\n templatesConfig,\n ...preparedTemplates,\n };\n}\n\nexport default prepareTemplateProps;\n","/*\n * Copyright 2011 Twitter, Inc.\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n(function (Hogan) {\n // Setup regex assignments\n // remove whitespace according to Mustache spec\n var rIsWhitespace = /\\S/,\n rQuot = /\\\"/g,\n rNewline = /\\n/g,\n rCr = /\\r/g,\n rSlash = /\\\\/g,\n rLineSep = /\\u2028/,\n rParagraphSep = /\\u2029/;\n\n Hogan.tags = {\n '#': 1, '^': 2, '<': 3, '$': 4,\n '/': 5, '!': 6, '>': 7, '=': 8, '_v': 9,\n '{': 10, '&': 11, '_t': 12\n };\n\n Hogan.scan = function scan(text, delimiters) {\n var len = text.length,\n IN_TEXT = 0,\n IN_TAG_TYPE = 1,\n IN_TAG = 2,\n state = IN_TEXT,\n tagType = null,\n tag = null,\n buf = '',\n tokens = [],\n seenTag = false,\n i = 0,\n lineStart = 0,\n otag = '{{',\n ctag = '}}';\n\n function addBuf() {\n if (buf.length > 0) {\n tokens.push({tag: '_t', text: new String(buf)});\n buf = '';\n }\n }\n\n function lineIsWhitespace() {\n var isAllWhitespace = true;\n for (var j = lineStart; j < tokens.length; j++) {\n isAllWhitespace =\n (Hogan.tags[tokens[j].tag] < Hogan.tags['_v']) ||\n (tokens[j].tag == '_t' && tokens[j].text.match(rIsWhitespace) === null);\n if (!isAllWhitespace) {\n return false;\n }\n }\n\n return isAllWhitespace;\n }\n\n function filterLine(haveSeenTag, noNewLine) {\n addBuf();\n\n if (haveSeenTag && lineIsWhitespace()) {\n for (var j = lineStart, next; j < tokens.length; j++) {\n if (tokens[j].text) {\n if ((next = tokens[j+1]) && next.tag == '>') {\n // set indent to token value\n next.indent = tokens[j].text.toString()\n }\n tokens.splice(j, 1);\n }\n }\n } else if (!noNewLine) {\n tokens.push({tag:'\\n'});\n }\n\n seenTag = false;\n lineStart = tokens.length;\n }\n\n function changeDelimiters(text, index) {\n var close = '=' + ctag,\n closeIndex = text.indexOf(close, index),\n delimiters = trim(\n text.substring(text.indexOf('=', index) + 1, closeIndex)\n ).split(' ');\n\n otag = delimiters[0];\n ctag = delimiters[delimiters.length - 1];\n\n return closeIndex + close.length - 1;\n }\n\n if (delimiters) {\n delimiters = delimiters.split(' ');\n otag = delimiters[0];\n ctag = delimiters[1];\n }\n\n for (i = 0; i < len; i++) {\n if (state == IN_TEXT) {\n if (tagChange(otag, text, i)) {\n --i;\n addBuf();\n state = IN_TAG_TYPE;\n } else {\n if (text.charAt(i) == '\\n') {\n filterLine(seenTag);\n } else {\n buf += text.charAt(i);\n }\n }\n } else if (state == IN_TAG_TYPE) {\n i += otag.length - 1;\n tag = Hogan.tags[text.charAt(i + 1)];\n tagType = tag ? text.charAt(i + 1) : '_v';\n if (tagType == '=') {\n i = changeDelimiters(text, i);\n state = IN_TEXT;\n } else {\n if (tag) {\n i++;\n }\n state = IN_TAG;\n }\n seenTag = i;\n } else {\n if (tagChange(ctag, text, i)) {\n tokens.push({tag: tagType, n: trim(buf), otag: otag, ctag: ctag,\n i: (tagType == '/') ? seenTag - otag.length : i + ctag.length});\n buf = '';\n i += ctag.length - 1;\n state = IN_TEXT;\n if (tagType == '{') {\n if (ctag == '}}') {\n i++;\n } else {\n cleanTripleStache(tokens[tokens.length - 1]);\n }\n }\n } else {\n buf += text.charAt(i);\n }\n }\n }\n\n filterLine(seenTag, true);\n\n return tokens;\n }\n\n function cleanTripleStache(token) {\n if (token.n.substr(token.n.length - 1) === '}') {\n token.n = token.n.substring(0, token.n.length - 1);\n }\n }\n\n function trim(s) {\n if (s.trim) {\n return s.trim();\n }\n\n return s.replace(/^\\s*|\\s*$/g, '');\n }\n\n function tagChange(tag, text, index) {\n if (text.charAt(index) != tag.charAt(0)) {\n return false;\n }\n\n for (var i = 1, l = tag.length; i < l; i++) {\n if (text.charAt(index + i) != tag.charAt(i)) {\n return false;\n }\n }\n\n return true;\n }\n\n // the tags allowed inside super templates\n var allowedInSuper = {'_t': true, '\\n': true, '$': true, '/': true};\n\n function buildTree(tokens, kind, stack, customTags) {\n var instructions = [],\n opener = null,\n tail = null,\n token = null;\n\n tail = stack[stack.length - 1];\n\n while (tokens.length > 0) {\n token = tokens.shift();\n\n if (tail && tail.tag == '<' && !(token.tag in allowedInSuper)) {\n throw new Error('Illegal content in < super tag.');\n }\n\n if (Hogan.tags[token.tag] <= Hogan.tags['$'] || isOpener(token, customTags)) {\n stack.push(token);\n token.nodes = buildTree(tokens, token.tag, stack, customTags);\n } else if (token.tag == '/') {\n if (stack.length === 0) {\n throw new Error('Closing tag without opener: /' + token.n);\n }\n opener = stack.pop();\n if (token.n != opener.n && !isCloser(token.n, opener.n, customTags)) {\n throw new Error('Nesting error: ' + opener.n + ' vs. ' + token.n);\n }\n opener.end = token.i;\n return instructions;\n } else if (token.tag == '\\n') {\n token.last = (tokens.length == 0) || (tokens[0].tag == '\\n');\n }\n\n instructions.push(token);\n }\n\n if (stack.length > 0) {\n throw new Error('missing closing tag: ' + stack.pop().n);\n }\n\n return instructions;\n }\n\n function isOpener(token, tags) {\n for (var i = 0, l = tags.length; i < l; i++) {\n if (tags[i].o == token.n) {\n token.tag = '#';\n return true;\n }\n }\n }\n\n function isCloser(close, open, tags) {\n for (var i = 0, l = tags.length; i < l; i++) {\n if (tags[i].c == close && tags[i].o == open) {\n return true;\n }\n }\n }\n\n function stringifySubstitutions(obj) {\n var items = [];\n for (var key in obj) {\n items.push('\"' + esc(key) + '\": function(c,p,t,i) {' + obj[key] + '}');\n }\n return \"{ \" + items.join(\",\") + \" }\";\n }\n\n function stringifyPartials(codeObj) {\n var partials = [];\n for (var key in codeObj.partials) {\n partials.push('\"' + esc(key) + '\":{name:\"' + esc(codeObj.partials[key].name) + '\", ' + stringifyPartials(codeObj.partials[key]) + \"}\");\n }\n return \"partials: {\" + partials.join(\",\") + \"}, subs: \" + stringifySubstitutions(codeObj.subs);\n }\n\n Hogan.stringify = function(codeObj, text, options) {\n return \"{code: function (c,p,i) { \" + Hogan.wrapMain(codeObj.code) + \" },\" + stringifyPartials(codeObj) + \"}\";\n }\n\n var serialNo = 0;\n Hogan.generate = function(tree, text, options) {\n serialNo = 0;\n var context = { code: '', subs: {}, partials: {} };\n Hogan.walk(tree, context);\n\n if (options.asString) {\n return this.stringify(context, text, options);\n }\n\n return this.makeTemplate(context, text, options);\n }\n\n Hogan.wrapMain = function(code) {\n return 'var t=this;t.b(i=i||\"\");' + code + 'return t.fl();';\n }\n\n Hogan.template = Hogan.Template;\n\n Hogan.makeTemplate = function(codeObj, text, options) {\n var template = this.makePartials(codeObj);\n template.code = new Function('c', 'p', 'i', this.wrapMain(codeObj.code));\n return new this.template(template, text, this, options);\n }\n\n Hogan.makePartials = function(codeObj) {\n var key, template = {subs: {}, partials: codeObj.partials, name: codeObj.name};\n for (key in template.partials) {\n template.partials[key] = this.makePartials(template.partials[key]);\n }\n for (key in codeObj.subs) {\n template.subs[key] = new Function('c', 'p', 't', 'i', codeObj.subs[key]);\n }\n return template;\n }\n\n function esc(s) {\n return s.replace(rSlash, '\\\\\\\\')\n .replace(rQuot, '\\\\\\\"')\n .replace(rNewline, '\\\\n')\n .replace(rCr, '\\\\r')\n .replace(rLineSep, '\\\\u2028')\n .replace(rParagraphSep, '\\\\u2029');\n }\n\n function chooseMethod(s) {\n return (~s.indexOf('.')) ? 'd' : 'f';\n }\n\n function createPartial(node, context) {\n var prefix = \"<\" + (context.prefix || \"\");\n var sym = prefix + node.n + serialNo++;\n context.partials[sym] = {name: node.n, partials: {}};\n context.code += 't.b(t.rp(\"' + esc(sym) + '\",c,p,\"' + (node.indent || '') + '\"));';\n return sym;\n }\n\n Hogan.codegen = {\n '#': function(node, context) {\n context.code += 'if(t.s(t.' + chooseMethod(node.n) + '(\"' + esc(node.n) + '\",c,p,1),' +\n 'c,p,0,' + node.i + ',' + node.end + ',\"' + node.otag + \" \" + node.ctag + '\")){' +\n 't.rs(c,p,' + 'function(c,p,t){';\n Hogan.walk(node.nodes, context);\n context.code += '});c.pop();}';\n },\n\n '^': function(node, context) {\n context.code += 'if(!t.s(t.' + chooseMethod(node.n) + '(\"' + esc(node.n) + '\",c,p,1),c,p,1,0,0,\"\")){';\n Hogan.walk(node.nodes, context);\n context.code += '};';\n },\n\n '>': createPartial,\n '<': function(node, context) {\n var ctx = {partials: {}, code: '', subs: {}, inPartial: true};\n Hogan.walk(node.nodes, ctx);\n var template = context.partials[createPartial(node, context)];\n template.subs = ctx.subs;\n template.partials = ctx.partials;\n },\n\n '$': function(node, context) {\n var ctx = {subs: {}, code: '', partials: context.partials, prefix: node.n};\n Hogan.walk(node.nodes, ctx);\n context.subs[node.n] = ctx.code;\n if (!context.inPartial) {\n context.code += 't.sub(\"' + esc(node.n) + '\",c,p,i);';\n }\n },\n\n '\\n': function(node, context) {\n context.code += write('\"\\\\n\"' + (node.last ? '' : ' + i'));\n },\n\n '_v': function(node, context) {\n context.code += 't.b(t.v(t.' + chooseMethod(node.n) + '(\"' + esc(node.n) + '\",c,p,0)));';\n },\n\n '_t': function(node, context) {\n context.code += write('\"' + esc(node.text) + '\"');\n },\n\n '{': tripleStache,\n\n '&': tripleStache\n }\n\n function tripleStache(node, context) {\n context.code += 't.b(t.t(t.' + chooseMethod(node.n) + '(\"' + esc(node.n) + '\",c,p,0)));';\n }\n\n function write(s) {\n return 't.b(' + s + ');';\n }\n\n Hogan.walk = function(nodelist, context) {\n var func;\n for (var i = 0, l = nodelist.length; i < l; i++) {\n func = Hogan.codegen[nodelist[i].tag];\n func && func(nodelist[i], context);\n }\n return context;\n }\n\n Hogan.parse = function(tokens, text, options) {\n options = options || {};\n return buildTree(tokens, '', [], options.sectionTags || []);\n }\n\n Hogan.cache = {};\n\n Hogan.cacheKey = function(text, options) {\n return [text, !!options.asString, !!options.disableLambda, options.delimiters, !!options.modelGet].join('||');\n }\n\n Hogan.compile = function(text, options) {\n options = options || {};\n var key = Hogan.cacheKey(text, options);\n var template = this.cache[key];\n\n if (template) {\n var partials = template.partials;\n for (var name in partials) {\n delete partials[name].instance;\n }\n return template;\n }\n\n template = this.generate(this.parse(this.scan(text, options.delimiters), text, options), text, options);\n return this.cache[key] = template;\n }\n})(typeof exports !== 'undefined' ? exports : Hogan);\n","/*\n * Copyright 2011 Twitter, Inc.\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nvar Hogan = {};\n\n(function (Hogan) {\n Hogan.Template = function (codeObj, text, compiler, options) {\n codeObj = codeObj || {};\n this.r = codeObj.code || this.r;\n this.c = compiler;\n this.options = options || {};\n this.text = text || '';\n this.partials = codeObj.partials || {};\n this.subs = codeObj.subs || {};\n this.buf = '';\n }\n\n Hogan.Template.prototype = {\n // render: replaced by generated code.\n r: function (context, partials, indent) { return ''; },\n\n // variable escaping\n v: hoganEscape,\n\n // triple stache\n t: coerceToString,\n\n render: function render(context, partials, indent) {\n return this.ri([context], partials || {}, indent);\n },\n\n // render internal -- a hook for overrides that catches partials too\n ri: function (context, partials, indent) {\n return this.r(context, partials, indent);\n },\n\n // ensurePartial\n ep: function(symbol, partials) {\n var partial = this.partials[symbol];\n\n // check to see that if we've instantiated this partial before\n var template = partials[partial.name];\n if (partial.instance && partial.base == template) {\n return partial.instance;\n }\n\n if (typeof template == 'string') {\n if (!this.c) {\n throw new Error(\"No compiler available.\");\n }\n template = this.c.compile(template, this.options);\n }\n\n if (!template) {\n return null;\n }\n\n // We use this to check whether the partials dictionary has changed\n this.partials[symbol].base = template;\n\n if (partial.subs) {\n // Make sure we consider parent template now\n if (!partials.stackText) partials.stackText = {};\n for (key in partial.subs) {\n if (!partials.stackText[key]) {\n partials.stackText[key] = (this.activeSub !== undefined && partials.stackText[this.activeSub]) ? partials.stackText[this.activeSub] : this.text;\n }\n }\n template = createSpecializedPartial(template, partial.subs, partial.partials,\n this.stackSubs, this.stackPartials, partials.stackText);\n }\n this.partials[symbol].instance = template;\n\n return template;\n },\n\n // tries to find a partial in the current scope and render it\n rp: function(symbol, context, partials, indent) {\n var partial = this.ep(symbol, partials);\n if (!partial) {\n return '';\n }\n\n return partial.ri(context, partials, indent);\n },\n\n // render a section\n rs: function(context, partials, section) {\n var tail = context[context.length - 1];\n\n if (!isArray(tail)) {\n section(context, partials, this);\n return;\n }\n\n for (var i = 0; i < tail.length; i++) {\n context.push(tail[i]);\n section(context, partials, this);\n context.pop();\n }\n },\n\n // maybe start a section\n s: function(val, ctx, partials, inverted, start, end, tags) {\n var pass;\n\n if (isArray(val) && val.length === 0) {\n return false;\n }\n\n if (typeof val == 'function') {\n val = this.ms(val, ctx, partials, inverted, start, end, tags);\n }\n\n pass = !!val;\n\n if (!inverted && pass && ctx) {\n ctx.push((typeof val == 'object') ? val : ctx[ctx.length - 1]);\n }\n\n return pass;\n },\n\n // find values with dotted names\n d: function(key, ctx, partials, returnFound) {\n var found,\n names = key.split('.'),\n val = this.f(names[0], ctx, partials, returnFound),\n doModelGet = this.options.modelGet,\n cx = null;\n\n if (key === '.' && isArray(ctx[ctx.length - 2])) {\n val = ctx[ctx.length - 1];\n } else {\n for (var i = 1; i < names.length; i++) {\n found = findInScope(names[i], val, doModelGet);\n if (found !== undefined) {\n cx = val;\n val = found;\n } else {\n val = '';\n }\n }\n }\n\n if (returnFound && !val) {\n return false;\n }\n\n if (!returnFound && typeof val == 'function') {\n ctx.push(cx);\n val = this.mv(val, ctx, partials);\n ctx.pop();\n }\n\n return val;\n },\n\n // find values with normal names\n f: function(key, ctx, partials, returnFound) {\n var val = false,\n v = null,\n found = false,\n doModelGet = this.options.modelGet;\n\n for (var i = ctx.length - 1; i >= 0; i--) {\n v = ctx[i];\n val = findInScope(key, v, doModelGet);\n if (val !== undefined) {\n found = true;\n break;\n }\n }\n\n if (!found) {\n return (returnFound) ? false : \"\";\n }\n\n if (!returnFound && typeof val == 'function') {\n val = this.mv(val, ctx, partials);\n }\n\n return val;\n },\n\n // higher order templates\n ls: function(func, cx, partials, text, tags) {\n var oldTags = this.options.delimiters;\n\n this.options.delimiters = tags;\n this.b(this.ct(coerceToString(func.call(cx, text)), cx, partials));\n this.options.delimiters = oldTags;\n\n return false;\n },\n\n // compile text\n ct: function(text, cx, partials) {\n if (this.options.disableLambda) {\n throw new Error('Lambda features disabled.');\n }\n return this.c.compile(text, this.options).render(cx, partials);\n },\n\n // template result buffering\n b: function(s) { this.buf += s; },\n\n fl: function() { var r = this.buf; this.buf = ''; return r; },\n\n // method replace section\n ms: function(func, ctx, partials, inverted, start, end, tags) {\n var textSource,\n cx = ctx[ctx.length - 1],\n result = func.call(cx);\n\n if (typeof result == 'function') {\n if (inverted) {\n return true;\n } else {\n textSource = (this.activeSub && this.subsText && this.subsText[this.activeSub]) ? this.subsText[this.activeSub] : this.text;\n return this.ls(result, cx, partials, textSource.substring(start, end), tags);\n }\n }\n\n return result;\n },\n\n // method replace variable\n mv: function(func, ctx, partials) {\n var cx = ctx[ctx.length - 1];\n var result = func.call(cx);\n\n if (typeof result == 'function') {\n return this.ct(coerceToString(result.call(cx)), cx, partials);\n }\n\n return result;\n },\n\n sub: function(name, context, partials, indent) {\n var f = this.subs[name];\n if (f) {\n this.activeSub = name;\n f(context, partials, this, indent);\n this.activeSub = false;\n }\n }\n\n };\n\n //Find a key in an object\n function findInScope(key, scope, doModelGet) {\n var val;\n\n if (scope && typeof scope == 'object') {\n\n if (scope[key] !== undefined) {\n val = scope[key];\n\n // try lookup with get for backbone or similar model data\n } else if (doModelGet && scope.get && typeof scope.get == 'function') {\n val = scope.get(key);\n }\n }\n\n return val;\n }\n\n function createSpecializedPartial(instance, subs, partials, stackSubs, stackPartials, stackText) {\n function PartialTemplate() {};\n PartialTemplate.prototype = instance;\n function Substitutions() {};\n Substitutions.prototype = instance.subs;\n var key;\n var partial = new PartialTemplate();\n partial.subs = new Substitutions();\n partial.subsText = {}; //hehe. substext.\n partial.buf = '';\n\n stackSubs = stackSubs || {};\n partial.stackSubs = stackSubs;\n partial.subsText = stackText;\n for (key in subs) {\n if (!stackSubs[key]) stackSubs[key] = subs[key];\n }\n for (key in stackSubs) {\n partial.subs[key] = stackSubs[key];\n }\n\n stackPartials = stackPartials || {};\n partial.stackPartials = stackPartials;\n for (key in partials) {\n if (!stackPartials[key]) stackPartials[key] = partials[key];\n }\n for (key in stackPartials) {\n partial.partials[key] = stackPartials[key];\n }\n\n return partial;\n }\n\n var rAmp = /&/g,\n rLt = //g,\n rApos = /\\'/g,\n rQuot = /\\\"/g,\n hChars = /[&<>\\\"\\']/;\n\n function coerceToString(val) {\n return String((val === null || val === undefined) ? '' : val);\n }\n\n function hoganEscape(str) {\n str = coerceToString(str);\n return hChars.test(str) ?\n str\n .replace(rAmp, '&')\n .replace(rLt, '<')\n .replace(rGt, '>')\n .replace(rApos, ''')\n .replace(rQuot, '"') :\n str;\n }\n\n var isArray = Array.isArray || function(a) {\n return Object.prototype.toString.call(a) === '[object Array]';\n };\n\n})(typeof exports !== 'undefined' ? exports : Hogan);\n","/*\n * Copyright 2011 Twitter, Inc.\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n// This file is for use with Node.js. See dist/ for browser files.\n\nvar Hogan = require('./compiler');\nHogan.Template = require('./template').Template;\nHogan.template = Hogan.Template;\nmodule.exports = Hogan;\n","import type { HoganOptions, Template } from 'hogan.js';\nimport hogan from 'hogan.js';\nimport type { Templates, HoganHelpers } from '../../types';\nimport type { BindEventForHits } from './createSendEventForHits';\n\ntype TransformedHoganHelpers = {\n [helper: string]: () => (text: string) => string;\n};\n\n// We add all our template helper methods to the template as lambdas. Note\n// that lambdas in Mustache are supposed to accept a second argument of\n// `render` to get the rendered value, not the literal `{{value}}`. But\n// this is currently broken (see https://github.com/twitter/hogan.js/issues/222).\nfunction transformHelpersToHogan(\n helpers: HoganHelpers = {},\n compileOptions?: HoganOptions,\n data?: Record\n) {\n return Object.keys(helpers).reduce(\n (acc, helperKey) => ({\n ...acc,\n [helperKey]() {\n return (text) => {\n const render = (value: string) =>\n (hogan.compile(value, compileOptions) as Template).render(this);\n\n return helpers[helperKey].call(data, text, render);\n };\n },\n }),\n {}\n );\n}\n\nfunction renderTemplate({\n templates,\n templateKey,\n compileOptions,\n helpers,\n data,\n bindEvent,\n}: {\n templates: Templates;\n templateKey: string;\n compileOptions?: HoganOptions;\n helpers?: HoganHelpers;\n data?: Record;\n bindEvent?: BindEventForHits;\n}) {\n const template = templates[templateKey];\n\n if (typeof template !== 'string' && typeof template !== 'function') {\n throw new Error(\n `Template must be 'string' or 'function', was '${typeof template}' (key: ${templateKey})`\n );\n }\n\n if (typeof template === 'function') {\n return template(data, bindEvent!);\n }\n\n const transformedHelpers = transformHelpersToHogan(\n helpers,\n compileOptions,\n data\n );\n\n return (hogan.compile(template, compileOptions) as Template)\n .render({\n ...data,\n helpers: transformedHelpers,\n })\n .replace(/[ \\n\\r\\t\\f\\xA0]+/g, (spaces) =>\n spaces.replace(/(^|\\xA0+)[^\\xA0]+/g, '$1 ')\n )\n .trim();\n}\n\nexport default renderTemplate;\n","// We aren't using the native `Array.prototype.find` because the refactor away from Lodash is not\n// published as a major version.\n// Relying on the `find` polyfill on user-land, which before was only required for niche use-cases,\n// was decided as too risky.\n// @MAJOR Replace with the native `Array.prototype.find` method\n// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find\nfunction find(\n items: TItem[],\n predicate: (value: TItem, index: number, obj: TItem[]) => boolean\n): TItem | undefined {\n let value: TItem;\n for (let i = 0; i < items.length; i++) {\n value = items[i];\n // inlined for performance: if (Call(predicate, thisArg, [value, i, list])) {\n if (predicate(value, i, items)) {\n return value;\n }\n }\n\n return undefined;\n}\n\nexport default find;\n","type FacetValue = string | number | undefined;\n\nexport function unescapeFacetValue(\n value: TFacetValue\n): TFacetValue {\n if (typeof value === 'string') {\n return value.replace(/^\\\\-/, '-') as TFacetValue;\n }\n\n return value;\n}\n\nexport function escapeFacetValue(\n value: TFacetValue\n): TFacetValue {\n if ((typeof value === 'number' && value < 0) || typeof value === 'string') {\n return String(value).replace(/^-/, '\\\\-') as TFacetValue;\n }\n\n return value;\n}\n","import type { SearchParameters, SearchResults } from 'algoliasearch-helper';\nimport find from './find';\nimport { unescapeFacetValue, escapeFacetValue } from './escapeFacetValue';\n\nexport type FacetRefinement = {\n type: 'facet' | 'disjunctive' | 'hierarchical';\n attribute: string;\n name: string;\n escapedValue: string;\n count?: number;\n exhaustive?: boolean;\n};\n\nexport type TagRefinement = {\n type: 'tag';\n attribute: string;\n name: string;\n};\n\nexport type QueryRefinement = {\n type: 'query';\n attribute: 'query';\n query: string;\n name: string;\n};\n\nexport type NumericRefinement = {\n type: 'numeric';\n numericValue: number;\n operator: '<' | '<=' | '=' | '!=' | '>=' | '>';\n attribute: string;\n name: string;\n count?: number;\n exhaustive?: boolean;\n};\n\nexport type FacetExcludeRefinement = {\n type: 'exclude';\n exclude: boolean;\n attribute: string;\n name: string;\n count?: number;\n exhaustive?: boolean;\n};\n\nexport type Refinement =\n | FacetRefinement\n | QueryRefinement\n | NumericRefinement\n | FacetExcludeRefinement\n | TagRefinement;\n\nfunction getRefinement(\n state: SearchParameters,\n type: FacetRefinement['type'],\n attribute: FacetRefinement['attribute'],\n name: FacetRefinement['name'],\n resultsFacets: SearchResults['facets' | 'hierarchicalFacets'] = []\n): FacetRefinement {\n const res: FacetRefinement = {\n type,\n attribute,\n name,\n escapedValue: escapeFacetValue(name),\n };\n let facet: any = find(\n resultsFacets,\n (resultsFacet) => resultsFacet.name === attribute\n );\n let count: number;\n\n if (type === 'hierarchical') {\n const facetDeclaration = state.getHierarchicalFacetByName(attribute);\n const nameParts = name.split(facetDeclaration.separator);\n\n const getFacetRefinement =\n (facetData: any): ((refinementKey: string) => any) =>\n (refinementKey: string): any =>\n facetData[refinementKey];\n\n for (let i = 0; facet !== undefined && i < nameParts.length; ++i) {\n facet =\n facet &&\n facet.data &&\n find(\n Object.keys(facet.data).map(getFacetRefinement(facet.data)),\n (refinement) => refinement.name === nameParts[i]\n );\n }\n\n count = facet && facet.count;\n } else {\n count = facet && facet.data && facet.data[res.name];\n }\n\n if (count !== undefined) {\n res.count = count;\n }\n\n if (facet && facet.exhaustive !== undefined) {\n res.exhaustive = facet.exhaustive;\n }\n\n return res;\n}\n\nexport default function getRefinements(\n results: SearchResults | Record,\n state: SearchParameters,\n includesQuery: boolean = false\n): Refinement[] {\n const refinements: Refinement[] = [];\n const {\n facetsRefinements = {},\n facetsExcludes = {},\n disjunctiveFacetsRefinements = {},\n hierarchicalFacetsRefinements = {},\n numericRefinements = {},\n tagRefinements = [],\n } = state;\n\n Object.keys(facetsRefinements).forEach((attribute) => {\n const refinementNames = facetsRefinements[attribute];\n\n refinementNames.forEach((refinementName) => {\n refinements.push(\n getRefinement(state, 'facet', attribute, refinementName, results.facets)\n );\n });\n });\n\n Object.keys(facetsExcludes).forEach((attribute) => {\n const refinementNames = facetsExcludes[attribute];\n\n refinementNames.forEach((refinementName) => {\n refinements.push({\n type: 'exclude',\n attribute,\n name: refinementName,\n exclude: true,\n });\n });\n });\n\n Object.keys(disjunctiveFacetsRefinements).forEach((attribute) => {\n const refinementNames = disjunctiveFacetsRefinements[attribute];\n\n refinementNames.forEach((refinementName) => {\n refinements.push(\n getRefinement(\n state,\n 'disjunctive',\n attribute,\n // We unescape any disjunctive refined values with `unescapeFacetValue` because\n // they can be escaped on negative numeric values with `escapeFacetValue`.\n unescapeFacetValue(refinementName),\n results.disjunctiveFacets\n )\n );\n });\n });\n\n Object.keys(hierarchicalFacetsRefinements).forEach((attribute) => {\n const refinementNames = hierarchicalFacetsRefinements[attribute];\n\n refinementNames.forEach((refinement) => {\n refinements.push(\n getRefinement(\n state,\n 'hierarchical',\n attribute,\n refinement,\n results.hierarchicalFacets\n )\n );\n });\n });\n\n Object.keys(numericRefinements).forEach((attribute) => {\n const operators = numericRefinements[attribute];\n\n Object.keys(operators).forEach((operatorOriginal) => {\n const operator = operatorOriginal as SearchParameters.Operator;\n const valueOrValues = operators[operator];\n const refinementNames = Array.isArray(valueOrValues)\n ? valueOrValues\n : [valueOrValues];\n\n refinementNames.forEach((refinementName: any) => {\n refinements.push({\n type: 'numeric',\n attribute,\n name: `${refinementName}`,\n numericValue: refinementName,\n operator: operator as NumericRefinement['operator'],\n });\n });\n });\n });\n\n tagRefinements.forEach((refinementName) => {\n refinements.push({ type: 'tag', attribute: '_tags', name: refinementName });\n });\n\n if (includesQuery && state.query && state.query.trim()) {\n refinements.push({\n attribute: 'query',\n type: 'query',\n name: state.query,\n query: state.query,\n });\n }\n\n return refinements;\n}\n","import type { AlgoliaSearchHelper } from 'algoliasearch-helper';\n\n/**\n * Clears the refinements of a SearchParameters object based on rules provided.\n * The included attributes list is applied before the excluded attributes list. If the list\n * is not provided, this list of all the currently refined attributes is used as included attributes.\n * @param {object} $0 parameters\n * @param {Helper} $0.helper instance of the Helper\n * @param {string[]} [$0.attributesToClear = []] list of parameters to clear\n * @returns {SearchParameters} search parameters with refinements cleared\n */\nfunction clearRefinements({\n helper,\n attributesToClear = [],\n}: {\n helper: AlgoliaSearchHelper;\n attributesToClear?: string[];\n}) {\n let finalState = helper.state.setPage(0);\n\n finalState = attributesToClear.reduce((state, attribute) => {\n if (finalState.isNumericRefined(attribute)) {\n return state.removeNumericRefinement(attribute);\n }\n if (finalState.isHierarchicalFacet(attribute)) {\n return state.removeHierarchicalFacetRefinement(attribute);\n }\n if (finalState.isDisjunctiveFacet(attribute)) {\n return state.removeDisjunctiveFacetRefinement(attribute);\n }\n if (finalState.isConjunctiveFacet(attribute)) {\n return state.removeFacetRefinement(attribute);\n }\n\n return state;\n }, finalState);\n\n if (attributesToClear.indexOf('query') !== -1) {\n finalState = finalState.setQuery('');\n }\n\n return finalState;\n}\n\nexport default clearRefinements;\n","function getObjectType(object: unknown): string {\n return Object.prototype.toString.call(object).slice(8, -1);\n}\n\nexport default getObjectType;\n","import type { Renderer } from '../../types/connector';\nimport getObjectType from './getObjectType';\n\nfunction checkRendering(\n rendering: Renderer,\n usage: string\n): void {\n if (rendering === undefined || typeof rendering !== 'function') {\n throw new Error(`The render function is not valid (received type ${getObjectType(\n rendering\n )}).\n\n${usage}`);\n }\n}\n\nexport default checkRendering;\n","function noop(..._args: any[]): void {}\n\nexport default noop;\n","import noop from './noop';\n\ntype Warn = (message: string) => void;\n\ntype Warning = {\n (condition: boolean, message: string): void;\n cache: { [message: string]: boolean };\n};\n\n/**\n * Logs a warning when this function is called, in development environment only.\n */\nlet deprecate = any>(\n fn: TCallback,\n // @ts-ignore this parameter is used in the __DEV__ branch\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n message: string\n) => fn;\n\n/**\n * Logs a warning\n * This is used to log issues in development environment only.\n */\nlet warn: Warn = noop;\n\n/**\n * Logs a warning if the condition is not met.\n * This is used to log issues in development environment only.\n */\nlet warning = noop as Warning;\n\nif (__DEV__) {\n warn = (message) => {\n // eslint-disable-next-line no-console\n console.warn(`[InstantSearch.js]: ${message.trim()}`);\n };\n\n deprecate = (fn, message) => {\n let hasAlreadyPrinted = false;\n\n return function (...args) {\n if (!hasAlreadyPrinted) {\n hasAlreadyPrinted = true;\n\n warn(message);\n }\n\n return fn(...args);\n } as typeof fn;\n };\n\n warning = ((condition, message) => {\n if (condition) {\n return;\n }\n\n const hasAlreadyPrinted = warning.cache[message];\n\n if (!hasAlreadyPrinted) {\n warning.cache[message] = true;\n\n warn(message);\n }\n }) as Warning;\n\n warning.cache = {};\n}\n\nexport { warn, deprecate, warning };\n","/**\n * A typed version of Object.keys, to use when looping over a static object\n * inspired from https://stackoverflow.com/a/65117465/3185307\n */\nexport const keys = Object.keys as >(\n yourObject: TObject\n) => Array;\n","import capitalize from './capitalize';\nimport { warning } from './logger';\nimport type { IndexWidget } from '../../widgets/index/index';\nimport type { Widget, IndexUiState } from '../../types';\nimport { keys } from './typedObject';\n\n// Some connectors are responsible for multiple widgets so we need\n// to map them.\nfunction getWidgetNames(connectorName: string): string[] {\n switch (connectorName) {\n case 'range':\n return [];\n\n case 'menu':\n return ['menu', 'menuSelect'];\n\n default:\n return [connectorName];\n }\n}\n\ntype WidgetType = Required['$$type'];\n\ntype StateDescription = {\n connectors: string[];\n widgets: WidgetType[];\n};\n\ntype StateToWidgets = {\n [TParameter in keyof IndexUiState]: StateDescription;\n};\n\ntype WidgetDescription = {\n connectors: string[];\n // no longer widget type, \"ais.\" is stripped\n widgets: string[];\n};\n\ntype MissingWidgets = Array<[string, WidgetDescription]>;\n\nconst stateToWidgetsMap: StateToWidgets = {\n query: {\n connectors: ['connectSearchBox'],\n widgets: ['ais.searchBox', 'ais.autocomplete', 'ais.voiceSearch'],\n },\n refinementList: {\n connectors: ['connectRefinementList'],\n widgets: ['ais.refinementList'],\n },\n menu: {\n connectors: ['connectMenu'],\n widgets: ['ais.menu'],\n },\n hierarchicalMenu: {\n connectors: ['connectHierarchicalMenu'],\n widgets: ['ais.hierarchicalMenu'],\n },\n numericMenu: {\n connectors: ['connectNumericMenu'],\n widgets: ['ais.numericMenu'],\n },\n ratingMenu: {\n connectors: ['connectRatingMenu'],\n widgets: ['ais.ratingMenu'],\n },\n range: {\n connectors: ['connectRange'],\n widgets: ['ais.rangeInput', 'ais.rangeSlider', 'ais.range'],\n },\n toggle: {\n connectors: ['connectToggleRefinement'],\n widgets: ['ais.toggleRefinement'],\n },\n geoSearch: {\n connectors: ['connectGeoSearch'],\n widgets: ['ais.geoSearch'],\n },\n sortBy: {\n connectors: ['connectSortBy'],\n widgets: ['ais.sortBy'],\n },\n page: {\n connectors: ['connectPagination'],\n widgets: ['ais.pagination', 'ais.infiniteHits'],\n },\n hitsPerPage: {\n connectors: ['connectHitsPerPage'],\n widgets: ['ais.hitsPerPage'],\n },\n configure: {\n connectors: ['connectConfigure'],\n widgets: ['ais.configure'],\n },\n places: {\n connectors: [],\n widgets: ['ais.places'],\n },\n};\n\ntype CheckIndexUiStateParams = {\n index: IndexWidget;\n indexUiState: IndexUiState;\n};\n\nexport function checkIndexUiState({\n index,\n indexUiState,\n}: CheckIndexUiStateParams) {\n const mountedWidgets = index\n .getWidgets()\n .map((widget) => widget.$$type)\n .filter(Boolean);\n\n const missingWidgets = keys(indexUiState).reduce(\n (acc, parameter) => {\n const widgetUiState = stateToWidgetsMap[parameter];\n\n if (!widgetUiState) {\n return acc;\n }\n\n const requiredWidgets = widgetUiState.widgets;\n\n if (\n requiredWidgets &&\n !requiredWidgets.some((requiredWidget) =>\n mountedWidgets.includes(requiredWidget)\n )\n ) {\n acc.push([\n parameter,\n {\n connectors: widgetUiState.connectors,\n widgets: widgetUiState.widgets.map(\n (widgetIdentifier) => widgetIdentifier.split('ais.')[1]\n ),\n },\n ]);\n }\n\n return acc;\n },\n []\n );\n\n warning(\n missingWidgets.length === 0,\n `The UI state for the index \"${index.getIndexId()}\" is not consistent with the widgets mounted.\n\nThis can happen when the UI state is specified via \\`initialUiState\\`, \\`routing\\` or \\`setUiState\\` but that the widgets responsible for this state were not added. This results in those query parameters not being sent to the API.\n\nTo fully reflect the state, some widgets need to be added to the index \"${index.getIndexId()}\":\n\n${missingWidgets\n .map(([stateParameter, { widgets }]) => {\n return `- \\`${stateParameter}\\` needs one of these widgets: ${(\n [] as string[]\n )\n .concat(...widgets.map((name) => getWidgetNames(name)))\n .map((name: string) => `\"${name}\"`)\n .join(', ')}`;\n })\n .join('\\n')}\n\nIf you do not wish to display widgets but still want to support their search parameters, you can mount \"virtual widgets\" that don't render anything:\n\n\\`\\`\\`\n${missingWidgets\n .filter(([_stateParameter, { connectors }]) => {\n return connectors.length > 0;\n })\n .map(([_stateParameter, { connectors, widgets }]) => {\n const capitalizedWidget = capitalize(widgets[0]!);\n const connectorName = connectors[0];\n\n return `const virtual${capitalizedWidget} = ${connectorName}(() => null);`;\n })\n .join('\\n')}\n\nsearch.addWidgets([\n ${missingWidgets\n .filter(([_stateParameter, { connectors }]) => {\n return connectors.length > 0;\n })\n .map(([_stateParameter, { widgets }]) => {\n const capitalizedWidget = capitalize(widgets[0]!);\n\n return `virtual${capitalizedWidget}({ /* ... */ })`;\n })\n .join(',\\n ')}\n]);\n\\`\\`\\`\n\nIf you're using custom widgets that do set these query parameters, we recommend using connectors instead.\n\nSee https://www.algolia.com/doc/guides/building-search-ui/widgets/customize-an-existing-widget/js/#customize-the-complete-ui-of-the-widgets`\n );\n}\n","function getPropertyByPath(\n object: Record | undefined,\n path: string | string[]\n): any {\n const parts = Array.isArray(path) ? path : path.split('.');\n\n return parts.reduce((current, key) => current && current[key], object);\n}\n\nexport default getPropertyByPath;\n","// This is the `Number.isFinite()` polyfill recommended by MDN.\n// We do not provide any tests for this function.\n// See: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isFinite#Polyfill\nfunction isFiniteNumber(value: any): value is number {\n return typeof value === 'number' && isFinite(value);\n}\n\nexport default isFiniteNumber;\n","/**\n * This implementation is taken from Lodash implementation.\n * See: https://github.com/lodash/lodash/blob/master/isPlainObject.js\n */\n\nfunction getTag(value: any): string {\n if (value === null) {\n return value === undefined ? '[object Undefined]' : '[object Null]';\n }\n\n return Object.prototype.toString.call(value);\n}\n\nfunction isObjectLike(value: any): boolean {\n return typeof value === 'object' && value !== null;\n}\n\n/**\n * Checks if `value` is a plain object.\n *\n * A plain object is an object created by the `Object`\n * constructor or with a `[[Prototype]]` of `null`.\n */\nfunction isPlainObject(value: any): boolean {\n if (!isObjectLike(value) || getTag(value) !== '[object Object]') {\n return false;\n }\n\n if (Object.getPrototypeOf(value) === null) {\n return true;\n }\n\n let proto = value;\n\n while (Object.getPrototypeOf(proto) !== null) {\n proto = Object.getPrototypeOf(proto);\n }\n\n return Object.getPrototypeOf(value) === proto;\n}\n\nexport default isPlainObject;\n","type RangeOptions = {\n start?: number;\n end: number;\n step?: number;\n};\n\nfunction range({ start = 0, end, step = 1 }: RangeOptions): number[] {\n // We can't divide by 0 so we re-assign the step to 1 if it happens.\n const limitStep = step === 0 ? 1 : step;\n\n // In some cases the array to create has a decimal length.\n // We therefore need to round the value.\n // Example:\n // { start: 1, end: 5000, step: 500 }\n // => Array length = (5000 - 1) / 500 = 9.998\n const arrayLength = Math.round((end - start) / limitStep);\n\n return [...Array(arrayLength)].map(\n (_, current) => start + current * limitStep\n );\n}\n\nexport default range;\n","function isPrimitive(obj: any): boolean {\n return obj !== Object(obj);\n}\n\nfunction isEqual(first: any, second: any): boolean {\n if (first === second) {\n return true;\n }\n\n if (\n isPrimitive(first) ||\n isPrimitive(second) ||\n typeof first === 'function' ||\n typeof second === 'function'\n ) {\n return first === second;\n }\n\n if (Object.keys(first).length !== Object.keys(second).length) {\n return false;\n }\n\n for (const key of Object.keys(first)) {\n if (!(key in second)) {\n return false;\n }\n\n if (!isEqual(first[key], second[key])) {\n return false;\n }\n }\n\n return true;\n}\n\nexport default isEqual;\n","/**\n * This implementation is taken from Lodash implementation.\n * See: https://github.com/lodash/lodash/blob/4.17.11-npm/escape.js\n */\n\n// Used to map characters to HTML entities.\nconst htmlEscapes = {\n '&': '&',\n '<': '<',\n '>': '>',\n '\"': '"',\n \"'\": ''',\n};\n\n// Used to match HTML entities and HTML characters.\nconst regexUnescapedHtml = /[&<>\"']/g;\nconst regexHasUnescapedHtml = RegExp(regexUnescapedHtml.source);\n\n/**\n * Converts the characters \"&\", \"<\", \">\", '\"', and \"'\" in `string` to their\n * corresponding HTML entities.\n */\nfunction escape(value: string): string {\n return value && regexHasUnescapedHtml.test(value)\n ? value.replace(\n regexUnescapedHtml,\n (character) => htmlEscapes[character as keyof typeof htmlEscapes]\n )\n : value;\n}\n\nexport default escape;\n","/**\n * This implementation is taken from Lodash implementation.\n * See: https://github.com/lodash/lodash/blob/4.17.11-npm/unescape.js\n */\n\n// Used to map HTML entities to characters.\nconst htmlEscapes = {\n '&': '&',\n '<': '<',\n '>': '>',\n '"': '\"',\n ''': \"'\",\n};\n\n// Used to match HTML entities and HTML characters.\nconst regexEscapedHtml = /&(amp|quot|lt|gt|#39);/g;\nconst regexHasEscapedHtml = RegExp(regexEscapedHtml.source);\n\n/**\n * Converts the HTML entities \"&\", \"<\", \">\", '\"', and \"'\" in `string` to their\n * characters.\n */\nexport default function unescape(value: string): string {\n return value && regexHasEscapedHtml.test(value)\n ? value.replace(\n regexEscapedHtml,\n (character) => htmlEscapes[character as keyof typeof htmlEscapes]\n )\n : value;\n}\n","import escape from './escape';\nimport isPlainObject from './isPlainObject';\nimport type { Hit, FacetHit, EscapedHits } from '../../types';\n\nexport const TAG_PLACEHOLDER = {\n highlightPreTag: '__ais-highlight__',\n highlightPostTag: '__/ais-highlight__',\n};\n\nexport const TAG_REPLACEMENT = {\n highlightPreTag: '',\n highlightPostTag: '',\n};\n\nfunction replaceTagsAndEscape(value: string): string {\n return escape(value)\n .replace(\n new RegExp(TAG_PLACEHOLDER.highlightPreTag, 'g'),\n TAG_REPLACEMENT.highlightPreTag\n )\n .replace(\n new RegExp(TAG_PLACEHOLDER.highlightPostTag, 'g'),\n TAG_REPLACEMENT.highlightPostTag\n );\n}\n\nfunction recursiveEscape(input: any): any {\n if (isPlainObject(input) && typeof input.value !== 'string') {\n return Object.keys(input).reduce(\n (acc, key) => ({\n ...acc,\n [key]: recursiveEscape(input[key]),\n }),\n {}\n );\n }\n\n if (Array.isArray(input)) {\n return input.map(recursiveEscape);\n }\n\n return {\n ...input,\n value: replaceTagsAndEscape(input.value),\n };\n}\n\nexport function escapeHits(\n hits: THit[] | EscapedHits\n): EscapedHits {\n if ((hits as any).__escaped === undefined) {\n // We don't override the value on hit because it will mutate the raw results\n // instead we make a shallow copy and we assign the escaped values on it.\n hits = hits.map(({ ...hit }) => {\n if (hit._highlightResult) {\n hit._highlightResult = recursiveEscape(hit._highlightResult);\n }\n\n if (hit._snippetResult) {\n hit._snippetResult = recursiveEscape(hit._snippetResult);\n }\n\n return hit;\n });\n\n (hits as any).__escaped = true;\n }\n\n return hits as unknown as EscapedHits;\n}\n\nexport function escapeFacets(facetHits: FacetHit[]): FacetHit[] {\n return facetHits.map((h) => ({\n ...h,\n highlighted: replaceTagsAndEscape(h.highlighted),\n }));\n}\n","import type { HighlightedParts } from '../../types';\nimport { TAG_REPLACEMENT } from './escape-highlight';\n\nexport default function concatHighlightedParts(parts: HighlightedParts[]) {\n const { highlightPreTag, highlightPostTag } = TAG_REPLACEMENT;\n\n return parts\n .map((part) =>\n part.isHighlighted\n ? highlightPreTag + part.value + highlightPostTag\n : part.value\n )\n .join('');\n}\n","import { TAG_REPLACEMENT } from './escape-highlight';\n\nexport default function getHighlightedParts(highlightedValue: string) {\n const { highlightPostTag, highlightPreTag } = TAG_REPLACEMENT;\n\n const splitByPreTag = highlightedValue.split(highlightPreTag);\n const firstValue = splitByPreTag.shift();\n const elements = !firstValue\n ? []\n : [{ value: firstValue, isHighlighted: false }];\n\n splitByPreTag.forEach((split) => {\n const splitByPostTag = split.split(highlightPostTag);\n\n elements.push({\n value: splitByPostTag[0],\n isHighlighted: true,\n });\n\n if (splitByPostTag[1] !== '') {\n elements.push({\n value: splitByPostTag[1],\n isHighlighted: false,\n });\n }\n });\n\n return elements;\n}\n","import unescape from './unescape';\nimport type { HighlightedParts } from '../../types';\n\nconst hasAlphanumeric = new RegExp(/\\w/i);\n\nexport default function getHighlightFromSiblings(\n parts: HighlightedParts[],\n i: number\n) {\n const current = parts[i];\n const isNextHighlighted = parts[i + 1]?.isHighlighted || true;\n const isPreviousHighlighted = parts[i - 1]?.isHighlighted || true;\n\n if (\n !hasAlphanumeric.test(unescape(current.value)) &&\n isPreviousHighlighted === isNextHighlighted\n ) {\n return isPreviousHighlighted;\n }\n\n return current.isHighlighted;\n}\n","import type { HighlightedParts } from '../../types';\nimport getHighlightFromSiblings from './getHighlightFromSiblings';\n\nexport default function reverseHighlightedParts(parts: HighlightedParts[]) {\n if (!parts.some((part) => part.isHighlighted)) {\n return parts.map((part) => ({ ...part, isHighlighted: false }));\n }\n\n return parts.map((part, i) => ({\n ...part,\n isHighlighted: !getHighlightFromSiblings(parts, i),\n }));\n}\n","// We aren't using the native `Array.prototype.findIndex` because the refactor away from Lodash is not\n// published as a major version.\n// Relying on the `findIndex` polyfill on user-land, which before was only required for niche use-cases,\n// was decided as too risky.\n// @MAJOR Replace with the native `Array.prototype.findIndex` method\n// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/findIndex\nfunction findIndex(\n array: TItem[],\n comparator: (value: TItem) => boolean\n): number {\n if (!Array.isArray(array)) {\n return -1;\n }\n\n for (let i = 0; i < array.length; i++) {\n if (comparator(array[i])) {\n return i;\n }\n }\n return -1;\n}\n\nexport default findIndex;\n","import type { SearchParameters } from 'algoliasearch-helper';\nimport findIndex from './findIndex';\nimport uniq from './uniq';\n\ntype Merger = (\n left: SearchParameters,\n right: SearchParameters\n) => SearchParameters;\n\nconst mergeWithRest: Merger = (left, right) => {\n const {\n facets,\n disjunctiveFacets,\n facetsRefinements,\n facetsExcludes,\n disjunctiveFacetsRefinements,\n numericRefinements,\n tagRefinements,\n hierarchicalFacets,\n hierarchicalFacetsRefinements,\n ruleContexts,\n ...rest\n } = right;\n\n return left.setQueryParameters(rest);\n};\n\n// Merge facets\nconst mergeFacets: Merger = (left, right) =>\n right.facets!.reduce((_, name) => _.addFacet(name), left);\n\nconst mergeDisjunctiveFacets: Merger = (left, right) =>\n right.disjunctiveFacets.reduce(\n (_, name) => _.addDisjunctiveFacet(name),\n left\n );\n\nconst mergeHierarchicalFacets: Merger = (left, right) =>\n left.setQueryParameters({\n hierarchicalFacets: right.hierarchicalFacets.reduce((facets, facet) => {\n const index = findIndex(facets, (_) => _.name === facet.name);\n\n if (index === -1) {\n return facets.concat(facet);\n }\n\n const nextFacets = facets.slice();\n nextFacets.splice(index, 1, facet);\n\n return nextFacets;\n }, left.hierarchicalFacets),\n });\n\n// Merge facet refinements\nconst mergeTagRefinements: Merger = (left, right) =>\n right.tagRefinements.reduce((_, value) => _.addTagRefinement(value), left);\n\nconst mergeFacetRefinements: Merger = (left, right) =>\n left.setQueryParameters({\n facetsRefinements: {\n ...left.facetsRefinements,\n ...right.facetsRefinements,\n },\n });\n\nconst mergeFacetsExcludes: Merger = (left, right) =>\n left.setQueryParameters({\n facetsExcludes: {\n ...left.facetsExcludes,\n ...right.facetsExcludes,\n },\n });\n\nconst mergeDisjunctiveFacetsRefinements: Merger = (left, right) =>\n left.setQueryParameters({\n disjunctiveFacetsRefinements: {\n ...left.disjunctiveFacetsRefinements,\n ...right.disjunctiveFacetsRefinements,\n },\n });\n\nconst mergeNumericRefinements: Merger = (left, right) =>\n left.setQueryParameters({\n numericRefinements: {\n ...left.numericRefinements,\n ...right.numericRefinements,\n },\n });\n\nconst mergeHierarchicalFacetsRefinements: Merger = (left, right) =>\n left.setQueryParameters({\n hierarchicalFacetsRefinements: {\n ...left.hierarchicalFacetsRefinements,\n ...right.hierarchicalFacetsRefinements,\n },\n });\n\nconst mergeRuleContexts: Merger = (left, right) => {\n const ruleContexts: string[] = uniq(\n ([] as any)\n .concat(left.ruleContexts)\n .concat(right.ruleContexts)\n .filter(Boolean)\n );\n\n if (ruleContexts.length > 0) {\n return left.setQueryParameters({\n ruleContexts,\n });\n }\n\n return left;\n};\n\nconst merge = (...parameters: SearchParameters[]): SearchParameters =>\n parameters.reduce((left, right) => {\n const hierarchicalFacetsRefinementsMerged =\n mergeHierarchicalFacetsRefinements(left, right);\n const hierarchicalFacetsMerged = mergeHierarchicalFacets(\n hierarchicalFacetsRefinementsMerged,\n right\n );\n const tagRefinementsMerged = mergeTagRefinements(\n hierarchicalFacetsMerged,\n right\n );\n const numericRefinementsMerged = mergeNumericRefinements(\n tagRefinementsMerged,\n right\n );\n const disjunctiveFacetsRefinementsMerged =\n mergeDisjunctiveFacetsRefinements(numericRefinementsMerged, right);\n const facetsExcludesMerged = mergeFacetsExcludes(\n disjunctiveFacetsRefinementsMerged,\n right\n );\n const facetRefinementsMerged = mergeFacetRefinements(\n facetsExcludesMerged,\n right\n );\n const disjunctiveFacetsMerged = mergeDisjunctiveFacets(\n facetRefinementsMerged,\n right\n );\n const ruleContextsMerged = mergeRuleContexts(\n disjunctiveFacetsMerged,\n right\n );\n const facetsMerged = mergeFacets(ruleContextsMerged, right);\n\n return mergeWithRest(facetsMerged, right);\n });\n\nexport default merge;\n","import type { SearchParameters } from 'algoliasearch-helper';\nimport type { IndexWidget } from '../../widgets/index/index';\n\nconst resolveSearchParameters = (current: IndexWidget): SearchParameters[] => {\n let parent = current.getParent();\n let states = [current.getHelper()!.state];\n\n while (parent !== null) {\n states = [parent.getHelper()!.state].concat(states);\n parent = parent.getParent();\n }\n\n return states;\n};\n\nexport default resolveSearchParameters;\n","function toArray(value: any) {\n return Array.isArray(value) ? value : [value];\n}\n\nexport default toArray;\n","type WidgetParam = {\n name: string;\n connector?: boolean;\n};\n\nexport const createDocumentationLink = ({\n name,\n connector = false,\n}: WidgetParam): string => {\n return [\n 'https://www.algolia.com/doc/api-reference/widgets/',\n name,\n '/js/',\n connector ? '#connector' : '',\n ].join('');\n};\n\ntype DocumentationMessageGenerator = (message?: string) => string;\n\nexport const createDocumentationMessageGenerator = (\n ...widgets: WidgetParam[]\n): DocumentationMessageGenerator => {\n const links = widgets\n .map((widget) => createDocumentationLink(widget))\n .join(', ');\n\n return (message?: string) =>\n [message, `See documentation: ${links}`].filter(Boolean).join('\\n\\n');\n};\n","const latLngRegExp = /^(-?\\d+(?:\\.\\d+)?),\\s*(-?\\d+(?:\\.\\d+)?)$/;\n\nexport function aroundLatLngToPosition(value: string) {\n const pattern = value.match(latLngRegExp);\n\n // Since the value provided is the one send with the request, the API should\n // throw an error due to the wrong format. So throw an error should be safe.\n if (!pattern) {\n throw new Error(`Invalid value for \"aroundLatLng\" parameter: \"${value}\"`);\n }\n\n return {\n lat: parseFloat(pattern[1]),\n lng: parseFloat(pattern[2]),\n };\n}\n\nexport type LatLng = Array<[number, number, number, number]>;\n\nfunction insideBoundingBoxArrayToBoundingBox(value: LatLng) {\n const [\n [neLat, neLng, swLat, swLng] = [undefined, undefined, undefined, undefined],\n ] = value;\n\n // Since the value provided is the one send with the request, the API should\n // throw an error due to the wrong format. So throw an error should be safe.\n if (!neLat || !neLng || !swLat || !swLng) {\n throw new Error(\n `Invalid value for \"insideBoundingBox\" parameter: [${value}]`\n );\n }\n\n return {\n northEast: {\n lat: neLat,\n lng: neLng,\n },\n southWest: {\n lat: swLat,\n lng: swLng,\n },\n };\n}\n\nfunction insideBoundingBoxStringToBoundingBox(value: string) {\n const [neLat, neLng, swLat, swLng] = value.split(',').map(parseFloat);\n\n // Since the value provided is the one send with the request, the API should\n // throw an error due to the wrong format. So throw an error should be safe.\n if (!neLat || !neLng || !swLat || !swLng) {\n throw new Error(\n `Invalid value for \"insideBoundingBox\" parameter: \"${value}\"`\n );\n }\n\n return {\n northEast: {\n lat: neLat,\n lng: neLng,\n },\n southWest: {\n lat: swLat,\n lng: swLng,\n },\n };\n}\n\nexport function insideBoundingBoxToBoundingBox(value: string | LatLng) {\n if (Array.isArray(value)) {\n return insideBoundingBoxArrayToBoundingBox(value);\n }\n\n return insideBoundingBoxStringToBoundingBox(value);\n}\n","import type { AlgoliaHit } from '../../types';\n\nexport function addAbsolutePosition(\n hits: THit[],\n page: number,\n hitsPerPage: number\n): Array {\n return hits.map((hit, idx) => ({\n ...hit,\n __position: hitsPerPage * page + idx + 1,\n }));\n}\n","import type { AlgoliaHit } from '../../types';\n\nexport function addQueryID(\n hits: THit[],\n queryID?: string\n): Array {\n if (!queryID) {\n return hits;\n }\n return hits.map((hit) => ({\n ...hit,\n __queryID: queryID,\n }));\n}\n","import type { AlgoliaSearchHelper } from 'algoliasearch-helper';\n\nexport default function isFacetRefined(\n helper: AlgoliaSearchHelper,\n facet: string,\n value: string\n) {\n if (helper.state.isHierarchicalFacet(facet)) {\n return helper.state.isHierarchicalFacetRefined(facet, value);\n } else if (helper.state.isConjunctiveFacet(facet)) {\n return helper.state.isFacetRefined(facet, value);\n } else {\n return helper.state.isDisjunctiveFacetRefined(facet, value);\n }\n}\n","import type { AlgoliaSearchHelper } from 'algoliasearch-helper';\nimport type { InstantSearch } from '../../types';\nimport isFacetRefined from './isFacetRefined';\n\ntype BuiltInSendEventForFacet = (\n eventType: string,\n facetValue: string,\n eventName?: string\n) => void;\ntype CustomSendEventForFacet = (customPayload: any) => void;\n\nexport type SendEventForFacet = BuiltInSendEventForFacet &\n CustomSendEventForFacet;\n\nexport function createSendEventForFacet({\n instantSearchInstance,\n helper,\n attribute,\n widgetType,\n}: {\n instantSearchInstance: InstantSearch;\n helper: AlgoliaSearchHelper;\n attribute: string;\n widgetType: string;\n}): SendEventForFacet {\n const sendEventForFacet: SendEventForFacet = (...args: any[]) => {\n const [eventType, facetValue, eventName = 'Filter Applied'] = args;\n if (args.length === 1 && typeof args[0] === 'object') {\n instantSearchInstance.sendEventToInsights(args[0]);\n } else if (\n eventType === 'click' &&\n (args.length === 2 || args.length === 3)\n ) {\n if (!isFacetRefined(helper, attribute, facetValue)) {\n // send event only when the facet is being checked \"ON\"\n instantSearchInstance.sendEventToInsights({\n insightsMethod: 'clickedFilters',\n widgetType,\n eventType,\n payload: {\n eventName,\n index: helper.getIndex(),\n filters: [`${attribute}:${facetValue}`],\n },\n attribute,\n });\n }\n } else if (__DEV__) {\n throw new Error(\n `You need to pass two arguments like:\n sendEvent('click', facetValue);\n\nIf you want to send a custom payload, you can pass one object: sendEvent(customPayload);\n`\n );\n }\n };\n return sendEventForFacet;\n}\n","export function serializePayload(payload: TPayload): string {\n return btoa(encodeURIComponent(JSON.stringify(payload)));\n}\n\nexport function deserializePayload(serialized: string): TPayload {\n return JSON.parse(decodeURIComponent(atob(serialized)));\n}\n","/**\n * @jest-environment jsdom\n */\n\nimport type { InstantSearch, Hit, Hits, EscapedHits } from '../../types';\nimport { serializePayload } from '../../lib/utils/serializer';\nimport type { InsightsEvent } from '../../middlewares/createInsightsMiddleware';\n\ntype BuiltInSendEventForHits = (\n eventType: string,\n hits: Hit | Hits,\n eventName?: string\n) => void;\ntype CustomSendEventForHits = (customPayload: any) => void;\nexport type SendEventForHits = BuiltInSendEventForHits & CustomSendEventForHits;\n\ntype BuiltInBindEventForHits = (\n eventType: string,\n hits: Hit | Hits,\n eventName?: string\n) => string;\ntype CustomBindEventForHits = (customPayload: any) => string;\nexport type BindEventForHits = BuiltInBindEventForHits & CustomBindEventForHits;\n\nfunction chunk(arr: TItem[], chunkSize: number = 20): TItem[][] {\n const chunks: TItem[][] = [];\n for (let i = 0; i < Math.ceil(arr.length / chunkSize); i++) {\n chunks.push(arr.slice(i * chunkSize, (i + 1) * chunkSize));\n }\n return chunks;\n}\n\nconst buildPayloads = ({\n index,\n widgetType,\n methodName,\n args,\n}: {\n widgetType: string;\n index: string;\n methodName: 'sendEvent' | 'bindEvent';\n args: any[];\n}): InsightsEvent[] => {\n // when there's only one argument, that means it's custom\n if (args.length === 1 && typeof args[0] === 'object') {\n return [args[0]];\n }\n const eventType: string = args[0];\n const hits: Hit | Hits | EscapedHits = args[1];\n const eventName: string | undefined = args[2];\n if (!hits) {\n if (__DEV__) {\n throw new Error(\n `You need to pass hit or hits as the second argument like:\n ${methodName}(eventType, hit);\n `\n );\n } else {\n return [];\n }\n }\n if ((eventType === 'click' || eventType === 'conversion') && !eventName) {\n if (__DEV__) {\n throw new Error(\n `You need to pass eventName as the third argument for 'click' or 'conversion' events like:\n ${methodName}('click', hit, 'Product Purchased');\n\n To learn more about event naming: https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/in-depth/clicks-conversions-best-practices/\n `\n );\n } else {\n return [];\n }\n }\n const hitsArray: Hits = Array.isArray(hits)\n ? removeEscapedFromHits(hits)\n : [hits];\n\n if (hitsArray.length === 0) {\n return [];\n }\n const queryID = hitsArray[0].__queryID;\n const hitsChunks = chunk(hitsArray);\n const objectIDsByChunk = hitsChunks.map((batch) =>\n batch.map((hit) => hit.objectID)\n );\n const positionsByChunk = hitsChunks.map((batch) =>\n batch.map((hit) => hit.__position)\n );\n\n if (eventType === 'view') {\n return hitsChunks.map((batch, i) => {\n return {\n insightsMethod: 'viewedObjectIDs',\n widgetType,\n eventType,\n payload: {\n eventName: eventName || 'Hits Viewed',\n index,\n objectIDs: objectIDsByChunk[i],\n },\n hits: batch,\n };\n });\n } else if (eventType === 'click') {\n return hitsChunks.map((batch, i) => {\n return {\n insightsMethod: 'clickedObjectIDsAfterSearch',\n widgetType,\n eventType,\n payload: {\n eventName,\n index,\n queryID,\n objectIDs: objectIDsByChunk[i],\n positions: positionsByChunk[i],\n },\n hits: batch,\n };\n });\n } else if (eventType === 'conversion') {\n return hitsChunks.map((batch, i) => {\n return {\n insightsMethod: 'convertedObjectIDsAfterSearch',\n widgetType,\n eventType,\n payload: {\n eventName,\n index,\n queryID,\n objectIDs: objectIDsByChunk[i],\n },\n hits: batch,\n };\n });\n } else if (__DEV__) {\n throw new Error(`eventType(\"${eventType}\") is not supported.\n If you want to send a custom payload, you can pass one object: ${methodName}(customPayload);\n `);\n } else {\n return [];\n }\n};\n\nfunction removeEscapedFromHits(hits: Hits | EscapedHits): Hits {\n // remove `hits.__escaped` without mutating\n return hits.slice();\n}\n\nexport function createSendEventForHits({\n instantSearchInstance,\n index,\n widgetType,\n}: {\n instantSearchInstance: InstantSearch;\n index: string;\n widgetType: string;\n}): SendEventForHits {\n const sendEventForHits: SendEventForHits = (...args: any[]) => {\n const payloads = buildPayloads({\n widgetType,\n index,\n methodName: 'sendEvent',\n args,\n });\n\n payloads.forEach((payload) =>\n instantSearchInstance.sendEventToInsights(payload)\n );\n };\n return sendEventForHits;\n}\n\nexport function createBindEventForHits({\n index,\n widgetType,\n}: {\n index: string;\n widgetType: string;\n}): BindEventForHits {\n const bindEventForHits: BindEventForHits = (...args: any[]) => {\n const payloads = buildPayloads({\n widgetType,\n index,\n methodName: 'bindEvent',\n args,\n });\n\n return payloads.length\n ? `data-insights-event=${serializePayload(payloads)}`\n : '';\n };\n return bindEventForHits;\n}\n","// typed as any, since it accepts the _real_ js clients, not the interface we otherwise expect\nexport function getAppIdAndApiKey(searchClient: any) {\n if (searchClient.transporter) {\n // searchClient v4\n const { headers, queryParameters } = searchClient.transporter;\n const APP_ID = 'x-algolia-application-id';\n const API_KEY = 'x-algolia-api-key';\n const appId = headers[APP_ID] || queryParameters[APP_ID];\n const apiKey = headers[API_KEY] || queryParameters[API_KEY];\n return [appId, apiKey];\n } else {\n // searchClient v3\n return [searchClient.applicationID, searchClient.apiKey];\n }\n}\n","import type { SearchParameters } from 'algoliasearch-helper';\n\nexport function convertNumericRefinementsToFilters(\n state: SearchParameters | null,\n attribute: string\n) {\n if (!state) {\n return null;\n }\n const filtersObj = state.numericRefinements[attribute];\n /*\n filtersObj === {\n \"<=\": [10],\n \"=\": [],\n \">=\": [5]\n }\n */\n const filters: string[] = [];\n (Object.keys(filtersObj) as SearchParameters.Operator[])\n .filter(\n (operator) =>\n Array.isArray(filtersObj[operator]) && filtersObj[operator]!.length > 0\n )\n .forEach((operator) => {\n filtersObj[operator]!.forEach((value) => {\n filters.push(`${attribute}${operator}${value}`);\n });\n });\n return filters;\n}\n","export type MaybePromise =\n | Readonly>\n | Promise\n | TResolution;\n\n// copied from\n// https://github.com/algolia/autocomplete.js/blob/307a7acc4283e10a19cb7d067f04f1bea79dc56f/packages/autocomplete-core/src/utils/createConcurrentSafePromise.ts#L1:L1\n/**\n * Creates a runner that executes promises in a concurrent-safe way.\n *\n * This is useful to prevent older promises to resolve after a newer promise,\n * otherwise resulting in stale resolved values.\n */\nexport function createConcurrentSafePromise() {\n let basePromiseId = -1;\n let latestResolvedId = -1;\n let latestResolvedValue: TValue | undefined = undefined;\n\n return function runConcurrentSafePromise(promise: MaybePromise) {\n const currentPromiseId = ++basePromiseId;\n\n return Promise.resolve(promise).then((x) => {\n // The promise might take too long to resolve and get outdated. This would\n // result in resolving stale values.\n // When this happens, we ignore the promise value and return the one\n // coming from the latest resolved value.\n //\n // +----------------------------------+\n // | 100ms |\n // | run(1) +---> R1 |\n // | 300ms |\n // | run(2) +-------------> R2 (SKIP) |\n // | 200ms |\n // | run(3) +--------> R3 |\n // +----------------------------------+\n if (latestResolvedValue && currentPromiseId < latestResolvedId) {\n return latestResolvedValue;\n }\n\n latestResolvedId = currentPromiseId;\n latestResolvedValue = x;\n\n return x;\n });\n };\n}\n","import type { Awaited } from '../../types';\n\ntype Func = (...args: any[]) => any;\n\nexport type DebouncedFunction = (\n this: ThisParameterType,\n ...args: Parameters\n) => Promise>>;\n\n// Debounce a function call to the trailing edge.\n// The debounced function returns a promise.\nexport function debounce(\n func: TFunction,\n wait: number\n): DebouncedFunction {\n let lastTimeout: ReturnType | null = null;\n return function (...args) {\n return new Promise((resolve, reject) => {\n if (lastTimeout) {\n clearTimeout(lastTimeout);\n }\n lastTimeout = setTimeout(() => {\n lastTimeout = null;\n Promise.resolve(func(...args))\n .then(resolve)\n .catch(reject);\n }, wait);\n });\n };\n}\n","import type { InitOptions, Widget } from '../../types';\nimport type { IndexWidget } from '../../widgets/index/index';\n\nexport function getWidgetAttribute(\n widget: Widget | IndexWidget,\n initOptions: InitOptions\n): string {\n const renderState = widget.getWidgetRenderState?.(initOptions);\n\n let attribute = null;\n\n if (renderState && renderState.widgetParams) {\n // casting as widgetParams is checked just before\n const widgetParams = renderState.widgetParams as Record;\n\n if (widgetParams.attribute) {\n attribute = widgetParams.attribute;\n } else if (Array.isArray(widgetParams.attributes)) {\n attribute = widgetParams.attributes[0];\n }\n }\n\n if (typeof attribute !== 'string') {\n throw new Error(`Could not find the attribute of the widget:\n\n${JSON.stringify(widget)}\n\nPlease check whether the widget's getWidgetRenderState returns widgetParams.attribute correctly.`);\n }\n\n return attribute;\n}\n","// eslint-disable-next-line no-restricted-globals\ntype BrowserCallback = (params: { window: typeof window }) => TReturn;\ntype SafelyRunOnBrowserOptions = {\n /**\n * Fallback to run on server environments.\n */\n fallback: () => TReturn;\n};\n\n/**\n * Runs code on browser enviromnents safely.\n */\nexport function safelyRunOnBrowser(\n callback: BrowserCallback,\n { fallback }: SafelyRunOnBrowserOptions = {\n fallback: () => undefined as any,\n }\n): TReturn {\n // eslint-disable-next-line no-restricted-globals\n if (typeof window === 'undefined') {\n return fallback();\n }\n\n // eslint-disable-next-line no-restricted-globals\n return callback({ window });\n}\n","import type {\n AlgoliaSearchHelper as Helper,\n DerivedHelper,\n PlainSearchParameters,\n SearchParameters,\n SearchResults,\n AlgoliaSearchHelper,\n} from 'algoliasearch-helper';\nimport algoliasearchHelper from 'algoliasearch-helper';\nimport type {\n InstantSearch,\n UiState,\n IndexUiState,\n Widget,\n InitOptions,\n RenderOptions,\n ScopedResult,\n SearchClient,\n IndexRenderState,\n} from '../../types';\nimport {\n checkIndexUiState,\n createDocumentationMessageGenerator,\n resolveSearchParameters,\n mergeSearchParameters,\n warning,\n} from '../../lib/utils';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'index-widget',\n});\n\nexport type IndexWidgetParams = {\n indexName: string;\n indexId?: string;\n};\n\ntype IndexInitOptions = Pick<\n InitOptions,\n 'instantSearchInstance' | 'parent' | 'uiState'\n>;\n\ntype IndexRenderOptions = Pick;\n\ntype WidgetSearchParametersOptions = Parameters<\n NonNullable\n>[1];\ntype LocalWidgetSearchParametersOptions = WidgetSearchParametersOptions & {\n initialSearchParameters: SearchParameters;\n};\n\nexport type IndexWidgetDescription = {\n $$type: 'ais.index';\n $$widgetType: 'ais.index';\n};\n\nexport type IndexWidget = Omit<\n Widget,\n 'getWidgetUiState' | 'getWidgetState'\n> & {\n getIndexName(): string;\n getIndexId(): string;\n getHelper(): Helper | null;\n getResults(): SearchResults | null;\n getScopedResults(): ScopedResult[];\n getParent(): IndexWidget | null;\n getWidgets(): Array;\n createURL(state: SearchParameters): string;\n\n addWidgets(widgets: Array): IndexWidget;\n removeWidgets(widgets: Array): IndexWidget;\n\n init(options: IndexInitOptions): void;\n render(options: IndexRenderOptions): void;\n dispose(): void;\n /**\n * @deprecated\n */\n getWidgetState(uiState: UiState): UiState;\n getWidgetUiState(uiState: TUiState): TUiState;\n getWidgetSearchParameters(\n searchParameters: SearchParameters,\n searchParametersOptions: { uiState: IndexUiState }\n ): SearchParameters;\n refreshUiState(): void;\n};\n\nexport function isIndexWidget(\n widget: Widget | IndexWidget\n): widget is IndexWidget {\n return widget.$$type === 'ais.index';\n}\n\n/**\n * This is the same content as helper._change / setState, but allowing for extra\n * UiState to be synchronized.\n * see: https://github.com/algolia/algoliasearch-helper-js/blob/6b835ffd07742f2d6b314022cce6848f5cfecd4a/src/algoliasearch.helper.js#L1311-L1324\n */\nfunction privateHelperSetState(\n helper: AlgoliaSearchHelper,\n {\n state,\n isPageReset,\n _uiState,\n }: {\n state: SearchParameters;\n isPageReset?: boolean;\n _uiState?: IndexUiState;\n }\n) {\n if (state !== helper.state) {\n helper.state = state;\n\n helper.emit('change', {\n state: helper.state,\n results: helper.lastResults,\n isPageReset,\n _uiState,\n });\n }\n}\n\ntype WidgetUiStateOptions = Parameters<\n NonNullable\n>[1];\n\nfunction getLocalWidgetsUiState(\n widgets: Array,\n widgetStateOptions: WidgetUiStateOptions,\n initialUiState: IndexUiState = {}\n) {\n return widgets.reduce((uiState, widget) => {\n if (isIndexWidget(widget)) {\n return uiState;\n }\n\n if (!widget.getWidgetUiState && !widget.getWidgetState) {\n return uiState;\n }\n\n if (widget.getWidgetUiState) {\n return widget.getWidgetUiState(uiState, widgetStateOptions);\n }\n\n return widget.getWidgetState!(uiState, widgetStateOptions);\n }, initialUiState);\n}\n\nfunction getLocalWidgetsSearchParameters(\n widgets: Array,\n widgetSearchParametersOptions: LocalWidgetSearchParametersOptions\n): SearchParameters {\n const { initialSearchParameters, ...rest } = widgetSearchParametersOptions;\n\n return widgets\n .filter((widget) => !isIndexWidget(widget))\n .reduce((state, widget) => {\n if (!widget.getWidgetSearchParameters) {\n return state;\n }\n\n return widget.getWidgetSearchParameters(state, rest);\n }, initialSearchParameters);\n}\n\nfunction resetPageFromWidgets(widgets: Array): void {\n const indexWidgets = widgets.filter(isIndexWidget);\n\n if (indexWidgets.length === 0) {\n return;\n }\n\n indexWidgets.forEach((widget) => {\n const widgetHelper = widget.getHelper()!;\n\n privateHelperSetState(widgetHelper, {\n state: widgetHelper.state.resetPage(),\n isPageReset: true,\n });\n\n resetPageFromWidgets(widget.getWidgets());\n });\n}\n\nfunction resolveScopedResultsFromWidgets(\n widgets: Array\n): ScopedResult[] {\n const indexWidgets = widgets.filter(isIndexWidget);\n\n return indexWidgets.reduce((scopedResults, current) => {\n return scopedResults.concat(\n {\n indexId: current.getIndexId(),\n results: current.getResults()!,\n helper: current.getHelper()!,\n },\n ...resolveScopedResultsFromWidgets(current.getWidgets())\n );\n }, []);\n}\n\nconst index = (widgetParams: IndexWidgetParams): IndexWidget => {\n if (widgetParams === undefined || widgetParams.indexName === undefined) {\n throw new Error(withUsage('The `indexName` option is required.'));\n }\n\n const { indexName, indexId = indexName } = widgetParams;\n\n let localWidgets: Array = [];\n let localUiState: IndexUiState = {};\n let localInstantSearchInstance: InstantSearch | null = null;\n let localParent: IndexWidget | null = null;\n let helper: Helper | null = null;\n let derivedHelper: DerivedHelper | null = null;\n\n return {\n $$type: 'ais.index',\n $$widgetType: 'ais.index',\n\n getIndexName() {\n return indexName;\n },\n\n getIndexId() {\n return indexId;\n },\n\n getHelper() {\n return helper;\n },\n\n getResults() {\n return derivedHelper && derivedHelper.lastResults;\n },\n\n getScopedResults() {\n const widgetParent = this.getParent();\n\n // If the widget is the root, we consider itself as the only sibling.\n const widgetSiblings = widgetParent ? widgetParent.getWidgets() : [this];\n\n return resolveScopedResultsFromWidgets(widgetSiblings);\n },\n\n getParent() {\n return localParent;\n },\n\n createURL(nextState: SearchParameters) {\n return localInstantSearchInstance!._createURL({\n [indexId]: getLocalWidgetsUiState(localWidgets, {\n searchParameters: nextState,\n helper: helper!,\n }),\n });\n },\n\n getWidgets() {\n return localWidgets;\n },\n\n addWidgets(widgets) {\n if (!Array.isArray(widgets)) {\n throw new Error(\n withUsage('The `addWidgets` method expects an array of widgets.')\n );\n }\n\n if (\n widgets.some(\n (widget) =>\n typeof widget.init !== 'function' &&\n typeof widget.render !== 'function'\n )\n ) {\n throw new Error(\n withUsage(\n 'The widget definition expects a `render` and/or an `init` method.'\n )\n );\n }\n\n localWidgets = localWidgets.concat(widgets);\n\n if (localInstantSearchInstance && Boolean(widgets.length)) {\n privateHelperSetState(helper!, {\n state: getLocalWidgetsSearchParameters(localWidgets, {\n uiState: localUiState,\n initialSearchParameters: helper!.state,\n }),\n _uiState: localUiState,\n });\n\n // We compute the render state before calling `init` in a separate loop\n // to construct the whole render state object that is then passed to\n // `init`.\n widgets.forEach((widget) => {\n if (widget.getRenderState) {\n const renderState = widget.getRenderState(\n localInstantSearchInstance!.renderState[this.getIndexId()] || {},\n {\n uiState: localInstantSearchInstance!._initialUiState,\n helper: this.getHelper()!,\n parent: this,\n instantSearchInstance: localInstantSearchInstance!,\n state: helper!.state,\n renderState: localInstantSearchInstance!.renderState,\n templatesConfig: localInstantSearchInstance!.templatesConfig,\n createURL: this.createURL,\n scopedResults: [],\n searchMetadata: {\n isSearchStalled: localInstantSearchInstance!._isSearchStalled,\n },\n }\n );\n\n storeRenderState({\n renderState,\n instantSearchInstance: localInstantSearchInstance!,\n parent: this,\n });\n }\n });\n\n widgets.forEach((widget) => {\n if (widget.init) {\n widget.init({\n helper: helper!,\n parent: this,\n uiState: localInstantSearchInstance!._initialUiState,\n instantSearchInstance: localInstantSearchInstance!,\n state: helper!.state,\n renderState: localInstantSearchInstance!.renderState,\n templatesConfig: localInstantSearchInstance!.templatesConfig,\n createURL: this.createURL,\n scopedResults: [],\n searchMetadata: {\n isSearchStalled: localInstantSearchInstance!._isSearchStalled,\n },\n });\n }\n });\n\n localInstantSearchInstance.scheduleSearch();\n }\n\n return this;\n },\n\n removeWidgets(widgets) {\n if (!Array.isArray(widgets)) {\n throw new Error(\n withUsage('The `removeWidgets` method expects an array of widgets.')\n );\n }\n\n if (widgets.some((widget) => typeof widget.dispose !== 'function')) {\n throw new Error(\n withUsage('The widget definition expects a `dispose` method.')\n );\n }\n\n localWidgets = localWidgets.filter(\n (widget) => widgets.indexOf(widget) === -1\n );\n\n if (localInstantSearchInstance && Boolean(widgets.length)) {\n const nextState = widgets.reduce((state, widget) => {\n // the `dispose` method exists at this point we already assert it\n const next = widget.dispose!({\n helper: helper!,\n state,\n parent: this,\n });\n\n return next || state;\n }, helper!.state);\n\n localUiState = getLocalWidgetsUiState(localWidgets, {\n searchParameters: nextState,\n helper: helper!,\n });\n\n helper!.setState(\n getLocalWidgetsSearchParameters(localWidgets, {\n uiState: localUiState,\n initialSearchParameters: nextState,\n })\n );\n\n if (localWidgets.length) {\n localInstantSearchInstance.scheduleSearch();\n }\n }\n\n return this;\n },\n\n init({ instantSearchInstance, parent, uiState }: IndexInitOptions) {\n if (helper !== null) {\n // helper is already initialized, therefore we do not need to set up\n // any listeners\n return;\n }\n\n localInstantSearchInstance = instantSearchInstance;\n localParent = parent;\n localUiState = uiState[indexId] || {};\n\n // The `mainHelper` is already defined at this point. The instance is created\n // inside InstantSearch at the `start` method, which occurs before the `init`\n // step.\n const mainHelper = instantSearchInstance.mainHelper!;\n const parameters = getLocalWidgetsSearchParameters(localWidgets, {\n uiState: localUiState,\n initialSearchParameters: new algoliasearchHelper.SearchParameters({\n index: indexName,\n }),\n });\n\n // This Helper is only used for state management we do not care about the\n // `searchClient`. Only the \"main\" Helper created at the `InstantSearch`\n // level is aware of the client.\n helper = algoliasearchHelper(\n {} as SearchClient,\n parameters.index,\n parameters\n );\n\n // We forward the call to `search` to the \"main\" instance of the Helper\n // which is responsible for managing the queries (it's the only one that is\n // aware of the `searchClient`).\n helper.search = () => {\n if (instantSearchInstance.onStateChange) {\n instantSearchInstance.onStateChange({\n uiState: instantSearchInstance.mainIndex.getWidgetUiState({}),\n setUiState: instantSearchInstance.setUiState.bind(\n instantSearchInstance\n ),\n });\n\n // We don't trigger a search when controlled because it becomes the\n // responsibility of `setUiState`.\n return mainHelper;\n }\n\n return mainHelper.search();\n };\n\n helper.searchWithoutTriggeringOnStateChange = () => {\n return mainHelper.search();\n };\n\n // We use the same pattern for the `searchForFacetValues`.\n helper.searchForFacetValues = (\n facetName,\n facetValue,\n maxFacetHits,\n userState: PlainSearchParameters\n ) => {\n const state = helper!.state.setQueryParameters(userState);\n\n return mainHelper.searchForFacetValues(\n facetName,\n facetValue,\n maxFacetHits,\n state\n );\n };\n\n derivedHelper = mainHelper.derive(() =>\n mergeSearchParameters(...resolveSearchParameters(this))\n );\n\n const indexInitialResults =\n instantSearchInstance._initialResults?.[this.getIndexId()];\n\n if (indexInitialResults) {\n // We restore the shape of the results provided to the instance to respect\n // the helper's structure.\n const results = new algoliasearchHelper.SearchResults(\n new algoliasearchHelper.SearchParameters(indexInitialResults.state),\n indexInitialResults.results\n );\n\n derivedHelper.lastResults = results;\n helper.lastResults = results;\n }\n\n // Subscribe to the Helper state changes for the page before widgets\n // are initialized. This behavior mimics the original one of the Helper.\n // It makes sense to replicate it at the `init` step. We have another\n // listener on `change` below, once `init` is done.\n helper.on('change', ({ isPageReset }) => {\n if (isPageReset) {\n resetPageFromWidgets(localWidgets);\n }\n });\n\n derivedHelper.on('search', () => {\n // The index does not manage the \"staleness\" of the search. This is the\n // responsibility of the main instance. It does not make sense to manage\n // it at the index level because it's either: all of them or none of them\n // that are stalled. The queries are performed into a single network request.\n instantSearchInstance.scheduleStalledRender();\n\n if (__DEV__) {\n checkIndexUiState({ index: this, indexUiState: localUiState });\n }\n });\n\n derivedHelper.on('result', ({ results }) => {\n // The index does not render the results it schedules a new render\n // to let all the other indices emit their own results. It allows us to\n // run the render process in one pass.\n instantSearchInstance.scheduleRender();\n\n // the derived helper is the one which actually searches, but the helper\n // which is exposed e.g. via instance.helper, doesn't search, and thus\n // does not have access to lastResults, which it used to in pre-federated\n // search behavior.\n helper!.lastResults = results;\n });\n\n // We compute the render state before calling `init` in a separate loop\n // to construct the whole render state object that is then passed to\n // `init`.\n localWidgets.forEach((widget) => {\n if (widget.getRenderState) {\n const renderState = widget.getRenderState(\n instantSearchInstance.renderState[this.getIndexId()] || {},\n {\n uiState,\n helper: helper!,\n parent: this,\n instantSearchInstance,\n state: helper!.state,\n renderState: instantSearchInstance.renderState,\n templatesConfig: instantSearchInstance.templatesConfig,\n createURL: this.createURL,\n scopedResults: [],\n searchMetadata: {\n isSearchStalled: instantSearchInstance._isSearchStalled,\n },\n }\n );\n\n storeRenderState({\n renderState,\n instantSearchInstance,\n parent: this,\n });\n }\n });\n\n localWidgets.forEach((widget) => {\n warning(\n // if it has NO getWidgetState or if it has getWidgetUiState, we don't warn\n // aka we warn if there's _only_ getWidgetState\n !widget.getWidgetState || Boolean(widget.getWidgetUiState),\n 'The `getWidgetState` method is renamed `getWidgetUiState` and will no longer exist under that name in InstantSearch.js 5.x. Please use `getWidgetUiState` instead.'\n );\n\n if (widget.init) {\n widget.init({\n uiState,\n helper: helper!,\n parent: this,\n instantSearchInstance,\n state: helper!.state,\n renderState: instantSearchInstance.renderState,\n templatesConfig: instantSearchInstance.templatesConfig,\n createURL: this.createURL,\n scopedResults: [],\n searchMetadata: {\n isSearchStalled: instantSearchInstance._isSearchStalled,\n },\n });\n }\n });\n\n // Subscribe to the Helper state changes for the `uiState` once widgets\n // are initialized. Until the first render, state changes are part of the\n // configuration step. This is mainly for backward compatibility with custom\n // widgets. When the subscription happens before the `init` step, the (static)\n // configuration of the widget is pushed in the URL. That's what we want to avoid.\n // https://github.com/algolia/instantsearch.js/pull/994/commits/4a672ae3fd78809e213de0368549ef12e9dc9454\n helper.on('change', (event) => {\n const { state } = event;\n\n const _uiState = (event as any)._uiState;\n\n localUiState = getLocalWidgetsUiState(\n localWidgets,\n {\n searchParameters: state,\n helper: helper!,\n },\n _uiState || {}\n );\n\n // We don't trigger an internal change when controlled because it\n // becomes the responsibility of `setUiState`.\n if (!instantSearchInstance.onStateChange) {\n instantSearchInstance.onInternalStateChange();\n }\n });\n\n if (indexInitialResults) {\n // If there are initial results, we're not notified of the next results\n // because we don't trigger an initial search. We therefore need to directly\n // schedule a render that will render the results injected on the helper.\n instantSearchInstance.scheduleRender();\n }\n },\n\n render({ instantSearchInstance }: IndexRenderOptions) {\n if (!this.getResults()) {\n return;\n }\n\n localWidgets.forEach((widget) => {\n if (widget.getRenderState) {\n const renderState = widget.getRenderState(\n instantSearchInstance.renderState[this.getIndexId()] || {},\n {\n helper: this.getHelper()!,\n parent: this,\n instantSearchInstance,\n results: this.getResults()!,\n scopedResults: this.getScopedResults(),\n state: this.getResults()!._state,\n renderState: instantSearchInstance.renderState,\n templatesConfig: instantSearchInstance.templatesConfig,\n createURL: this.createURL,\n searchMetadata: {\n isSearchStalled: instantSearchInstance._isSearchStalled,\n },\n }\n );\n\n storeRenderState({\n renderState,\n instantSearchInstance,\n parent: this,\n });\n }\n });\n\n localWidgets.forEach((widget) => {\n // At this point, all the variables used below are set. Both `helper`\n // and `derivedHelper` have been created at the `init` step. The attribute\n // `lastResults` might be `null` though. It's possible that a stalled render\n // happens before the result e.g with a dynamically added index the request might\n // be delayed. The render is triggered for the complete tree but some parts do\n // not have results yet.\n\n if (widget.render) {\n widget.render({\n helper: helper!,\n parent: this,\n instantSearchInstance,\n results: this.getResults()!,\n scopedResults: this.getScopedResults(),\n state: this.getResults()!._state,\n renderState: instantSearchInstance.renderState,\n templatesConfig: instantSearchInstance.templatesConfig,\n createURL: this.createURL,\n searchMetadata: {\n isSearchStalled: instantSearchInstance._isSearchStalled,\n },\n });\n }\n });\n },\n\n dispose() {\n localWidgets.forEach((widget) => {\n if (widget.dispose) {\n // The dispose function is always called once the instance is started\n // (it's an effect of `removeWidgets`). The index is initialized and\n // the Helper is available. We don't care about the return value of\n // `dispose` because the index is removed. We can't call `removeWidgets`\n // because we want to keep the widgets on the instance, to allow idempotent\n // operations on `add` & `remove`.\n widget.dispose({\n helper: helper!,\n state: helper!.state,\n parent: this,\n });\n }\n });\n\n localInstantSearchInstance = null;\n localParent = null;\n helper!.removeAllListeners();\n helper = null;\n\n derivedHelper!.detach();\n derivedHelper = null;\n },\n\n getWidgetUiState(uiState: TUiState) {\n return localWidgets\n .filter(isIndexWidget)\n .reduce(\n (previousUiState, innerIndex) =>\n innerIndex.getWidgetUiState(previousUiState),\n {\n ...uiState,\n [this.getIndexId()]: localUiState,\n }\n );\n },\n\n getWidgetState(uiState: UiState) {\n warning(\n false,\n 'The `getWidgetState` method is renamed `getWidgetUiState` and will no longer exist under that name in InstantSearch.js 5.x. Please use `getWidgetUiState` instead.'\n );\n\n return this.getWidgetUiState(uiState);\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n return getLocalWidgetsSearchParameters(localWidgets, {\n uiState,\n initialSearchParameters: searchParameters,\n });\n },\n\n refreshUiState() {\n localUiState = getLocalWidgetsUiState(\n localWidgets,\n {\n searchParameters: this.getHelper()!.state,\n helper: this.getHelper()!,\n },\n localUiState\n );\n },\n };\n};\n\nexport default index;\n\nfunction storeRenderState({\n renderState,\n instantSearchInstance,\n parent,\n}: {\n renderState: IndexRenderState;\n instantSearchInstance: InstantSearch;\n parent?: IndexWidget;\n}) {\n const parentIndexName = parent\n ? parent.getIndexId()\n : instantSearchInstance.mainIndex.getIndexId();\n\n instantSearchInstance.renderState = {\n ...instantSearchInstance.renderState,\n [parentIndexName]: {\n ...instantSearchInstance.renderState[parentIndexName],\n ...renderState,\n },\n };\n}\n","export default '4.40.5';\n","const NAMESPACE = 'ais';\n\ntype SuitOptions = {\n descendantName?: string;\n modifierName?: string;\n};\n\ntype SuitSelector = (names?: SuitOptions) => string;\n\nexport const component =\n (componentName: string): SuitSelector =>\n ({ descendantName, modifierName }: SuitOptions = {}) => {\n const descendent = descendantName ? `-${descendantName}` : '';\n const modifier = modifierName ? `--${modifierName}` : '';\n\n return `${NAMESPACE}-${componentName}${descendent}${modifier}`;\n };\n","import type { Hit } from '../types';\nimport { component } from '../lib/suit';\nimport { getPropertyByPath, TAG_REPLACEMENT, warning } from '../lib/utils';\n\nexport type HighlightOptions = {\n // @MAJOR string should no longer be allowed to be a path, only array can be a path\n attribute: string | string[];\n highlightedTagName?: string;\n hit: Partial;\n cssClasses?: Partial<{\n highlighted: string;\n }>;\n};\n\nconst suit = component('Highlight');\n\nexport default function highlight({\n attribute,\n highlightedTagName = 'mark',\n hit,\n cssClasses = {},\n}: HighlightOptions): string {\n const highlightAttributeResult = getPropertyByPath(\n hit._highlightResult,\n attribute\n );\n\n // @MAJOR fallback to attribute value if highlight is not found\n warning(\n highlightAttributeResult,\n `Could not enable highlight for \"${attribute}\", will display an empty string.\nPlease check whether this attribute exists and is either searchable or specified in \\`attributesToHighlight\\`.\n\nSee: https://alg.li/highlighting\n`\n );\n\n const { value: attributeValue = '' } = highlightAttributeResult || {};\n\n // cx is not used, since it would be bundled as a dependency for Vue & Angular\n const className =\n suit({\n descendantName: 'highlighted',\n }) + (cssClasses.highlighted ? ` ${cssClasses.highlighted}` : '');\n\n return attributeValue\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPreTag, 'g'),\n `<${highlightedTagName} class=\"${className}\">`\n )\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPostTag, 'g'),\n ``\n );\n}\n","import type { Hit } from '../types';\nimport {\n TAG_REPLACEMENT,\n getPropertyByPath,\n getHighlightedParts,\n reverseHighlightedParts,\n concatHighlightedParts,\n warning,\n} from '../lib/utils';\nimport { component } from '../lib/suit';\n\nexport type ReverseHighlightOptions = {\n // @MAJOR string should no longer be allowed to be a path, only array can be a path\n attribute: string | string[];\n highlightedTagName?: string;\n hit: Partial;\n cssClasses?: Partial<{\n highlighted: string;\n }>;\n};\n\nconst suit = component('ReverseHighlight');\n\nexport default function reverseHighlight({\n attribute,\n highlightedTagName = 'mark',\n hit,\n cssClasses = {},\n}: ReverseHighlightOptions): string {\n const highlightAttributeResult = getPropertyByPath(\n hit._highlightResult,\n attribute\n );\n\n // @MAJOR fallback to attribute value if highlight is not found\n warning(\n highlightAttributeResult,\n `Could not enable reverse highlight for \"${attribute}\", will display an empty string.\nPlease check whether this attribute exists and is either searchable or specified in \\`attributesToHighlight\\`.\n\nSee: https://alg.li/highlighting\n`\n );\n\n const { value: attributeValue = '' } = highlightAttributeResult || {};\n\n // cx is not used, since it would be bundled as a dependency for Vue & Angular\n const className =\n suit({\n descendantName: 'highlighted',\n }) + (cssClasses.highlighted ? ` ${cssClasses.highlighted}` : '');\n\n const reverseHighlightedValue = concatHighlightedParts(\n reverseHighlightedParts(getHighlightedParts(attributeValue))\n );\n\n return reverseHighlightedValue\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPreTag, 'g'),\n `<${highlightedTagName} class=\"${className}\">`\n )\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPostTag, 'g'),\n ``\n );\n}\n","import type { Hit } from '../types';\nimport { component } from '../lib/suit';\nimport { TAG_REPLACEMENT, getPropertyByPath, warning } from '../lib/utils';\n\nexport type SnippetOptions = {\n // @MAJOR string should no longer be allowed to be a path, only array can be a path\n attribute: string | string[];\n highlightedTagName?: string;\n hit: Partial;\n cssClasses?: {\n highlighted?: string;\n };\n};\n\nconst suit = component('Snippet');\n\nexport default function snippet({\n attribute,\n highlightedTagName = 'mark',\n hit,\n cssClasses = {},\n}: SnippetOptions): string {\n const snippetAttributeResult = getPropertyByPath(\n hit._snippetResult,\n attribute\n );\n\n // @MAJOR fallback to attribute value if snippet is not found\n warning(\n snippetAttributeResult,\n `Could not enable snippet for \"${attribute}\", will display an empty string.\nPlease check whether this attribute exists and is specified in \\`attributesToSnippet\\`.\n\nSee: https://alg.li/highlighting\n`\n );\n\n const { value: attributeValue = '' } = snippetAttributeResult || {};\n\n // cx is not used, since it would be bundled as a dependency for Vue & Angular\n const className =\n suit({\n descendantName: 'highlighted',\n }) + (cssClasses.highlighted ? ` ${cssClasses.highlighted}` : '');\n\n return attributeValue\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPreTag, 'g'),\n `<${highlightedTagName} class=\"${className}\">`\n )\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPostTag, 'g'),\n ``\n );\n}\n","import type { Hit } from '../types';\nimport {\n TAG_REPLACEMENT,\n getPropertyByPath,\n getHighlightedParts,\n reverseHighlightedParts,\n concatHighlightedParts,\n warning,\n} from '../lib/utils';\nimport { component } from '../lib/suit';\n\nexport type ReverseSnippetOptions = {\n // @MAJOR string should no longer be allowed to be a path, only array can be a path\n attribute: string | string[];\n highlightedTagName?: string;\n hit: Partial;\n cssClasses?: Partial<{\n highlighted: string;\n }>;\n};\n\nconst suit = component('ReverseSnippet');\n\nexport default function reverseSnippet({\n attribute,\n highlightedTagName = 'mark',\n hit,\n cssClasses = {},\n}: ReverseSnippetOptions): string {\n const snippetAttributeResult = getPropertyByPath(\n hit._snippetResult,\n attribute\n );\n\n // @MAJOR fallback to attribute value if snippet is not found\n warning(\n snippetAttributeResult,\n `Could not enable reverse snippet for \"${attribute}\", will display an empty string.\nPlease check whether this attribute exists and is specified in \\`attributesToSnippet\\`.\n\nSee: https://alg.li/highlighting\n`\n );\n\n const { value: attributeValue = '' } = snippetAttributeResult || {};\n\n // cx is not used, since it would be bundled as a dependency for Vue & Angular\n const className =\n suit({\n descendantName: 'highlighted',\n }) + (cssClasses.highlighted ? ` ${cssClasses.highlighted}` : '');\n\n const reverseHighlightedValue = concatHighlightedParts(\n reverseHighlightedParts(getHighlightedParts(attributeValue))\n );\n\n return reverseHighlightedValue\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPreTag, 'g'),\n `<${highlightedTagName} class=\"${className}\">`\n )\n .replace(\n new RegExp(TAG_REPLACEMENT.highlightPostTag, 'g'),\n ``\n );\n}\n","import type { InsightsClientMethod, InsightsClientPayload } from '../types';\nimport { warning, serializePayload, deserializePayload } from '../lib/utils';\n\nexport function readDataAttributes(domElement: HTMLElement): {\n method: InsightsClientMethod;\n payload: Partial;\n} {\n const method = domElement.getAttribute(\n 'data-insights-method'\n ) as InsightsClientMethod;\n\n const serializedPayload = domElement.getAttribute('data-insights-payload');\n\n if (typeof serializedPayload !== 'string') {\n throw new Error(\n 'The insights helper expects `data-insights-payload` to be a base64-encoded JSON string.'\n );\n }\n\n try {\n const payload =\n deserializePayload>(serializedPayload);\n return { method, payload };\n } catch (error) {\n throw new Error(\n 'The insights helper was unable to parse `data-insights-payload`.'\n );\n }\n}\n\nexport function hasDataAttributes(domElement: HTMLElement): boolean {\n return domElement.hasAttribute('data-insights-method');\n}\n\nexport function writeDataAttributes({\n method,\n payload,\n}: {\n method: InsightsClientMethod;\n payload: Partial;\n}): string {\n if (typeof payload !== 'object') {\n throw new Error(`The insights helper expects the payload to be an object.`);\n }\n\n let serializedPayload: string;\n\n try {\n serializedPayload = serializePayload(payload);\n } catch (error) {\n throw new Error(`Could not JSON serialize the payload object.`);\n }\n\n return `data-insights-method=\"${method}\" data-insights-payload=\"${serializedPayload}\"`;\n}\n\n/**\n * @deprecated This function will be still supported in 4.x releases, but not further. It is replaced by the `insights` middleware. For more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/\n */\nexport default function insights(\n method: InsightsClientMethod,\n payload: Partial\n): string {\n warning(\n false,\n `\\`insights\\` function has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the \\`insights\\` middleware.\n\nFor more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/`\n );\n return writeDataAttributes({ method, payload });\n}\n","import { warning } from '../lib/utils';\n\nexport const ANONYMOUS_TOKEN_COOKIE_KEY = '_ALGOLIA';\n\nfunction getCookie(name: string): string | undefined {\n const prefix = `${name}=`;\n const cookies = document.cookie.split(';');\n for (let i = 0; i < cookies.length; i++) {\n let cookie = cookies[i];\n while (cookie.charAt(0) === ' ') {\n cookie = cookie.substring(1);\n }\n if (cookie.indexOf(prefix) === 0) {\n return cookie.substring(prefix.length, cookie.length);\n }\n }\n return undefined;\n}\n\nexport function getInsightsAnonymousUserTokenInternal(): string | undefined {\n return getCookie(ANONYMOUS_TOKEN_COOKIE_KEY);\n}\n\n/**\n * @deprecated This function will be still supported in 4.x releases, but not further. It is replaced by the `insights` middleware. For more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/\n */\nexport default function getInsightsAnonymousUserToken(): string | undefined {\n warning(\n false,\n `\\`getInsightsAnonymousUserToken\\` function has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the \\`insights\\` middleware.\n\nFor more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/`\n );\n return getInsightsAnonymousUserTokenInternal();\n}\n","import type {\n HighlightOptions,\n ReverseHighlightOptions,\n SnippetOptions,\n ReverseSnippetOptions,\n} from '../helpers';\nimport {\n highlight,\n reverseHighlight,\n snippet,\n reverseSnippet,\n insights,\n} from '../helpers';\nimport type {\n Hit,\n HoganHelpers,\n InsightsClientMethod,\n InsightsClientPayload,\n} from '../types';\n\ntype DefaultHoganHelpers = HoganHelpers<\n | 'formatNumber'\n | 'highlight'\n | 'reverseHighlight'\n | 'snippet'\n | 'reverseSnippet'\n | 'insights'\n>;\n\nexport default function hoganHelpers({\n numberLocale,\n}: {\n numberLocale?: string;\n}): DefaultHoganHelpers {\n return {\n formatNumber(value, render) {\n return Number(render(value)).toLocaleString(numberLocale);\n },\n highlight(options, render) {\n try {\n const highlightOptions: Omit =\n JSON.parse(options);\n\n return render(\n highlight({\n ...highlightOptions,\n hit: this,\n })\n );\n } catch (error) {\n throw new Error(`\nThe highlight helper expects a JSON object of the format:\n{ \"attribute\": \"name\", \"highlightedTagName\": \"mark\" }`);\n }\n },\n reverseHighlight(options, render) {\n try {\n const reverseHighlightOptions: Omit =\n JSON.parse(options);\n\n return render(\n reverseHighlight({\n ...reverseHighlightOptions,\n hit: this,\n })\n );\n } catch (error) {\n throw new Error(`\n The reverseHighlight helper expects a JSON object of the format:\n { \"attribute\": \"name\", \"highlightedTagName\": \"mark\" }`);\n }\n },\n snippet(options, render) {\n try {\n const snippetOptions: Omit = JSON.parse(options);\n\n return render(\n snippet({\n ...snippetOptions,\n hit: this,\n })\n );\n } catch (error) {\n throw new Error(`\nThe snippet helper expects a JSON object of the format:\n{ \"attribute\": \"name\", \"highlightedTagName\": \"mark\" }`);\n }\n },\n reverseSnippet(options, render) {\n try {\n const reverseSnippetOptions: Omit =\n JSON.parse(options);\n\n return render(\n reverseSnippet({\n ...reverseSnippetOptions,\n hit: this,\n })\n );\n } catch (error) {\n throw new Error(`\n The reverseSnippet helper expects a JSON object of the format:\n { \"attribute\": \"name\", \"highlightedTagName\": \"mark\" }`);\n }\n },\n insights(this: Hit, options, render) {\n try {\n type InsightsHelperOptions = {\n method: InsightsClientMethod;\n payload: Partial;\n };\n const { method, payload }: InsightsHelperOptions = JSON.parse(options);\n\n return render(\n insights(method, { objectIDs: [this.objectID], ...payload })\n );\n } catch (error) {\n throw new Error(`\nThe insights helper expects a JSON object of the format:\n{ \"method\": \"method-name\", \"payload\": { \"eventName\": \"name of the event\" } }`);\n }\n },\n };\n}\n","import type { UiState, IndexUiState, StateMapping } from '../../types';\n\nfunction getIndexStateWithoutConfigure(\n uiState: TIndexUiState\n): Omit {\n const { configure, ...trackedUiState } = uiState;\n return trackedUiState;\n}\n\n// technically a URL could contain any key, since users provide it,\n// which is why the input to this function is UiState, not something\n// which excludes \"configure\" as this function does.\nexport default function simpleStateMapping<\n TUiState extends UiState = UiState\n>(): StateMapping {\n return {\n stateToRoute(uiState) {\n return Object.keys(uiState).reduce(\n (state, indexId) => ({\n ...state,\n [indexId]: getIndexStateWithoutConfigure(uiState[indexId]),\n }),\n {} as TUiState\n );\n },\n\n routeToState(routeState = {} as TUiState) {\n return Object.keys(routeState).reduce(\n (state, indexId) => ({\n ...state,\n [indexId]: getIndexStateWithoutConfigure(routeState[indexId]),\n }),\n {} as TUiState\n );\n },\n };\n}\n","'use strict';\n\nvar replace = String.prototype.replace;\nvar percentTwenties = /%20/g;\n\nvar Format = {\n RFC1738: 'RFC1738',\n RFC3986: 'RFC3986'\n};\n\nmodule.exports = {\n 'default': Format.RFC3986,\n formatters: {\n RFC1738: function (value) {\n return replace.call(value, percentTwenties, '+');\n },\n RFC3986: function (value) {\n return String(value);\n }\n },\n RFC1738: Format.RFC1738,\n RFC3986: Format.RFC3986\n};\n","'use strict';\n\nvar formats = require('./formats');\n\nvar has = Object.prototype.hasOwnProperty;\nvar isArray = Array.isArray;\n\nvar hexTable = (function () {\n var array = [];\n for (var i = 0; i < 256; ++i) {\n array.push('%' + ((i < 16 ? '0' : '') + i.toString(16)).toUpperCase());\n }\n\n return array;\n}());\n\nvar compactQueue = function compactQueue(queue) {\n while (queue.length > 1) {\n var item = queue.pop();\n var obj = item.obj[item.prop];\n\n if (isArray(obj)) {\n var compacted = [];\n\n for (var j = 0; j < obj.length; ++j) {\n if (typeof obj[j] !== 'undefined') {\n compacted.push(obj[j]);\n }\n }\n\n item.obj[item.prop] = compacted;\n }\n }\n};\n\nvar arrayToObject = function arrayToObject(source, options) {\n var obj = options && options.plainObjects ? Object.create(null) : {};\n for (var i = 0; i < source.length; ++i) {\n if (typeof source[i] !== 'undefined') {\n obj[i] = source[i];\n }\n }\n\n return obj;\n};\n\nvar merge = function merge(target, source, options) {\n /* eslint no-param-reassign: 0 */\n if (!source) {\n return target;\n }\n\n if (typeof source !== 'object') {\n if (isArray(target)) {\n target.push(source);\n } else if (target && typeof target === 'object') {\n if ((options && (options.plainObjects || options.allowPrototypes)) || !has.call(Object.prototype, source)) {\n target[source] = true;\n }\n } else {\n return [target, source];\n }\n\n return target;\n }\n\n if (!target || typeof target !== 'object') {\n return [target].concat(source);\n }\n\n var mergeTarget = target;\n if (isArray(target) && !isArray(source)) {\n mergeTarget = arrayToObject(target, options);\n }\n\n if (isArray(target) && isArray(source)) {\n source.forEach(function (item, i) {\n if (has.call(target, i)) {\n var targetItem = target[i];\n if (targetItem && typeof targetItem === 'object' && item && typeof item === 'object') {\n target[i] = merge(targetItem, item, options);\n } else {\n target.push(item);\n }\n } else {\n target[i] = item;\n }\n });\n return target;\n }\n\n return Object.keys(source).reduce(function (acc, key) {\n var value = source[key];\n\n if (has.call(acc, key)) {\n acc[key] = merge(acc[key], value, options);\n } else {\n acc[key] = value;\n }\n return acc;\n }, mergeTarget);\n};\n\nvar assign = function assignSingleSource(target, source) {\n return Object.keys(source).reduce(function (acc, key) {\n acc[key] = source[key];\n return acc;\n }, target);\n};\n\nvar decode = function (str, decoder, charset) {\n var strWithoutPlus = str.replace(/\\+/g, ' ');\n if (charset === 'iso-8859-1') {\n // unescape never throws, no try...catch needed:\n return strWithoutPlus.replace(/%[0-9a-f]{2}/gi, unescape);\n }\n // utf-8\n try {\n return decodeURIComponent(strWithoutPlus);\n } catch (e) {\n return strWithoutPlus;\n }\n};\n\nvar encode = function encode(str, defaultEncoder, charset, kind, format) {\n // This code was originally written by Brian White (mscdex) for the io.js core querystring library.\n // It has been adapted here for stricter adherence to RFC 3986\n if (str.length === 0) {\n return str;\n }\n\n var string = str;\n if (typeof str === 'symbol') {\n string = Symbol.prototype.toString.call(str);\n } else if (typeof str !== 'string') {\n string = String(str);\n }\n\n if (charset === 'iso-8859-1') {\n return escape(string).replace(/%u[0-9a-f]{4}/gi, function ($0) {\n return '%26%23' + parseInt($0.slice(2), 16) + '%3B';\n });\n }\n\n var out = '';\n for (var i = 0; i < string.length; ++i) {\n var c = string.charCodeAt(i);\n\n if (\n c === 0x2D // -\n || c === 0x2E // .\n || c === 0x5F // _\n || c === 0x7E // ~\n || (c >= 0x30 && c <= 0x39) // 0-9\n || (c >= 0x41 && c <= 0x5A) // a-z\n || (c >= 0x61 && c <= 0x7A) // A-Z\n || (format === formats.RFC1738 && (c === 0x28 || c === 0x29)) // ( )\n ) {\n out += string.charAt(i);\n continue;\n }\n\n if (c < 0x80) {\n out = out + hexTable[c];\n continue;\n }\n\n if (c < 0x800) {\n out = out + (hexTable[0xC0 | (c >> 6)] + hexTable[0x80 | (c & 0x3F)]);\n continue;\n }\n\n if (c < 0xD800 || c >= 0xE000) {\n out = out + (hexTable[0xE0 | (c >> 12)] + hexTable[0x80 | ((c >> 6) & 0x3F)] + hexTable[0x80 | (c & 0x3F)]);\n continue;\n }\n\n i += 1;\n c = 0x10000 + (((c & 0x3FF) << 10) | (string.charCodeAt(i) & 0x3FF));\n out += hexTable[0xF0 | (c >> 18)]\n + hexTable[0x80 | ((c >> 12) & 0x3F)]\n + hexTable[0x80 | ((c >> 6) & 0x3F)]\n + hexTable[0x80 | (c & 0x3F)];\n }\n\n return out;\n};\n\nvar compact = function compact(value) {\n var queue = [{ obj: { o: value }, prop: 'o' }];\n var refs = [];\n\n for (var i = 0; i < queue.length; ++i) {\n var item = queue[i];\n var obj = item.obj[item.prop];\n\n var keys = Object.keys(obj);\n for (var j = 0; j < keys.length; ++j) {\n var key = keys[j];\n var val = obj[key];\n if (typeof val === 'object' && val !== null && refs.indexOf(val) === -1) {\n queue.push({ obj: obj, prop: key });\n refs.push(val);\n }\n }\n }\n\n compactQueue(queue);\n\n return value;\n};\n\nvar isRegExp = function isRegExp(obj) {\n return Object.prototype.toString.call(obj) === '[object RegExp]';\n};\n\nvar isBuffer = function isBuffer(obj) {\n if (!obj || typeof obj !== 'object') {\n return false;\n }\n\n return !!(obj.constructor && obj.constructor.isBuffer && obj.constructor.isBuffer(obj));\n};\n\nvar combine = function combine(a, b) {\n return [].concat(a, b);\n};\n\nvar maybeMap = function maybeMap(val, fn) {\n if (isArray(val)) {\n var mapped = [];\n for (var i = 0; i < val.length; i += 1) {\n mapped.push(fn(val[i]));\n }\n return mapped;\n }\n return fn(val);\n};\n\nmodule.exports = {\n arrayToObject: arrayToObject,\n assign: assign,\n combine: combine,\n compact: compact,\n decode: decode,\n encode: encode,\n isBuffer: isBuffer,\n isRegExp: isRegExp,\n maybeMap: maybeMap,\n merge: merge\n};\n","'use strict';\n\nvar utils = require('./utils');\nvar formats = require('./formats');\nvar has = Object.prototype.hasOwnProperty;\n\nvar arrayPrefixGenerators = {\n brackets: function brackets(prefix) {\n return prefix + '[]';\n },\n comma: 'comma',\n indices: function indices(prefix, key) {\n return prefix + '[' + key + ']';\n },\n repeat: function repeat(prefix) {\n return prefix;\n }\n};\n\nvar isArray = Array.isArray;\nvar push = Array.prototype.push;\nvar pushToArray = function (arr, valueOrArray) {\n push.apply(arr, isArray(valueOrArray) ? valueOrArray : [valueOrArray]);\n};\n\nvar toISO = Date.prototype.toISOString;\n\nvar defaultFormat = formats['default'];\nvar defaults = {\n addQueryPrefix: false,\n allowDots: false,\n charset: 'utf-8',\n charsetSentinel: false,\n delimiter: '&',\n encode: true,\n encoder: utils.encode,\n encodeValuesOnly: false,\n format: defaultFormat,\n formatter: formats.formatters[defaultFormat],\n // deprecated\n indices: false,\n serializeDate: function serializeDate(date) {\n return toISO.call(date);\n },\n skipNulls: false,\n strictNullHandling: false\n};\n\nvar isNonNullishPrimitive = function isNonNullishPrimitive(v) {\n return typeof v === 'string'\n || typeof v === 'number'\n || typeof v === 'boolean'\n || typeof v === 'symbol'\n || typeof v === 'bigint';\n};\n\nvar stringify = function stringify(\n object,\n prefix,\n generateArrayPrefix,\n strictNullHandling,\n skipNulls,\n encoder,\n filter,\n sort,\n allowDots,\n serializeDate,\n format,\n formatter,\n encodeValuesOnly,\n charset\n) {\n var obj = object;\n if (typeof filter === 'function') {\n obj = filter(prefix, obj);\n } else if (obj instanceof Date) {\n obj = serializeDate(obj);\n } else if (generateArrayPrefix === 'comma' && isArray(obj)) {\n obj = utils.maybeMap(obj, function (value) {\n if (value instanceof Date) {\n return serializeDate(value);\n }\n return value;\n });\n }\n\n if (obj === null) {\n if (strictNullHandling) {\n return encoder && !encodeValuesOnly ? encoder(prefix, defaults.encoder, charset, 'key', format) : prefix;\n }\n\n obj = '';\n }\n\n if (isNonNullishPrimitive(obj) || utils.isBuffer(obj)) {\n if (encoder) {\n var keyValue = encodeValuesOnly ? prefix : encoder(prefix, defaults.encoder, charset, 'key', format);\n return [formatter(keyValue) + '=' + formatter(encoder(obj, defaults.encoder, charset, 'value', format))];\n }\n return [formatter(prefix) + '=' + formatter(String(obj))];\n }\n\n var values = [];\n\n if (typeof obj === 'undefined') {\n return values;\n }\n\n var objKeys;\n if (generateArrayPrefix === 'comma' && isArray(obj)) {\n // we need to join elements in\n objKeys = [{ value: obj.length > 0 ? obj.join(',') || null : undefined }];\n } else if (isArray(filter)) {\n objKeys = filter;\n } else {\n var keys = Object.keys(obj);\n objKeys = sort ? keys.sort(sort) : keys;\n }\n\n for (var i = 0; i < objKeys.length; ++i) {\n var key = objKeys[i];\n var value = typeof key === 'object' && key.value !== undefined ? key.value : obj[key];\n\n if (skipNulls && value === null) {\n continue;\n }\n\n var keyPrefix = isArray(obj)\n ? typeof generateArrayPrefix === 'function' ? generateArrayPrefix(prefix, key) : prefix\n : prefix + (allowDots ? '.' + key : '[' + key + ']');\n\n pushToArray(values, stringify(\n value,\n keyPrefix,\n generateArrayPrefix,\n strictNullHandling,\n skipNulls,\n encoder,\n filter,\n sort,\n allowDots,\n serializeDate,\n format,\n formatter,\n encodeValuesOnly,\n charset\n ));\n }\n\n return values;\n};\n\nvar normalizeStringifyOptions = function normalizeStringifyOptions(opts) {\n if (!opts) {\n return defaults;\n }\n\n if (opts.encoder !== null && opts.encoder !== undefined && typeof opts.encoder !== 'function') {\n throw new TypeError('Encoder has to be a function.');\n }\n\n var charset = opts.charset || defaults.charset;\n if (typeof opts.charset !== 'undefined' && opts.charset !== 'utf-8' && opts.charset !== 'iso-8859-1') {\n throw new TypeError('The charset option must be either utf-8, iso-8859-1, or undefined');\n }\n\n var format = formats['default'];\n if (typeof opts.format !== 'undefined') {\n if (!has.call(formats.formatters, opts.format)) {\n throw new TypeError('Unknown format option provided.');\n }\n format = opts.format;\n }\n var formatter = formats.formatters[format];\n\n var filter = defaults.filter;\n if (typeof opts.filter === 'function' || isArray(opts.filter)) {\n filter = opts.filter;\n }\n\n return {\n addQueryPrefix: typeof opts.addQueryPrefix === 'boolean' ? opts.addQueryPrefix : defaults.addQueryPrefix,\n allowDots: typeof opts.allowDots === 'undefined' ? defaults.allowDots : !!opts.allowDots,\n charset: charset,\n charsetSentinel: typeof opts.charsetSentinel === 'boolean' ? opts.charsetSentinel : defaults.charsetSentinel,\n delimiter: typeof opts.delimiter === 'undefined' ? defaults.delimiter : opts.delimiter,\n encode: typeof opts.encode === 'boolean' ? opts.encode : defaults.encode,\n encoder: typeof opts.encoder === 'function' ? opts.encoder : defaults.encoder,\n encodeValuesOnly: typeof opts.encodeValuesOnly === 'boolean' ? opts.encodeValuesOnly : defaults.encodeValuesOnly,\n filter: filter,\n format: format,\n formatter: formatter,\n serializeDate: typeof opts.serializeDate === 'function' ? opts.serializeDate : defaults.serializeDate,\n skipNulls: typeof opts.skipNulls === 'boolean' ? opts.skipNulls : defaults.skipNulls,\n sort: typeof opts.sort === 'function' ? opts.sort : null,\n strictNullHandling: typeof opts.strictNullHandling === 'boolean' ? opts.strictNullHandling : defaults.strictNullHandling\n };\n};\n\nmodule.exports = function (object, opts) {\n var obj = object;\n var options = normalizeStringifyOptions(opts);\n\n var objKeys;\n var filter;\n\n if (typeof options.filter === 'function') {\n filter = options.filter;\n obj = filter('', obj);\n } else if (isArray(options.filter)) {\n filter = options.filter;\n objKeys = filter;\n }\n\n var keys = [];\n\n if (typeof obj !== 'object' || obj === null) {\n return '';\n }\n\n var arrayFormat;\n if (opts && opts.arrayFormat in arrayPrefixGenerators) {\n arrayFormat = opts.arrayFormat;\n } else if (opts && 'indices' in opts) {\n arrayFormat = opts.indices ? 'indices' : 'repeat';\n } else {\n arrayFormat = 'indices';\n }\n\n var generateArrayPrefix = arrayPrefixGenerators[arrayFormat];\n\n if (!objKeys) {\n objKeys = Object.keys(obj);\n }\n\n if (options.sort) {\n objKeys.sort(options.sort);\n }\n\n for (var i = 0; i < objKeys.length; ++i) {\n var key = objKeys[i];\n\n if (options.skipNulls && obj[key] === null) {\n continue;\n }\n pushToArray(keys, stringify(\n obj[key],\n key,\n generateArrayPrefix,\n options.strictNullHandling,\n options.skipNulls,\n options.encode ? options.encoder : null,\n options.filter,\n options.sort,\n options.allowDots,\n options.serializeDate,\n options.format,\n options.formatter,\n options.encodeValuesOnly,\n options.charset\n ));\n }\n\n var joined = keys.join(options.delimiter);\n var prefix = options.addQueryPrefix === true ? '?' : '';\n\n if (options.charsetSentinel) {\n if (options.charset === 'iso-8859-1') {\n // encodeURIComponent('✓'), the \"numeric entity\" representation of a checkmark\n prefix += 'utf8=%26%2310003%3B&';\n } else {\n // encodeURIComponent('✓')\n prefix += 'utf8=%E2%9C%93&';\n }\n }\n\n return joined.length > 0 ? prefix + joined : '';\n};\n","'use strict';\n\nvar utils = require('./utils');\n\nvar has = Object.prototype.hasOwnProperty;\nvar isArray = Array.isArray;\n\nvar defaults = {\n allowDots: false,\n allowPrototypes: false,\n arrayLimit: 20,\n charset: 'utf-8',\n charsetSentinel: false,\n comma: false,\n decoder: utils.decode,\n delimiter: '&',\n depth: 5,\n ignoreQueryPrefix: false,\n interpretNumericEntities: false,\n parameterLimit: 1000,\n parseArrays: true,\n plainObjects: false,\n strictNullHandling: false\n};\n\nvar interpretNumericEntities = function (str) {\n return str.replace(/&#(\\d+);/g, function ($0, numberStr) {\n return String.fromCharCode(parseInt(numberStr, 10));\n });\n};\n\nvar parseArrayValue = function (val, options) {\n if (val && typeof val === 'string' && options.comma && val.indexOf(',') > -1) {\n return val.split(',');\n }\n\n return val;\n};\n\n// This is what browsers will submit when the ✓ character occurs in an\n// application/x-www-form-urlencoded body and the encoding of the page containing\n// the form is iso-8859-1, or when the submitted form has an accept-charset\n// attribute of iso-8859-1. Presumably also with other charsets that do not contain\n// the ✓ character, such as us-ascii.\nvar isoSentinel = 'utf8=%26%2310003%3B'; // encodeURIComponent('✓')\n\n// These are the percent-encoded utf-8 octets representing a checkmark, indicating that the request actually is utf-8 encoded.\nvar charsetSentinel = 'utf8=%E2%9C%93'; // encodeURIComponent('✓')\n\nvar parseValues = function parseQueryStringValues(str, options) {\n var obj = {};\n var cleanStr = options.ignoreQueryPrefix ? str.replace(/^\\?/, '') : str;\n var limit = options.parameterLimit === Infinity ? undefined : options.parameterLimit;\n var parts = cleanStr.split(options.delimiter, limit);\n var skipIndex = -1; // Keep track of where the utf8 sentinel was found\n var i;\n\n var charset = options.charset;\n if (options.charsetSentinel) {\n for (i = 0; i < parts.length; ++i) {\n if (parts[i].indexOf('utf8=') === 0) {\n if (parts[i] === charsetSentinel) {\n charset = 'utf-8';\n } else if (parts[i] === isoSentinel) {\n charset = 'iso-8859-1';\n }\n skipIndex = i;\n i = parts.length; // The eslint settings do not allow break;\n }\n }\n }\n\n for (i = 0; i < parts.length; ++i) {\n if (i === skipIndex) {\n continue;\n }\n var part = parts[i];\n\n var bracketEqualsPos = part.indexOf(']=');\n var pos = bracketEqualsPos === -1 ? part.indexOf('=') : bracketEqualsPos + 1;\n\n var key, val;\n if (pos === -1) {\n key = options.decoder(part, defaults.decoder, charset, 'key');\n val = options.strictNullHandling ? null : '';\n } else {\n key = options.decoder(part.slice(0, pos), defaults.decoder, charset, 'key');\n val = utils.maybeMap(\n parseArrayValue(part.slice(pos + 1), options),\n function (encodedVal) {\n return options.decoder(encodedVal, defaults.decoder, charset, 'value');\n }\n );\n }\n\n if (val && options.interpretNumericEntities && charset === 'iso-8859-1') {\n val = interpretNumericEntities(val);\n }\n\n if (part.indexOf('[]=') > -1) {\n val = isArray(val) ? [val] : val;\n }\n\n if (has.call(obj, key)) {\n obj[key] = utils.combine(obj[key], val);\n } else {\n obj[key] = val;\n }\n }\n\n return obj;\n};\n\nvar parseObject = function (chain, val, options, valuesParsed) {\n var leaf = valuesParsed ? val : parseArrayValue(val, options);\n\n for (var i = chain.length - 1; i >= 0; --i) {\n var obj;\n var root = chain[i];\n\n if (root === '[]' && options.parseArrays) {\n obj = [].concat(leaf);\n } else {\n obj = options.plainObjects ? Object.create(null) : {};\n var cleanRoot = root.charAt(0) === '[' && root.charAt(root.length - 1) === ']' ? root.slice(1, -1) : root;\n var index = parseInt(cleanRoot, 10);\n if (!options.parseArrays && cleanRoot === '') {\n obj = { 0: leaf };\n } else if (\n !isNaN(index)\n && root !== cleanRoot\n && String(index) === cleanRoot\n && index >= 0\n && (options.parseArrays && index <= options.arrayLimit)\n ) {\n obj = [];\n obj[index] = leaf;\n } else {\n obj[cleanRoot] = leaf;\n }\n }\n\n leaf = obj;\n }\n\n return leaf;\n};\n\nvar parseKeys = function parseQueryStringKeys(givenKey, val, options, valuesParsed) {\n if (!givenKey) {\n return;\n }\n\n // Transform dot notation to bracket notation\n var key = options.allowDots ? givenKey.replace(/\\.([^.[]+)/g, '[$1]') : givenKey;\n\n // The regex chunks\n\n var brackets = /(\\[[^[\\]]*])/;\n var child = /(\\[[^[\\]]*])/g;\n\n // Get the parent\n\n var segment = options.depth > 0 && brackets.exec(key);\n var parent = segment ? key.slice(0, segment.index) : key;\n\n // Stash the parent if it exists\n\n var keys = [];\n if (parent) {\n // If we aren't using plain objects, optionally prefix keys that would overwrite object prototype properties\n if (!options.plainObjects && has.call(Object.prototype, parent)) {\n if (!options.allowPrototypes) {\n return;\n }\n }\n\n keys.push(parent);\n }\n\n // Loop through children appending to the array until we hit depth\n\n var i = 0;\n while (options.depth > 0 && (segment = child.exec(key)) !== null && i < options.depth) {\n i += 1;\n if (!options.plainObjects && has.call(Object.prototype, segment[1].slice(1, -1))) {\n if (!options.allowPrototypes) {\n return;\n }\n }\n keys.push(segment[1]);\n }\n\n // If there's a remainder, just add whatever is left\n\n if (segment) {\n keys.push('[' + key.slice(segment.index) + ']');\n }\n\n return parseObject(keys, val, options, valuesParsed);\n};\n\nvar normalizeParseOptions = function normalizeParseOptions(opts) {\n if (!opts) {\n return defaults;\n }\n\n if (opts.decoder !== null && opts.decoder !== undefined && typeof opts.decoder !== 'function') {\n throw new TypeError('Decoder has to be a function.');\n }\n\n if (typeof opts.charset !== 'undefined' && opts.charset !== 'utf-8' && opts.charset !== 'iso-8859-1') {\n throw new TypeError('The charset option must be either utf-8, iso-8859-1, or undefined');\n }\n var charset = typeof opts.charset === 'undefined' ? defaults.charset : opts.charset;\n\n return {\n allowDots: typeof opts.allowDots === 'undefined' ? defaults.allowDots : !!opts.allowDots,\n allowPrototypes: typeof opts.allowPrototypes === 'boolean' ? opts.allowPrototypes : defaults.allowPrototypes,\n arrayLimit: typeof opts.arrayLimit === 'number' ? opts.arrayLimit : defaults.arrayLimit,\n charset: charset,\n charsetSentinel: typeof opts.charsetSentinel === 'boolean' ? opts.charsetSentinel : defaults.charsetSentinel,\n comma: typeof opts.comma === 'boolean' ? opts.comma : defaults.comma,\n decoder: typeof opts.decoder === 'function' ? opts.decoder : defaults.decoder,\n delimiter: typeof opts.delimiter === 'string' || utils.isRegExp(opts.delimiter) ? opts.delimiter : defaults.delimiter,\n // eslint-disable-next-line no-implicit-coercion, no-extra-parens\n depth: (typeof opts.depth === 'number' || opts.depth === false) ? +opts.depth : defaults.depth,\n ignoreQueryPrefix: opts.ignoreQueryPrefix === true,\n interpretNumericEntities: typeof opts.interpretNumericEntities === 'boolean' ? opts.interpretNumericEntities : defaults.interpretNumericEntities,\n parameterLimit: typeof opts.parameterLimit === 'number' ? opts.parameterLimit : defaults.parameterLimit,\n parseArrays: opts.parseArrays !== false,\n plainObjects: typeof opts.plainObjects === 'boolean' ? opts.plainObjects : defaults.plainObjects,\n strictNullHandling: typeof opts.strictNullHandling === 'boolean' ? opts.strictNullHandling : defaults.strictNullHandling\n };\n};\n\nmodule.exports = function (str, opts) {\n var options = normalizeParseOptions(opts);\n\n if (str === '' || str === null || typeof str === 'undefined') {\n return options.plainObjects ? Object.create(null) : {};\n }\n\n var tempObj = typeof str === 'string' ? parseValues(str, options) : str;\n var obj = options.plainObjects ? Object.create(null) : {};\n\n // Iterate over the keys and setup the new object\n\n var keys = Object.keys(tempObj);\n for (var i = 0; i < keys.length; ++i) {\n var key = keys[i];\n var newObj = parseKeys(key, tempObj[key], options, typeof str === 'string');\n obj = utils.merge(obj, newObj, options);\n }\n\n return utils.compact(obj);\n};\n","'use strict';\n\nvar stringify = require('./stringify');\nvar parse = require('./parse');\nvar formats = require('./formats');\n\nmodule.exports = {\n formats: formats,\n parse: parse,\n stringify: stringify\n};\n","import qs from 'qs';\nimport type { Router, UiState } from '../../types';\nimport { safelyRunOnBrowser } from '../utils';\n\ntype CreateURL = (args: {\n qsModule: typeof qs;\n routeState: TRouteState;\n location: Location;\n}) => string;\n\ntype ParseURL = (args: {\n qsModule: typeof qs;\n location: Location;\n}) => TRouteState;\n\ntype BrowserHistoryArgs = {\n windowTitle?: (routeState: TRouteState) => string;\n writeDelay: number;\n createURL: CreateURL;\n parseURL: ParseURL;\n // @MAJOR: The `Location` type is hard to simulate in non-browser environments\n // so we should accept a subset of it that is easier to work with in any\n // environments.\n getLocation(): Location;\n};\n\nconst setWindowTitle = (title?: string): void => {\n if (title) {\n // This function is only executed on browsers so we can disable this check.\n // eslint-disable-next-line no-restricted-globals\n window.document.title = title;\n }\n};\n\nclass BrowserHistory implements Router {\n /**\n * Transforms a UI state into a title for the page.\n */\n private readonly windowTitle?: BrowserHistoryArgs['windowTitle'];\n /**\n * Time in milliseconds before performing a write in the history.\n * It prevents from adding too many entries in the history and\n * makes the back button more usable.\n *\n * @default 400\n */\n private readonly writeDelay: Required<\n BrowserHistoryArgs\n >['writeDelay'];\n /**\n * Creates a full URL based on the route state.\n * The storage adaptor maps all syncable keys to the query string of the URL.\n */\n private readonly _createURL: Required<\n BrowserHistoryArgs\n >['createURL'];\n /**\n * Parses the URL into a route state.\n * It should be symmetrical to `createURL`.\n */\n private readonly parseURL: Required<\n BrowserHistoryArgs\n >['parseURL'];\n /**\n * Returns the location to store in the history.\n * @default () => window.location\n */\n private readonly getLocation: Required<\n BrowserHistoryArgs\n >['getLocation'];\n\n private writeTimer?: ReturnType;\n private _onPopState?(event: PopStateEvent): void;\n\n /**\n * Indicates if last action was back/forward in the browser.\n */\n private inPopState: boolean = false;\n\n /**\n * Indicates whether the history router is disposed or not.\n */\n private isDisposed: boolean = false;\n\n /**\n * Indicates the window.history.length before the last call to\n * window.history.pushState (called in `write`).\n * It allows to determine if a `pushState` has been triggered elsewhere,\n * and thus to prevent the `write` method from calling `pushState`.\n */\n private latestAcknowledgedHistory: number = 0;\n\n /**\n * Initializes a new storage provider that syncs the search state to the URL\n * using web APIs (`window.location.pushState` and `onpopstate` event).\n */\n public constructor({\n windowTitle,\n writeDelay = 400,\n createURL,\n parseURL,\n getLocation,\n }: BrowserHistoryArgs) {\n this.windowTitle = windowTitle;\n this.writeTimer = undefined;\n this.writeDelay = writeDelay;\n this._createURL = createURL;\n this.parseURL = parseURL;\n this.getLocation = getLocation;\n\n safelyRunOnBrowser(({ window }) => {\n const title = this.windowTitle && this.windowTitle(this.read());\n setWindowTitle(title);\n\n this.latestAcknowledgedHistory = window.history.length;\n });\n }\n\n /**\n * Reads the URL and returns a syncable UI search state.\n */\n public read(): TRouteState {\n return this.parseURL({ qsModule: qs, location: this.getLocation() });\n }\n\n /**\n * Pushes a search state into the URL.\n */\n public write(routeState: TRouteState): void {\n safelyRunOnBrowser(({ window }) => {\n const url = this.createURL(routeState);\n const title = this.windowTitle && this.windowTitle(routeState);\n\n if (this.writeTimer) {\n clearTimeout(this.writeTimer);\n }\n\n this.writeTimer = setTimeout(() => {\n setWindowTitle(title);\n\n if (this.shouldWrite(url)) {\n window.history.pushState(routeState, title || '', url);\n this.latestAcknowledgedHistory = window.history.length;\n }\n this.inPopState = false;\n this.writeTimer = undefined;\n }, this.writeDelay);\n });\n }\n\n /**\n * Sets a callback on the `onpopstate` event of the history API of the current page.\n * It enables the URL sync to keep track of the changes.\n */\n public onUpdate(callback: (routeState: TRouteState) => void): void {\n this._onPopState = (event) => {\n if (this.writeTimer) {\n clearTimeout(this.writeTimer);\n this.writeTimer = undefined;\n }\n\n this.inPopState = true;\n const routeState = event.state;\n\n // At initial load, the state is read from the URL without update.\n // Therefore the state object is not available.\n // In this case, we fallback and read the URL.\n if (!routeState) {\n callback(this.read());\n } else {\n callback(routeState);\n }\n };\n\n safelyRunOnBrowser(({ window }) => {\n window.addEventListener('popstate', this._onPopState!);\n });\n }\n\n /**\n * Creates a complete URL from a given syncable UI state.\n *\n * It always generates the full URL, not a relative one.\n * This allows to handle cases like using a .\n * See: https://github.com/algolia/instantsearch.js/issues/790\n */\n public createURL(routeState: TRouteState): string {\n return this._createURL({\n qsModule: qs,\n routeState,\n location: this.getLocation(),\n });\n }\n\n /**\n * Removes the event listener and cleans up the URL.\n */\n public dispose(): void {\n this.isDisposed = true;\n\n safelyRunOnBrowser(({ window }) => {\n if (this._onPopState) {\n window.removeEventListener('popstate', this._onPopState);\n }\n });\n\n if (this.writeTimer) {\n clearTimeout(this.writeTimer);\n }\n\n this.write({} as TRouteState);\n }\n\n private shouldWrite(url: string): boolean {\n return safelyRunOnBrowser(({ window }) => {\n // We do want to `pushState` if:\n // - the router is not disposed, IS.js needs to update the URL\n // OR\n // - the last write was from InstantSearch.js\n // (unlike a SPA, where it would have last written)\n const lastPushWasByISAfterDispose = !(\n this.isDisposed &&\n this.latestAcknowledgedHistory !== window.history.length\n );\n\n return (\n // When the last state change was through popstate, the IS.js state changes,\n // but that should not write the URL.\n !this.inPopState &&\n // When the previous pushState after dispose was by IS.js, we want to write the URL.\n lastPushWasByISAfterDispose &&\n // When the URL is the same as the current one, we do not want to write it.\n url !== window.location.href\n );\n });\n }\n}\n\nexport default function historyRouter({\n createURL = ({ qsModule, routeState, location }) => {\n const { protocol, hostname, port = '', pathname, hash } = location;\n const queryString = qsModule.stringify(routeState);\n const portWithPrefix = port === '' ? '' : `:${port}`;\n\n // IE <= 11 has no proper `location.origin` so we cannot rely on it.\n if (!queryString) {\n return `${protocol}//${hostname}${portWithPrefix}${pathname}${hash}`;\n }\n\n return `${protocol}//${hostname}${portWithPrefix}${pathname}?${queryString}${hash}`;\n },\n parseURL = ({ qsModule, location }) => {\n // `qs` by default converts arrays with more than 20 items to an object.\n // We want to avoid this because the data structure manipulated can therefore vary.\n // Setting the limit to `100` seems a good number because the engine's default is 100\n // (it can go up to 1000 but it is very unlikely to select more than 100 items in the UI).\n //\n // Using an `arrayLimit` of `n` allows `n + 1` items.\n //\n // See:\n // - https://github.com/ljharb/qs#parsing-arrays\n // - https://www.algolia.com/doc/api-reference/api-parameters/maxValuesPerFacet/\n return qsModule.parse(location.search.slice(1), { arrayLimit: 99 });\n },\n writeDelay = 400,\n windowTitle,\n getLocation = () => {\n return safelyRunOnBrowser(({ window }) => window.location, {\n fallback: () => {\n throw new Error(\n 'You need to provide `getLocation` to the `history` router in environments where `window` does not exist.'\n );\n },\n });\n },\n}: Partial> = {}): BrowserHistory {\n return new BrowserHistory({\n createURL,\n parseURL,\n writeDelay,\n windowTitle,\n getLocation,\n });\n}\n","import simpleStateMapping from '../lib/stateMappings/simple';\nimport historyRouter from '../lib/routers/history';\nimport type {\n Router,\n StateMapping,\n UiState,\n InternalMiddleware,\n CreateURL,\n} from '../types';\nimport { isEqual } from '../lib/utils';\n\nexport type RouterProps<\n TUiState extends UiState = UiState,\n TRouteState = TUiState\n> = {\n router?: Router;\n // ideally stateMapping should be required if TRouteState is given,\n // but there's no way to check if a generic is provided or the default value.\n stateMapping?: StateMapping;\n};\n\nexport const createRouterMiddleware = <\n TUiState extends UiState = UiState,\n TRouteState = TUiState\n>(\n props: RouterProps = {}\n): InternalMiddleware => {\n const {\n router = historyRouter(),\n // We have to cast simpleStateMapping as a StateMapping.\n // this is needed because simpleStateMapping is StateMapping.\n // While it's only used when UiState and RouteState are the same, unfortunately\n // TypeScript still considers them separate types.\n stateMapping = simpleStateMapping() as unknown as StateMapping<\n TUiState,\n TRouteState\n >,\n } = props;\n\n return ({ instantSearchInstance }) => {\n function topLevelCreateURL(nextState: TUiState) {\n const uiState: TUiState = Object.keys(nextState).reduce(\n (acc, indexId) => ({\n ...acc,\n [indexId]: nextState[indexId],\n }),\n instantSearchInstance.mainIndex.getWidgetUiState(\n {} as TUiState\n )\n );\n\n const route = stateMapping.stateToRoute(uiState);\n\n return router.createURL(route);\n }\n\n // casting to UiState here to keep createURL unaware of custom UiState\n // (as long as it's an object, it's ok)\n instantSearchInstance._createURL = topLevelCreateURL as CreateURL;\n\n let lastRouteState: TRouteState | undefined = undefined;\n\n const initialUiState = instantSearchInstance._initialUiState;\n\n return {\n onStateChange({ uiState }) {\n const routeState = stateMapping.stateToRoute(uiState);\n\n if (\n lastRouteState === undefined ||\n !isEqual(lastRouteState, routeState)\n ) {\n router.write(routeState);\n lastRouteState = routeState;\n }\n },\n\n subscribe() {\n instantSearchInstance._initialUiState = {\n ...initialUiState,\n ...stateMapping.routeToState(router.read()),\n };\n\n router.onUpdate((route) => {\n instantSearchInstance.setUiState(stateMapping.routeToState(route));\n });\n },\n\n unsubscribe() {\n router.dispose();\n },\n };\n };\n};\n","import { safelyRunOnBrowser } from '../lib/utils';\nimport type { InstantSearch, InternalMiddleware, Widget } from '../types';\nimport type { IndexWidget } from '../widgets/index/index';\n\ntype WidgetMetaData = {\n type: string | undefined;\n widgetType: string | undefined;\n params: string[];\n};\n\ntype Payload = {\n widgets: WidgetMetaData[];\n ua?: string;\n};\n\nfunction extractPayload(\n widgets: Array,\n instantSearchInstance: InstantSearch,\n payload: Payload\n) {\n const parent = instantSearchInstance.mainIndex;\n\n const initOptions = {\n instantSearchInstance,\n parent,\n scopedResults: [],\n state: parent.getHelper()!.state,\n helper: parent.getHelper()!,\n createURL: parent.createURL,\n uiState: instantSearchInstance._initialUiState,\n renderState: instantSearchInstance.renderState,\n templatesConfig: instantSearchInstance.templatesConfig,\n searchMetadata: {\n isSearchStalled: instantSearchInstance._isSearchStalled,\n },\n };\n\n widgets.forEach((widget) => {\n let widgetParams: Record = {};\n\n if (widget.getWidgetRenderState) {\n const renderState = widget.getWidgetRenderState(initOptions);\n\n if (renderState && renderState.widgetParams) {\n // casting, as we just earlier checked widgetParams exists, and thus an object\n widgetParams = renderState.widgetParams as Record;\n }\n }\n\n // since we destructure in all widgets, the parameters with defaults are set to \"undefined\"\n const params = Object.keys(widgetParams).filter(\n (key) => widgetParams[key] !== undefined\n );\n\n payload.widgets.push({\n type: widget.$$type,\n widgetType: widget.$$widgetType,\n params,\n });\n\n if (widget.$$type === 'ais.index') {\n extractPayload(\n (widget as IndexWidget).getWidgets(),\n instantSearchInstance,\n payload\n );\n }\n });\n}\n\nexport function isMetadataEnabled() {\n return safelyRunOnBrowser(\n ({ window }) =>\n window.navigator?.userAgent?.indexOf('Algolia Crawler') > -1,\n { fallback: () => false }\n );\n}\n\n/**\n * Exposes the metadata of mounted widgets in a custom\n * `` tag. The metadata per widget is:\n * - applied parameters\n * - widget name\n * - connector name\n */\nexport function createMetadataMiddleware(): InternalMiddleware {\n return ({ instantSearchInstance }) => {\n const payload: Payload = {\n widgets: [],\n };\n const payloadContainer = document.createElement('meta');\n const refNode = document.querySelector('head')!;\n payloadContainer.name = 'instantsearch:widgets';\n\n return {\n onStateChange() {},\n subscribe() {\n // using setTimeout here to delay extraction until widgets have been added in a tick (e.g. Vue)\n setTimeout(() => {\n const client = instantSearchInstance.client as any;\n payload.ua =\n client.transporter && client.transporter.userAgent\n ? client.transporter.userAgent.value\n : client._ua;\n\n extractPayload(\n instantSearchInstance.mainIndex.getWidgets(),\n instantSearchInstance,\n payload\n );\n\n payloadContainer.content = JSON.stringify(payload);\n refNode.appendChild(payloadContainer);\n }, 0);\n },\n\n unsubscribe() {\n payloadContainer.remove();\n },\n };\n };\n}\n","import type { AlgoliaSearchHelper } from 'algoliasearch-helper';\nimport algoliasearchHelper from 'algoliasearch-helper';\nimport EventEmitter from '@algolia/events';\n\nimport type { IndexWidget } from '../widgets/index/index';\nimport index, { isIndexWidget } from '../widgets/index/index';\nimport version from './version';\nimport createHelpers from './createHelpers';\nimport {\n createDocumentationMessageGenerator,\n createDocumentationLink,\n defer,\n noop,\n warning,\n checkIndexUiState,\n} from './utils';\nimport type {\n InsightsClient as AlgoliaInsightsClient,\n SearchClient,\n Widget,\n UiState,\n CreateURL,\n Middleware,\n MiddlewareDefinition,\n RenderState,\n InitialResults,\n} from '../types';\nimport type { RouterProps } from '../middlewares/createRouterMiddleware';\nimport { createRouterMiddleware } from '../middlewares/createRouterMiddleware';\nimport type { InsightsEvent } from '../middlewares/createInsightsMiddleware';\nimport {\n createMetadataMiddleware,\n isMetadataEnabled,\n} from '../middlewares/createMetadataMiddleware';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'instantsearch',\n});\n\nfunction defaultCreateURL() {\n return '#';\n}\n\n/**\n * Global options for an InstantSearch instance.\n */\nexport type InstantSearchOptions<\n TUiState extends UiState = UiState,\n TRouteState = TUiState\n> = {\n /**\n * The name of the main index\n */\n indexName: string;\n\n /**\n * The search client to plug to InstantSearch.js\n *\n * Usage:\n * ```javascript\n * // Using the default Algolia search client\n * instantsearch({\n * indexName: 'indexName',\n * searchClient: algoliasearch('appId', 'apiKey')\n * });\n *\n * // Using a custom search client\n * instantsearch({\n * indexName: 'indexName',\n * searchClient: {\n * search(requests) {\n * // fetch response based on requests\n * return response;\n * },\n * searchForFacetValues(requests) {\n * // fetch response based on requests\n * return response;\n * }\n * }\n * });\n * ```\n */\n searchClient: SearchClient;\n\n /**\n * The locale used to display numbers. This will be passed\n * to `Number.prototype.toLocaleString()`\n */\n numberLocale?: string;\n\n /**\n * A hook that will be called each time a search needs to be done, with the\n * helper as a parameter. It's your responsibility to call `helper.search()`.\n * This option allows you to avoid doing searches at page load for example.\n */\n searchFunction?: (helper: AlgoliaSearchHelper) => void;\n\n /**\n * Function called when the state changes.\n *\n * Using this function makes the instance controlled. This means that you\n * become in charge of updating the UI state with the `setUiState` function.\n */\n onStateChange?: (params: {\n uiState: UiState;\n setUiState(\n uiState: UiState | ((previousUiState: UiState) => UiState)\n ): void;\n }) => void;\n\n /**\n * Injects a `uiState` to the `instantsearch` instance. You can use this option\n * to provide an initial state to a widget. Note that the state is only used\n * for the first search. To unconditionally pass additional parameters to the\n * Algolia API, take a look at the `configure` widget.\n */\n initialUiState?: TUiState;\n\n /**\n * Time before a search is considered stalled. The default is 200ms\n */\n stalledSearchDelay?: number;\n\n /**\n * Router configuration used to save the UI State into the URL or any other\n * client side persistence. Passing `true` will use the default URL options.\n */\n routing?: RouterProps | boolean;\n\n /**\n * the instance of search-insights to use for sending insights events inside\n * widgets like `hits`.\n *\n * @deprecated This property will be still supported in 4.x releases, but not further. It is replaced by the `insights` middleware. For more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/\n */\n insightsClient?: AlgoliaInsightsClient;\n};\n\n/**\n * The actual implementation of the InstantSearch. This is\n * created using the `instantsearch` factory function.\n * It emits the 'render' event every time a search is done\n */\nclass InstantSearch<\n TUiState extends UiState = UiState,\n TRouteState = TUiState\n> extends EventEmitter {\n public client: InstantSearchOptions['searchClient'];\n public indexName: string;\n public insightsClient: AlgoliaInsightsClient | null;\n public onStateChange: InstantSearchOptions['onStateChange'] | null = null;\n public helper: AlgoliaSearchHelper | null;\n public mainHelper: AlgoliaSearchHelper | null;\n public mainIndex: IndexWidget;\n public started: boolean;\n public templatesConfig: Record;\n public renderState: RenderState = {};\n public _stalledSearchDelay: number;\n public _searchStalledTimer: any;\n public _isSearchStalled: boolean;\n public _initialUiState: UiState;\n public _initialResults: InitialResults | null;\n public _createURL: CreateURL;\n public _searchFunction?: InstantSearchOptions['searchFunction'];\n public _mainHelperSearch?: AlgoliaSearchHelper['search'];\n public middleware: Array<{\n creator: Middleware;\n instance: MiddlewareDefinition;\n }> = [];\n public sendEventToInsights: (event: InsightsEvent) => void;\n\n public constructor(options: InstantSearchOptions) {\n super();\n\n const {\n indexName = null,\n numberLocale,\n initialUiState = {},\n routing = null,\n searchFunction,\n stalledSearchDelay = 200,\n searchClient = null,\n insightsClient = null,\n onStateChange = null,\n } = options;\n\n if (indexName === null) {\n throw new Error(withUsage('The `indexName` option is required.'));\n }\n\n if (searchClient === null) {\n throw new Error(withUsage('The `searchClient` option is required.'));\n }\n\n if (typeof (searchClient as any).search !== 'function') {\n throw new Error(\n `The \\`searchClient\\` must implement a \\`search\\` method.\n\nSee: https://www.algolia.com/doc/guides/building-search-ui/going-further/backend-search/in-depth/backend-instantsearch/js/`\n );\n }\n\n if (typeof searchClient.addAlgoliaAgent === 'function') {\n searchClient.addAlgoliaAgent(`instantsearch.js (${version})`);\n }\n\n warning(\n insightsClient === null,\n `\\`insightsClient\\` property has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the \\`insights\\` middleware.\n\nFor more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/`\n );\n\n if (insightsClient && typeof insightsClient !== 'function') {\n throw new Error(\n withUsage('The `insightsClient` option should be a function.')\n );\n }\n\n warning(\n !(options as any).searchParameters,\n `The \\`searchParameters\\` option is deprecated and will not be supported in InstantSearch.js 4.x.\n\nYou can replace it with the \\`configure\\` widget:\n\n\\`\\`\\`\nsearch.addWidgets([\n configure(${JSON.stringify((options as any).searchParameters, null, 2)})\n]);\n\\`\\`\\`\n\nSee ${createDocumentationLink({\n name: 'configure',\n })}`\n );\n\n this.client = searchClient;\n this.insightsClient = insightsClient;\n this.indexName = indexName;\n this.helper = null;\n this.mainHelper = null;\n this.mainIndex = index({\n indexName,\n });\n this.onStateChange = onStateChange;\n\n this.started = false;\n this.templatesConfig = {\n helpers: createHelpers({ numberLocale }),\n compileOptions: {},\n };\n\n this._stalledSearchDelay = stalledSearchDelay;\n this._searchStalledTimer = null;\n this._isSearchStalled = false;\n\n this._createURL = defaultCreateURL;\n this._initialUiState = initialUiState;\n this._initialResults = null;\n\n if (searchFunction) {\n this._searchFunction = searchFunction;\n }\n\n this.sendEventToInsights = noop;\n\n if (routing) {\n const routerOptions = typeof routing === 'boolean' ? undefined : routing;\n this.use(createRouterMiddleware(routerOptions));\n }\n\n if (isMetadataEnabled()) {\n this.use(createMetadataMiddleware());\n }\n }\n\n /**\n * Hooks a middleware into the InstantSearch lifecycle.\n */\n public use(...middleware: Middleware[]): this {\n const newMiddlewareList = middleware.map((fn) => {\n const newMiddleware = {\n subscribe: noop,\n unsubscribe: noop,\n onStateChange: noop,\n ...fn({ instantSearchInstance: this }),\n };\n this.middleware.push({\n creator: fn,\n instance: newMiddleware,\n });\n return newMiddleware;\n });\n\n // If the instance has already started, we directly subscribe the\n // middleware so they're notified of changes.\n if (this.started) {\n newMiddlewareList.forEach((m) => {\n m.subscribe();\n });\n }\n\n return this;\n }\n\n /**\n * Removes a middleware from the InstantSearch lifecycle.\n */\n public unuse(...middlewareToUnuse: Middleware[]): this {\n this.middleware\n .filter((m) => middlewareToUnuse.includes(m.creator))\n .forEach((m) => m.instance.unsubscribe());\n\n this.middleware = this.middleware.filter(\n (m) => !middlewareToUnuse.includes(m.creator)\n );\n\n return this;\n }\n\n // @major we shipped with EXPERIMENTAL_use, but have changed that to just `use` now\n public EXPERIMENTAL_use(...middleware: Middleware[]): this {\n warning(\n false,\n 'The middleware API is now considered stable, so we recommend replacing `EXPERIMENTAL_use` with `use` before upgrading to the next major version.'\n );\n\n return this.use(...middleware);\n }\n\n /**\n * Adds a widget to the search instance.\n * A widget can be added either before or after InstantSearch has started.\n * @param widget The widget to add to InstantSearch.\n *\n * @deprecated This method will still be supported in 4.x releases, but not further. It is replaced by `addWidgets([widget])`.\n */\n public addWidget(widget: Widget) {\n warning(\n false,\n 'addWidget will still be supported in 4.x releases, but not further. It is replaced by `addWidgets([widget])`'\n );\n\n return this.addWidgets([widget]);\n }\n\n /**\n * Adds multiple widgets to the search instance.\n * Widgets can be added either before or after InstantSearch has started.\n * @param widgets The array of widgets to add to InstantSearch.\n */\n public addWidgets(widgets: Array) {\n if (!Array.isArray(widgets)) {\n throw new Error(\n withUsage(\n 'The `addWidgets` method expects an array of widgets. Please use `addWidget`.'\n )\n );\n }\n\n if (\n widgets.some(\n (widget) =>\n typeof widget.init !== 'function' &&\n typeof widget.render !== 'function'\n )\n ) {\n throw new Error(\n withUsage(\n 'The widget definition expects a `render` and/or an `init` method.'\n )\n );\n }\n\n this.mainIndex.addWidgets(widgets);\n\n return this;\n }\n\n /**\n * Removes a widget from the search instance.\n * @deprecated This method will still be supported in 4.x releases, but not further. It is replaced by `removeWidgets([widget])`\n * @param widget The widget instance to remove from InstantSearch.\n *\n * The widget must implement a `dispose()` method to clear its state.\n */\n public removeWidget(widget: Widget | IndexWidget) {\n warning(\n false,\n 'removeWidget will still be supported in 4.x releases, but not further. It is replaced by `removeWidgets([widget])`'\n );\n\n return this.removeWidgets([widget]);\n }\n\n /**\n * Removes multiple widgets from the search instance.\n * @param widgets Array of widgets instances to remove from InstantSearch.\n *\n * The widgets must implement a `dispose()` method to clear their states.\n */\n public removeWidgets(widgets: Array) {\n if (!Array.isArray(widgets)) {\n throw new Error(\n withUsage(\n 'The `removeWidgets` method expects an array of widgets. Please use `removeWidget`.'\n )\n );\n }\n\n if (widgets.some((widget) => typeof widget.dispose !== 'function')) {\n throw new Error(\n withUsage('The widget definition expects a `dispose` method.')\n );\n }\n\n this.mainIndex.removeWidgets(widgets);\n\n return this;\n }\n\n /**\n * Ends the initialization of InstantSearch.js and triggers the\n * first search. This method should be called after all widgets have been added\n * to the instance of InstantSearch.js. InstantSearch.js also supports adding and removing\n * widgets after the start as an **EXPERIMENTAL** feature.\n */\n public start() {\n if (this.started) {\n throw new Error(\n withUsage('The `start` method has already been called once.')\n );\n }\n\n // This Helper is used for the queries, we don't care about its state. The\n // states are managed at the `index` level. We use this Helper to create\n // DerivedHelper scoped into the `index` widgets.\n // In Vue InstantSearch' hydrate, a main helper gets set before start, so\n // we need to respect this helper as a way to keep all listeners correct.\n const mainHelper =\n this.mainHelper || algoliasearchHelper(this.client, this.indexName);\n\n mainHelper.search = () => {\n // This solution allows us to keep the exact same API for the users but\n // under the hood, we have a different implementation. It should be\n // completely transparent for the rest of the codebase. Only this module\n // is impacted.\n return mainHelper.searchOnlyWithDerivedHelpers();\n };\n\n if (this._searchFunction) {\n // this client isn't used to actually search, but required for the helper\n // to not throw errors\n const fakeClient = {\n search: () => new Promise(noop),\n } as any as SearchClient;\n\n this._mainHelperSearch = mainHelper.search.bind(mainHelper);\n mainHelper.search = () => {\n const mainIndexHelper = this.mainIndex.getHelper();\n const searchFunctionHelper = algoliasearchHelper(\n fakeClient,\n mainIndexHelper!.state.index,\n mainIndexHelper!.state\n );\n searchFunctionHelper.once('search', ({ state }) => {\n mainIndexHelper!.overrideStateWithoutTriggeringChangeEvent(state);\n this._mainHelperSearch!();\n });\n // Forward state changes from `searchFunctionHelper` to `mainIndexHelper`\n searchFunctionHelper.on('change', ({ state }) => {\n mainIndexHelper!.setState(state);\n });\n this._searchFunction!(searchFunctionHelper);\n return mainHelper;\n };\n }\n\n // Only the \"main\" Helper emits the `error` event vs the one for `search`\n // and `results` that are also emitted on the derived one.\n mainHelper.on('error', ({ error }) => {\n // If an error is emitted, it is re-thrown by events. In previous versions\n // we emitted {error}, which is thrown as:\n // \"Uncaught, unspecified \\\"error\\\" event. ([object Object])\"\n // To avoid breaking changes, we make the error available in both\n // `error` and `error.error`\n // @MAJOR emit only error\n (error as any).error = error;\n this.emit('error', error);\n });\n\n this.mainHelper = mainHelper;\n\n this.middleware.forEach(({ instance }) => {\n instance.subscribe();\n });\n\n this.mainIndex.init({\n instantSearchInstance: this,\n parent: null,\n uiState: this._initialUiState,\n });\n\n if (this._initialResults) {\n const originalScheduleSearch = this.scheduleSearch;\n // We don't schedule a first search when initial results are provided\n // because we already have the results to render. This skips the initial\n // network request on the browser on `start`.\n this.scheduleSearch = defer(noop);\n // We also skip the initial network request when widgets are dynamically\n // added in the first tick (that's the case in all the framework-based flavors).\n // When we add a widget to `index`, it calls `scheduleSearch`. We can rely\n // on our `defer` util to restore the original `scheduleSearch` value once\n // widgets are added to hook back to the regular lifecycle.\n defer(() => {\n this.scheduleSearch = originalScheduleSearch;\n })();\n } else {\n this.scheduleSearch();\n }\n\n // Keep the previous reference for legacy purpose, some pattern use\n // the direct Helper access `search.helper` (e.g multi-index).\n this.helper = this.mainIndex.getHelper();\n\n // track we started the search if we add more widgets,\n // to init them directly after add\n this.started = true;\n }\n\n /**\n * Removes all widgets without triggering a search afterwards. This is an **EXPERIMENTAL** feature,\n * if you find an issue with it, please\n * [open an issue](https://github.com/algolia/instantsearch.js/issues/new?title=Problem%20with%20dispose).\n * @return {undefined} This method does not return anything\n */\n public dispose(): void {\n this.scheduleSearch.cancel();\n this.scheduleRender.cancel();\n clearTimeout(this._searchStalledTimer);\n\n this.removeWidgets(this.mainIndex.getWidgets());\n this.mainIndex.dispose();\n\n // You can not start an instance two times, therefore a disposed instance\n // needs to set started as false otherwise this can not be restarted at a\n // later point.\n this.started = false;\n\n // The helper needs to be reset to perform the next search from a fresh state.\n // If not reset, it would use the state stored before calling `dispose()`.\n this.removeAllListeners();\n this.mainHelper!.removeAllListeners();\n this.mainHelper = null;\n this.helper = null;\n\n this.middleware.forEach(({ instance }) => {\n instance.unsubscribe();\n });\n }\n\n public scheduleSearch = defer(() => {\n if (this.started) {\n this.mainHelper!.search();\n }\n });\n\n public scheduleRender = defer(() => {\n if (!this.mainHelper!.hasPendingRequests()) {\n clearTimeout(this._searchStalledTimer);\n this._searchStalledTimer = null;\n this._isSearchStalled = false;\n }\n\n this.mainIndex.render({\n instantSearchInstance: this,\n });\n\n this.emit('render');\n });\n\n public scheduleStalledRender() {\n if (!this._searchStalledTimer) {\n this._searchStalledTimer = setTimeout(() => {\n this._isSearchStalled = true;\n this.scheduleRender();\n }, this._stalledSearchDelay);\n }\n }\n\n public setUiState(\n uiState: UiState | ((previousUiState: UiState) => UiState)\n ): void {\n if (!this.mainHelper) {\n throw new Error(\n withUsage('The `start` method needs to be called before `setUiState`.')\n );\n }\n\n // We refresh the index UI state to update the local UI state that the\n // main index passes to the function form of `setUiState`.\n this.mainIndex.refreshUiState();\n const nextUiState =\n typeof uiState === 'function'\n ? uiState(this.mainIndex.getWidgetUiState({}))\n : uiState;\n\n const setIndexHelperState = (indexWidget: IndexWidget) => {\n const nextIndexUiState = nextUiState[indexWidget.getIndexId()] || {};\n\n if (__DEV__) {\n checkIndexUiState({\n index: indexWidget,\n indexUiState: nextIndexUiState,\n });\n }\n\n indexWidget.getHelper()!.setState(\n indexWidget.getWidgetSearchParameters(indexWidget.getHelper()!.state, {\n uiState: nextIndexUiState,\n })\n );\n\n indexWidget\n .getWidgets()\n .filter(isIndexWidget)\n .forEach(setIndexHelperState);\n };\n\n setIndexHelperState(this.mainIndex);\n\n this.scheduleSearch();\n this.onInternalStateChange();\n }\n\n public getUiState(): UiState {\n if (this.started) {\n // We refresh the index UI state to make sure changes from `refine` are taken in account\n this.mainIndex.refreshUiState();\n }\n\n return this.mainIndex.getWidgetUiState({});\n }\n\n public onInternalStateChange = defer(() => {\n const nextUiState = this.mainIndex.getWidgetUiState({});\n\n this.middleware.forEach(({ instance }) => {\n instance.onStateChange({\n uiState: nextUiState,\n });\n });\n });\n\n public createURL(nextState: UiState = {}): string {\n if (!this.started) {\n throw new Error(\n withUsage('The `start` method needs to be called before `createURL`.')\n );\n }\n\n return this._createURL(nextState);\n }\n\n public refresh() {\n if (!this.mainHelper) {\n throw new Error(\n withUsage('The `start` method needs to be called before `refresh`.')\n );\n }\n\n this.mainHelper.clearCache().search();\n }\n}\n\nexport default InstantSearch;\n","import type { AlgoliaSearchHelper, SearchResults } from 'algoliasearch-helper';\nimport {\n checkRendering,\n clearRefinements,\n getRefinements,\n createDocumentationMessageGenerator,\n noop,\n uniq,\n mergeSearchParameters,\n} from '../../lib/utils';\nimport type {\n TransformItems,\n CreateURL,\n Connector,\n WidgetRenderState,\n ScopedResult,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'clear-refinements',\n connector: true,\n});\n\nexport type ClearRefinementsConnectorParams = {\n /**\n * The attributes to include in the refinements to clear (all by default). Cannot be used with `excludedAttributes`.\n */\n includedAttributes?: string[];\n\n /**\n * The attributes to exclude from the refinements to clear. Cannot be used with `includedAttributes`.\n */\n excludedAttributes?: string[];\n\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n};\n\nexport type ClearRefinementsRenderState = {\n /**\n * Triggers the clear of all the currently refined values.\n */\n refine: () => void;\n\n /**\n * Indicates if search state is refined.\n * @deprecated prefer reading canRefine\n */\n hasRefinements: boolean;\n\n /**\n * Indicates if search state can be refined.\n */\n canRefine: boolean;\n\n /**\n * Creates a url for the next state when refinements are cleared.\n */\n createURL: CreateURL;\n};\n\nexport type ClearRefinementsWidgetDescription = {\n $$type: 'ais.clearRefinements';\n renderState: ClearRefinementsRenderState;\n indexRenderState: {\n clearRefinements: WidgetRenderState<\n ClearRefinementsRenderState,\n ClearRefinementsConnectorParams\n >;\n };\n};\n\nexport type ClearRefinementsConnector = Connector<\n ClearRefinementsWidgetDescription,\n ClearRefinementsConnectorParams\n>;\n\ntype AttributesToClear = {\n helper: AlgoliaSearchHelper;\n items: string[];\n};\n\nconst connectClearRefinements: ClearRefinementsConnector =\n function connectClearRefinements(renderFn, unmountFn = noop) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n includedAttributes = [],\n excludedAttributes = ['query'],\n transformItems = ((items) => items) as NonNullable<\n ClearRefinementsConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n if (\n widgetParams &&\n widgetParams.includedAttributes &&\n widgetParams.excludedAttributes\n ) {\n throw new Error(\n withUsage(\n 'The options `includedAttributes` and `excludedAttributes` cannot be used together.'\n )\n );\n }\n\n type ConnectorState = {\n refine(): void;\n createURL(): string;\n attributesToClear: AttributesToClear[];\n };\n\n const connectorState: ConnectorState = {\n refine: noop,\n createURL: () => '',\n attributesToClear: [],\n };\n\n const cachedRefine = () => connectorState.refine();\n const cachedCreateURL = () => connectorState.createURL();\n\n return {\n $$type: 'ais.clearRefinements',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose() {\n unmountFn();\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n clearRefinements: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ createURL, scopedResults, results }) {\n connectorState.attributesToClear = scopedResults.reduce<\n Array>\n >((attributesToClear, scopedResult) => {\n return attributesToClear.concat(\n getAttributesToClear({\n scopedResult,\n includedAttributes,\n excludedAttributes,\n transformItems,\n results,\n })\n );\n }, []);\n\n connectorState.refine = () => {\n connectorState.attributesToClear.forEach(\n ({ helper: indexHelper, items }) => {\n indexHelper\n .setState(\n clearRefinements({\n helper: indexHelper,\n attributesToClear: items,\n })\n )\n .search();\n }\n );\n };\n\n connectorState.createURL = () =>\n createURL(\n mergeSearchParameters(\n ...connectorState.attributesToClear.map(\n ({ helper: indexHelper, items }) => {\n return clearRefinements({\n helper: indexHelper,\n attributesToClear: items,\n });\n }\n )\n )\n );\n\n const canRefine = connectorState.attributesToClear.some(\n (attributeToClear) => attributeToClear.items.length > 0\n );\n\n return {\n canRefine,\n hasRefinements: canRefine,\n refine: cachedRefine,\n createURL: cachedCreateURL,\n widgetParams,\n };\n },\n };\n };\n };\n\nfunction getAttributesToClear({\n scopedResult,\n includedAttributes,\n excludedAttributes,\n transformItems,\n results,\n}: {\n scopedResult: ScopedResult;\n includedAttributes: string[];\n excludedAttributes: string[];\n transformItems: TransformItems;\n results: SearchResults | undefined;\n}): AttributesToClear {\n const includesQuery =\n includedAttributes.indexOf('query') !== -1 ||\n excludedAttributes.indexOf('query') === -1;\n\n return {\n helper: scopedResult.helper,\n items: transformItems(\n uniq(\n getRefinements(\n scopedResult.results,\n scopedResult.helper.state,\n includesQuery\n )\n .map((refinement) => refinement.attribute)\n .filter(\n (attribute) =>\n // If the array is empty (default case), we keep all the attributes\n includedAttributes.length === 0 ||\n // Otherwise, only add the specified attributes\n includedAttributes.indexOf(attribute) !== -1\n )\n .filter(\n (attribute) =>\n // If the query is included, we ignore the default `excludedAttributes = ['query']`\n (attribute === 'query' && includesQuery) ||\n // Otherwise, ignore the excluded attributes\n excludedAttributes.indexOf(attribute) === -1\n )\n ),\n { results }\n ),\n };\n}\n\nexport default connectClearRefinements;\n","import type {\n AlgoliaSearchHelper,\n SearchParameters,\n SearchResults,\n} from 'algoliasearch-helper';\nimport {\n getRefinements,\n checkRendering,\n createDocumentationMessageGenerator,\n noop,\n warning,\n} from '../../lib/utils';\nimport type {\n Refinement,\n FacetRefinement,\n NumericRefinement,\n} from '../../lib/utils/getRefinements';\nimport type {\n Connector,\n TransformItems,\n CreateURL,\n WidgetRenderState,\n} from '../../types';\n\nexport type CurrentRefinementsConnectorParamsRefinement = {\n /**\n * The attribute on which the refinement is applied.\n */\n attribute: string;\n\n /**\n * The type of the refinement.\n */\n type:\n | 'facet'\n | 'exclude'\n | 'disjunctive'\n | 'hierarchical'\n | 'numeric'\n | 'query'\n | 'tag';\n\n /**\n * The raw value of the refinement.\n */\n value: string | number;\n\n /**\n * The label of the refinement to display.\n */\n label: string;\n\n /**\n * The value of the operator (only if applicable).\n */\n operator?: string;\n\n /**\n * The number of found items (only if applicable).\n */\n count?: number;\n\n /**\n * Whether the count is exhaustive (only if applicable).\n */\n exhaustive?: boolean;\n};\n\nexport type CurrentRefinementsConnectorParamsItem = {\n /**\n * The index name.\n */\n indexName: string;\n\n /**\n * The attribute on which the refinement is applied.\n */\n attribute: string;\n\n /**\n * The textual representation of this attribute.\n */\n label: string;\n\n /**\n * Currently applied refinements.\n */\n refinements: CurrentRefinementsConnectorParamsRefinement[];\n\n /**\n * Removes the given refinement and triggers a new search.\n */\n refine(refinement: CurrentRefinementsConnectorParamsRefinement): void;\n};\n\nexport type CurrentRefinementsConnectorParams = {\n /**\n * The attributes to include in the widget (all by default).\n * Cannot be used with `excludedAttributes`.\n *\n * @default `[]`\n */\n includedAttributes?: string[];\n\n /**\n * The attributes to exclude from the widget.\n * Cannot be used with `includedAttributes`.\n *\n * @default `['query']`\n */\n excludedAttributes?: string[];\n\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n};\n\nexport type CurrentRefinementsRenderState = {\n /**\n * All the currently refined items, grouped by attribute.\n */\n items: CurrentRefinementsConnectorParamsItem[];\n\n /**\n * Indicates if search state can be refined.\n */\n canRefine: boolean;\n\n /**\n * Removes the given refinement and triggers a new search.\n */\n refine(refinement: CurrentRefinementsConnectorParamsRefinement): void;\n\n /**\n * Generates a URL for the next state.\n */\n createURL: CreateURL;\n};\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'current-refinements',\n connector: true,\n});\n\nexport type CurrentRefinementsWidgetDescription = {\n $$type: 'ais.currentRefinements';\n renderState: CurrentRefinementsRenderState;\n indexRenderState: {\n currentRefinements: WidgetRenderState<\n CurrentRefinementsRenderState,\n CurrentRefinementsConnectorParams\n >;\n };\n};\n\nexport type CurrentRefinementsConnector = Connector<\n CurrentRefinementsWidgetDescription,\n CurrentRefinementsConnectorParams\n>;\n\nconst connectCurrentRefinements: CurrentRefinementsConnector =\n function connectCurrentRefinements(renderFn, unmountFn = noop) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n if (\n (widgetParams || {}).includedAttributes &&\n (widgetParams || {}).excludedAttributes\n ) {\n throw new Error(\n withUsage(\n 'The options `includedAttributes` and `excludedAttributes` cannot be used together.'\n )\n );\n }\n\n const {\n includedAttributes,\n excludedAttributes = ['query'],\n transformItems = ((items) => items) as NonNullable<\n CurrentRefinementsConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n return {\n $$type: 'ais.currentRefinements',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose() {\n unmountFn();\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n currentRefinements: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ results, scopedResults, createURL, helper }) {\n function getItems() {\n if (!results) {\n return transformItems(\n getRefinementsItems({\n results: {},\n helper,\n includedAttributes,\n excludedAttributes,\n }),\n { results }\n );\n }\n\n return scopedResults.reduce<\n CurrentRefinementsConnectorParamsItem[]\n >((accResults, scopedResult) => {\n return accResults.concat(\n transformItems(\n getRefinementsItems({\n results: scopedResult.results,\n helper: scopedResult.helper,\n includedAttributes,\n excludedAttributes,\n }),\n { results }\n )\n );\n }, []);\n }\n\n const items = getItems();\n\n return {\n items,\n canRefine: items.length > 0,\n refine: (refinement) => clearRefinement(helper, refinement),\n createURL: (refinement) =>\n createURL(clearRefinementFromState(helper.state, refinement)),\n widgetParams,\n };\n },\n };\n };\n };\n\nfunction getRefinementsItems({\n results,\n helper,\n includedAttributes,\n excludedAttributes,\n}: {\n results: SearchResults | Record;\n helper: AlgoliaSearchHelper;\n includedAttributes: CurrentRefinementsConnectorParams['includedAttributes'];\n excludedAttributes: CurrentRefinementsConnectorParams['excludedAttributes'];\n}): CurrentRefinementsConnectorParamsItem[] {\n const includesQuery =\n (includedAttributes || []).indexOf('query') !== -1 ||\n (excludedAttributes || []).indexOf('query') === -1;\n\n const filterFunction = includedAttributes\n ? (item: CurrentRefinementsConnectorParamsRefinement) =>\n includedAttributes.indexOf(item.attribute) !== -1\n : (item: CurrentRefinementsConnectorParamsRefinement) =>\n excludedAttributes!.indexOf(item.attribute) === -1;\n\n const items = getRefinements(results, helper.state, includesQuery)\n .map(normalizeRefinement)\n .filter(filterFunction);\n\n return items.reduce(\n (allItems, currentItem) => [\n ...allItems.filter((item) => item.attribute !== currentItem.attribute),\n {\n indexName: helper.state.index,\n attribute: currentItem.attribute,\n label: currentItem.attribute,\n refinements: items\n .filter((result) => result.attribute === currentItem.attribute)\n // We want to keep the order of refinements except the numeric ones.\n .sort((a, b) =>\n a.type === 'numeric' ? (a.value as number) - (b.value as number) : 0\n ),\n refine: (refinement) => clearRefinement(helper, refinement),\n },\n ],\n []\n );\n}\n\nfunction clearRefinementFromState(\n state: SearchParameters,\n refinement: CurrentRefinementsConnectorParamsRefinement\n): SearchParameters {\n switch (refinement.type) {\n case 'facet':\n return state.removeFacetRefinement(\n refinement.attribute,\n String(refinement.value)\n );\n case 'disjunctive':\n return state.removeDisjunctiveFacetRefinement(\n refinement.attribute,\n String(refinement.value)\n );\n case 'hierarchical':\n return state.removeHierarchicalFacetRefinement(refinement.attribute);\n case 'exclude':\n return state.removeExcludeRefinement(\n refinement.attribute,\n String(refinement.value)\n );\n case 'numeric':\n return state.removeNumericRefinement(\n refinement.attribute,\n refinement.operator,\n String(refinement.value)\n );\n case 'tag':\n return state.removeTagRefinement(String(refinement.value));\n case 'query':\n return state.setQueryParameter('query', '');\n default:\n warning(\n false,\n `The refinement type \"${refinement.type}\" does not exist and cannot be cleared from the current refinements.`\n );\n return state;\n }\n}\n\nfunction clearRefinement(\n helper: AlgoliaSearchHelper,\n refinement: CurrentRefinementsConnectorParamsRefinement\n): void {\n helper.setState(clearRefinementFromState(helper.state, refinement)).search();\n}\n\nfunction getOperatorSymbol(operator: SearchParameters.Operator): string {\n switch (operator) {\n case '>=':\n return '≥';\n case '<=':\n return '≤';\n default:\n return operator;\n }\n}\n\nfunction normalizeRefinement(\n refinement: Refinement\n): CurrentRefinementsConnectorParamsRefinement {\n const value = getValue(refinement);\n const label = (refinement as NumericRefinement).operator\n ? `${getOperatorSymbol(\n (refinement as NumericRefinement).operator as SearchParameters.Operator\n )} ${refinement.name}`\n : refinement.name;\n\n const normalizedRefinement: CurrentRefinementsConnectorParamsRefinement = {\n attribute: refinement.attribute,\n type: refinement.type,\n value,\n label,\n };\n\n if ((refinement as NumericRefinement).operator !== undefined) {\n normalizedRefinement.operator = (refinement as NumericRefinement).operator;\n }\n if ((refinement as FacetRefinement).count !== undefined) {\n normalizedRefinement.count = (refinement as FacetRefinement).count;\n }\n if ((refinement as FacetRefinement).exhaustive !== undefined) {\n normalizedRefinement.exhaustive = (\n refinement as FacetRefinement\n ).exhaustive;\n }\n\n return normalizedRefinement;\n}\n\nfunction getValue(refinement: Refinement) {\n if (refinement.type === 'numeric') {\n return Number(refinement.name);\n }\n\n if ('escapedValue' in refinement) {\n return refinement.escapedValue;\n }\n\n return refinement.name;\n}\n\nexport default connectCurrentRefinements;\n","import type { SendEventForFacet } from '../../lib/utils';\nimport {\n checkRendering,\n warning,\n createDocumentationMessageGenerator,\n createSendEventForFacet,\n isEqual,\n noop,\n} from '../../lib/utils';\nimport type { SearchResults } from 'algoliasearch-helper';\nimport type {\n Connector,\n CreateURL,\n TransformItems,\n RenderOptions,\n Widget,\n SortBy,\n WidgetRenderState,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'hierarchical-menu',\n connector: true,\n});\n\nconst DEFAULT_SORT = ['name:asc'];\n\nexport type HierarchicalMenuItem = {\n /**\n * Value of the menu item.\n */\n value: string;\n /**\n * Human-readable value of the menu item.\n */\n label: string;\n /**\n * Number of matched results after refinement is applied.\n */\n count: number;\n /**\n * Indicates if the refinement is applied.\n */\n isRefined: boolean;\n /**\n * n+1 level of items, same structure HierarchicalMenuItem\n */\n data: HierarchicalMenuItem[] | null;\n};\n\nexport type HierarchicalMenuConnectorParams = {\n /**\n * Attributes to use to generate the hierarchy of the menu.\n */\n attributes: string[];\n /**\n * Separator used in the attributes to separate level values.\n */\n separator?: string;\n /**\n * Prefix path to use if the first level is not the root level.\n */\n rootPath?: string | null;\n /**\n * Show the siblings of the selected parent levels of the current refined value. This\n * does not impact the root level.\n */\n showParentLevel?: boolean;\n /**\n * Max number of values to display.\n */\n limit?: number;\n /**\n * Whether to display the \"show more\" button.\n */\n showMore?: boolean;\n /**\n * Max number of values to display when showing more.\n */\n showMoreLimit?: number;\n /**\n * How to sort refinements. Possible values: `count|isRefined|name:asc|name:desc`.\n * You can also use a sort function that behaves like the standard Javascript [compareFunction](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#Syntax).\n *\n * If a facetOrdering is set in the index settings, it is used when sortBy isn't passed\n */\n sortBy?: SortBy;\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n};\n\nexport type HierarchicalMenuRenderState = {\n /**\n * Creates an url for the next state for a clicked item.\n */\n createURL: CreateURL;\n /**\n * Values to be rendered.\n */\n items: HierarchicalMenuItem[];\n /**\n * Sets the path of the hierarchical filter and triggers a new search.\n */\n refine: (value: string) => void;\n /**\n * Indicates if search state can be refined.\n */\n canRefine: boolean;\n /**\n * True if the menu is displaying all the menu items.\n */\n isShowingMore: boolean;\n /**\n * Toggles the number of values displayed between `limit` and `showMoreLimit`.\n */\n toggleShowMore: () => void;\n /**\n * `true` if the toggleShowMore button can be activated (enough items to display more or\n * already displaying more than `limit` items)\n */\n canToggleShowMore: boolean;\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEventForFacet;\n};\n\nexport type HierarchicalMenuWidgetDescription = {\n $$type: 'ais.hierarchicalMenu';\n renderState: HierarchicalMenuRenderState;\n indexRenderState: {\n hierarchicalMenu: {\n [rootAttribute: string]: WidgetRenderState<\n HierarchicalMenuRenderState,\n HierarchicalMenuConnectorParams\n >;\n };\n };\n indexUiState: {\n hierarchicalMenu: {\n [rootAttribute: string]: string[];\n };\n };\n};\n\nexport type HierarchicalMenuConnector = Connector<\n HierarchicalMenuWidgetDescription,\n HierarchicalMenuConnectorParams\n>;\n\n/**\n * **HierarchicalMenu** connector provides the logic to build a custom widget\n * that will give the user the ability to explore facets in a tree-like structure.\n *\n * This is commonly used for multi-level categorization of products on e-commerce\n * websites. From a UX point of view, we suggest not displaying more than two\n * levels deep.\n *\n * @type {Connector}\n * @param {function(HierarchicalMenuRenderingOptions, boolean)} renderFn Rendering function for the custom **HierarchicalMenu** widget.\n * @param {function} unmountFn Unmount function called when the widget is disposed.\n * @return {function(CustomHierarchicalMenuWidgetParams)} Re-usable widget factory for a custom **HierarchicalMenu** widget.\n */\nconst connectHierarchicalMenu: HierarchicalMenuConnector =\n function connectHierarchicalMenu(renderFn, unmountFn = noop) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n attributes,\n separator = ' > ',\n rootPath = null,\n showParentLevel = true,\n limit = 10,\n showMore = false,\n showMoreLimit = 20,\n sortBy = DEFAULT_SORT,\n transformItems = ((items) => items) as NonNullable<\n HierarchicalMenuConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n if (\n !attributes ||\n !Array.isArray(attributes) ||\n attributes.length === 0\n ) {\n throw new Error(\n withUsage('The `attributes` option expects an array of strings.')\n );\n }\n\n if (showMore === true && showMoreLimit <= limit) {\n throw new Error(\n withUsage('The `showMoreLimit` option must be greater than `limit`.')\n );\n }\n\n type ThisWidget = Widget<\n HierarchicalMenuWidgetDescription & {\n widgetParams: typeof widgetParams;\n }\n >;\n\n // we need to provide a hierarchicalFacet name for the search state\n // so that we can always map $hierarchicalFacetName => real attributes\n // we use the first attribute name\n const [hierarchicalFacetName] = attributes;\n let sendEvent: HierarchicalMenuRenderState['sendEvent'];\n\n // Provide the same function to the `renderFn` so that way the user\n // has to only bind it once when `isFirstRendering` for instance\n let toggleShowMore = () => {};\n function cachedToggleShowMore() {\n toggleShowMore();\n }\n\n let _refine: HierarchicalMenuRenderState['refine'] | undefined;\n\n let isShowingMore = false;\n\n function createToggleShowMore(\n renderOptions: RenderOptions,\n widget: ThisWidget\n ) {\n return () => {\n isShowingMore = !isShowingMore;\n widget.render!(renderOptions);\n };\n }\n\n function getLimit() {\n return isShowingMore ? showMoreLimit : limit;\n }\n\n function _prepareFacetValues(\n facetValues: SearchResults.HierarchicalFacet[]\n ): HierarchicalMenuItem[] {\n return facetValues\n .slice(0, getLimit())\n .map(\n ({ name: label, escapedValue: value, data, path, ...subValue }) => {\n const item: HierarchicalMenuItem = {\n ...subValue,\n value,\n label,\n data: null,\n };\n if (Array.isArray(data)) {\n item.data = _prepareFacetValues(data);\n }\n return item;\n }\n );\n }\n\n return {\n $$type: 'ais.hierarchicalMenu',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n toggleShowMore = createToggleShowMore(renderOptions, this);\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state\n .removeHierarchicalFacet(hierarchicalFacetName)\n .setQueryParameter('maxValuesPerFacet', undefined);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n hierarchicalMenu: {\n ...renderState.hierarchicalMenu,\n [hierarchicalFacetName]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState({\n results,\n state,\n createURL,\n instantSearchInstance,\n helper,\n }) {\n let items: HierarchicalMenuRenderState['items'] = [];\n let canToggleShowMore = false;\n\n // Bind createURL to this specific attribute\n function _createURL(facetValue: string) {\n return createURL(\n state\n .resetPage()\n .toggleFacetRefinement(hierarchicalFacetName, facetValue)\n );\n }\n\n if (!sendEvent) {\n sendEvent = createSendEventForFacet({\n instantSearchInstance,\n helper,\n attribute: hierarchicalFacetName,\n widgetType: this.$$type,\n });\n }\n\n if (!_refine) {\n _refine = function (facetValue) {\n sendEvent('click', facetValue);\n helper\n .toggleFacetRefinement(hierarchicalFacetName, facetValue)\n .search();\n };\n }\n\n if (results) {\n const facetValues = results.getFacetValues(hierarchicalFacetName, {\n sortBy,\n facetOrdering: sortBy === DEFAULT_SORT,\n });\n const facetItems =\n facetValues && !Array.isArray(facetValues) && facetValues.data\n ? facetValues.data\n : [];\n\n // If the limit is the max number of facet retrieved it is impossible to know\n // if the facets are exhaustive. The only moment we are sure it is exhaustive\n // is when it is strictly under the number requested unless we know that another\n // widget has requested more values (maxValuesPerFacet > getLimit()).\n // Because this is used for making the search of facets unable or not, it is important\n // to be conservative here.\n const hasExhaustiveItems =\n (state.maxValuesPerFacet || 0) > getLimit()\n ? facetItems.length <= getLimit()\n : facetItems.length < getLimit();\n\n canToggleShowMore =\n showMore && (isShowingMore || !hasExhaustiveItems);\n\n items = transformItems(_prepareFacetValues(facetItems), {\n results,\n });\n }\n\n return {\n items,\n refine: _refine,\n canRefine: items.length > 0,\n createURL: _createURL,\n sendEvent,\n widgetParams,\n isShowingMore,\n toggleShowMore: cachedToggleShowMore,\n canToggleShowMore,\n };\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const path = searchParameters.getHierarchicalFacetBreadcrumb(\n hierarchicalFacetName\n );\n\n if (!path.length) {\n return uiState;\n }\n\n return {\n ...uiState,\n hierarchicalMenu: {\n ...uiState.hierarchicalMenu,\n [hierarchicalFacetName]: path,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n const values =\n uiState.hierarchicalMenu &&\n uiState.hierarchicalMenu[hierarchicalFacetName];\n\n if (searchParameters.isHierarchicalFacet(hierarchicalFacetName)) {\n const facet = searchParameters.getHierarchicalFacetByName(\n hierarchicalFacetName\n );\n\n warning(\n isEqual(facet.attributes, attributes) &&\n facet.separator === separator &&\n facet.rootPath === rootPath,\n 'Using Breadcrumb and HierarchicalMenu on the same facet with different options overrides the configuration of the HierarchicalMenu.'\n );\n }\n\n const withFacetConfiguration = searchParameters\n .removeHierarchicalFacet(hierarchicalFacetName)\n .addHierarchicalFacet({\n name: hierarchicalFacetName,\n attributes,\n separator,\n rootPath,\n showParentLevel,\n });\n\n const currentMaxValuesPerFacet =\n withFacetConfiguration.maxValuesPerFacet || 0;\n\n const nextMaxValuesPerFacet = Math.max(\n currentMaxValuesPerFacet,\n showMore ? showMoreLimit : limit\n );\n\n const withMaxValuesPerFacet =\n withFacetConfiguration.setQueryParameter(\n 'maxValuesPerFacet',\n nextMaxValuesPerFacet\n );\n\n if (!values) {\n return withMaxValuesPerFacet.setQueryParameters({\n hierarchicalFacetsRefinements: {\n ...withMaxValuesPerFacet.hierarchicalFacetsRefinements,\n [hierarchicalFacetName]: [],\n },\n });\n }\n\n return withMaxValuesPerFacet.addHierarchicalFacetRefinement(\n hierarchicalFacetName,\n values.join(separator)\n );\n },\n };\n };\n };\n\nexport default connectHierarchicalMenu;\n","import type { SendEventForHits, BindEventForHits } from '../../lib/utils';\nimport {\n escapeHits,\n TAG_PLACEHOLDER,\n checkRendering,\n createDocumentationMessageGenerator,\n addAbsolutePosition,\n addQueryID,\n createSendEventForHits,\n createBindEventForHits,\n noop,\n} from '../../lib/utils';\nimport type {\n TransformItems,\n Connector,\n Hit,\n WidgetRenderState,\n BaseHit,\n} from '../../types';\nimport type { SearchResults } from 'algoliasearch-helper';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'hits',\n connector: true,\n});\n\nexport type HitsRenderState = {\n /**\n * The matched hits from Algolia API.\n */\n hits: Array>;\n\n /**\n * The response from the Algolia API.\n */\n results?: SearchResults>;\n\n /**\n * Sends an event to the Insights middleware.\n */\n sendEvent: SendEventForHits;\n\n /**\n * Returns a string for the `data-insights-event` attribute for the Insights middleware\n */\n bindEvent: BindEventForHits;\n};\n\nexport type HitsConnectorParams = {\n /**\n * Whether to escape HTML tags from hits string values.\n *\n * @default true\n */\n escapeHTML?: boolean;\n\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems>;\n};\n\nexport type HitsWidgetDescription = {\n $$type: 'ais.hits';\n renderState: HitsRenderState;\n indexRenderState: {\n hits: WidgetRenderState, HitsConnectorParams>;\n };\n};\n\nexport type HitsConnector = Connector<\n HitsWidgetDescription,\n HitsConnectorParams\n>;\n\nconst connectHits: HitsConnector = function connectHits(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n escapeHTML = true,\n transformItems = ((items) => items) as NonNullable<\n HitsConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n let sendEvent: SendEventForHits;\n let bindEvent: BindEventForHits;\n\n return {\n $$type: 'ais.hits',\n\n init(initOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance: initOptions.instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const renderState = this.getWidgetRenderState(renderOptions);\n\n renderFn(\n {\n ...renderState,\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n\n renderState.sendEvent('view', renderState.hits);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n hits: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ results, helper, instantSearchInstance }) {\n if (!sendEvent) {\n sendEvent = createSendEventForHits({\n instantSearchInstance,\n index: helper.getIndex(),\n widgetType: this.$$type,\n });\n }\n\n if (!bindEvent) {\n bindEvent = createBindEventForHits({\n index: helper.getIndex(),\n widgetType: this.$$type,\n });\n }\n\n if (!results) {\n return {\n hits: [],\n results: undefined,\n sendEvent,\n bindEvent,\n widgetParams,\n };\n }\n\n if (escapeHTML && results.hits.length > 0) {\n results.hits = escapeHits(results.hits);\n }\n\n const hitsWithAbsolutePosition = addAbsolutePosition(\n results.hits,\n results.page,\n results.hitsPerPage\n );\n\n const hitsWithAbsolutePositionAndQueryID = addQueryID(\n hitsWithAbsolutePosition,\n results.queryID\n );\n\n const transformedHits = transformItems(\n hitsWithAbsolutePositionAndQueryID,\n { results }\n );\n\n return {\n hits: transformedHits,\n results,\n sendEvent,\n bindEvent,\n widgetParams,\n };\n },\n\n dispose({ state }) {\n unmountFn();\n\n if (!escapeHTML) {\n return state;\n }\n\n return state.setQueryParameters(\n Object.keys(TAG_PLACEHOLDER).reduce(\n (acc, key) => ({\n ...acc,\n [key]: undefined,\n }),\n {}\n )\n );\n },\n\n getWidgetSearchParameters(state) {\n if (!escapeHTML) {\n return state;\n }\n\n return state.setQueryParameters(TAG_PLACEHOLDER);\n },\n };\n };\n};\n\nexport default connectHits;\n","import type { SearchResults } from 'algoliasearch-helper';\nimport {\n uniq,\n find,\n createDocumentationMessageGenerator,\n warning,\n} from '../utils';\nimport type {\n Hits,\n InsightsClient,\n InsightsClientMethod,\n InsightsClientPayload,\n Connector,\n} from '../../types';\n\nconst getSelectedHits = (hits: Hits, selectedObjectIDs: string[]): Hits => {\n return selectedObjectIDs.map((objectID) => {\n const hit = find(hits, (h) => h.objectID === objectID);\n if (typeof hit === 'undefined') {\n throw new Error(\n `Could not find objectID \"${objectID}\" passed to \\`clickedObjectIDsAfterSearch\\` in the returned hits. This is necessary to infer the absolute position and the query ID.`\n );\n }\n return hit;\n });\n};\n\nconst getQueryID = (selectedHits: Hits): string => {\n const queryIDs = uniq(selectedHits.map((hit) => hit.__queryID));\n if (queryIDs.length > 1) {\n throw new Error(\n 'Insights currently allows a single `queryID`. The `objectIDs` provided map to multiple `queryID`s.'\n );\n }\n const queryID = queryIDs[0];\n if (typeof queryID !== 'string') {\n throw new Error(\n `Could not infer \\`queryID\\`. Ensure InstantSearch \\`clickAnalytics: true\\` was added with the Configure widget.\n\nSee: https://alg.li/lNiZZ7`\n );\n }\n return queryID;\n};\n\nconst getPositions = (selectedHits: Hits): number[] =>\n selectedHits.map((hit) => hit.__position);\n\nexport const inferPayload = ({\n method,\n results,\n hits,\n objectIDs,\n}: {\n method: InsightsClientMethod;\n results: SearchResults;\n hits: Hits;\n objectIDs: string[];\n}): Omit => {\n const { index } = results;\n const selectedHits = getSelectedHits(hits, objectIDs);\n const queryID = getQueryID(selectedHits);\n\n switch (method) {\n case 'clickedObjectIDsAfterSearch': {\n const positions = getPositions(selectedHits);\n return { index, queryID, objectIDs, positions };\n }\n\n case 'convertedObjectIDsAfterSearch':\n return { index, queryID, objectIDs };\n\n default:\n throw new Error(`Unsupported method passed to insights: \"${method}\".`);\n }\n};\n\nconst wrapInsightsClient =\n (\n aa: InsightsClient | null,\n results: SearchResults,\n hits: Hits\n ): InsightsClient =>\n (method, ...payloads) => {\n const [payload] = payloads;\n warning(\n false,\n `\\`insights\\` function has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the \\`insights\\` middleware.\n\nFor more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/`\n );\n if (!aa) {\n const withInstantSearchUsage = createDocumentationMessageGenerator({\n name: 'instantsearch',\n });\n throw new Error(\n withInstantSearchUsage(\n 'The `insightsClient` option has not been provided to `instantsearch`.'\n )\n );\n }\n if (!Array.isArray(payload.objectIDs)) {\n throw new TypeError('Expected `objectIDs` to be an array.');\n }\n const inferredPayload = inferPayload({\n method,\n results,\n hits,\n objectIDs: payload.objectIDs,\n });\n aa(method, { ...inferredPayload, ...payload });\n };\n\n/**\n * @deprecated This function will be still supported in 4.x releases, but not further. It is replaced by the `insights` middleware. For more information, visit https://www.algolia.com/doc/guides/getting-insights-and-analytics/search-analytics/click-through-and-conversions/how-to/send-click-and-conversion-events-with-instantsearch/js/\n * It passes `insights` to `HitsWithInsightsListener` and `InfiniteHitsWithInsightsListener`.\n */\nexport default function withInsights>(\n connector: TConnector\n): TConnector {\n return ((renderFn, unmountFn) =>\n connector((renderOptions, isFirstRender) => {\n const { results, hits, instantSearchInstance } = renderOptions;\n if (results && hits && instantSearchInstance) {\n const insights = wrapInsightsClient(\n instantSearchInstance.insightsClient,\n results,\n hits\n );\n return renderFn({ ...renderOptions, insights }, isFirstRender);\n }\n return renderFn(renderOptions, isFirstRender);\n }, unmountFn)) as TConnector;\n}\n","var n,l,u,i,t,r,o,f,e={},c=[],s=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i;function a(n,l){for(var u in l)n[u]=l[u];return n}function h(n){var l=n.parentNode;l&&l.removeChild(n)}function v(l,u,i){var t,r,o,f={};for(o in u)\"key\"==o?t=u[o]:\"ref\"==o?r=u[o]:f[o]=u[o];if(arguments.length>2&&(f.children=arguments.length>3?n.call(arguments,2):i),\"function\"==typeof l&&null!=l.defaultProps)for(o in l.defaultProps)void 0===f[o]&&(f[o]=l.defaultProps[o]);return y(l,f,t,r,null)}function y(n,i,t,r,o){var f={type:n,props:i,key:t,ref:r,__k:null,__:null,__b:0,__e:null,__d:void 0,__c:null,__h:null,constructor:void 0,__v:null==o?++u:o};return null==o&&null!=l.vnode&&l.vnode(f),f}function p(){return{current:null}}function d(n){return n.children}function _(n,l){this.props=n,this.context=l}function k(n,l){if(null==l)return n.__?k(n.__,n.__.__k.indexOf(n)+1):null;for(var u;l0?y(_.type,_.props,_.key,null,_.__v):_)){if(_.__=u,_.__b=u.__b+1,null===(p=w[h])||p&&_.key==p.key&&_.type===p.type)w[h]=void 0;else for(v=0;v2&&(f.children=arguments.length>3?n.call(arguments,2):i),y(l.type,f,t||l.key,r||l.ref,null)}function D(n,l){var u={__c:l=\"__cC\"+f++,__:n,Consumer:function(n,l){return n.children(l)},Provider:function(n){var u,i;return this.getChildContext||(u=[],(i={})[l]=this,this.getChildContext=function(){return i},this.shouldComponentUpdate=function(n){this.props.value!==n.value&&u.some(m)},this.sub=function(n){u.push(n);var l=n.componentWillUnmount;n.componentWillUnmount=function(){u.splice(u.indexOf(n),1),l&&l.call(n)}}),n.children}};return u.Provider.__=u.Consumer.contextType=u}n=c.slice,l={__e:function(n,l){for(var u,i,t;l=l.__;)if((u=l.__c)&&!u.__)try{if((i=u.constructor)&&null!=i.getDerivedStateFromError&&(u.setState(i.getDerivedStateFromError(n)),t=u.__d),null!=u.componentDidCatch&&(u.componentDidCatch(n),t=u.__d),t)return u.__E=u}catch(l){n=l}throw n}},u=0,i=function(n){return null!=n&&void 0===n.constructor},_.prototype.setState=function(n,l){var u;u=null!=this.__s&&this.__s!==this.state?this.__s:this.__s=a({},this.state),\"function\"==typeof n&&(n=n(a({},u),this.props)),n&&a(u,n),null!=n&&this.__v&&(l&&this.__h.push(l),m(this))},_.prototype.forceUpdate=function(n){this.__v&&(this.__e=!0,n&&this.__h.push(n),m(this))},_.prototype.render=d,t=[],r=\"function\"==typeof Promise?Promise.prototype.then.bind(Promise.resolve()):setTimeout,g.__r=0,f=0;export{S as render,q as hydrate,v as createElement,v as h,d as Fragment,p as createRef,i as isValidElement,_ as Component,B as cloneElement,D as createContext,A as toChildArray,l as options};\n//# sourceMappingURL=preact.module.js.map\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport { deserializePayload } from '../utils';\nimport { readDataAttributes, hasDataAttributes } from '../../helpers/insights';\nimport type { InsightsClient } from '../../types';\nimport type { InsightsEvent } from '../../middlewares/createInsightsMiddleware';\n\ntype WithInsightsListenerProps = {\n [key: string]: unknown;\n insights: InsightsClient;\n sendEvent?: (event: InsightsEvent) => void;\n};\n\nconst findInsightsTarget = (\n startElement: HTMLElement | null,\n endElement: HTMLElement | null,\n validator: (element: HTMLElement) => boolean\n): HTMLElement | null => {\n let element: HTMLElement | null = startElement;\n while (element && !validator(element)) {\n if (element === endElement) {\n return null;\n }\n element = element.parentElement;\n }\n return element;\n};\n\nconst parseInsightsEvent = (element: HTMLElement): InsightsEvent[] => {\n const serializedPayload = element.getAttribute('data-insights-event');\n\n if (typeof serializedPayload !== 'string') {\n throw new Error(\n 'The insights middleware expects `data-insights-event` to be a base64-encoded JSON string.'\n );\n }\n\n try {\n return deserializePayload(serializedPayload);\n } catch (error) {\n throw new Error(\n 'The insights middleware was unable to parse `data-insights-event`.'\n );\n }\n};\n\nconst insightsListener = (BaseComponent: any) => {\n function WithInsightsListener(props: WithInsightsListenerProps) {\n const handleClick = (event: MouseEvent): void => {\n if (props.sendEvent) {\n // new way with insights middleware\n const targetWithEvent = findInsightsTarget(\n event.target as HTMLElement | null,\n event.currentTarget as HTMLElement | null,\n (element) => element.hasAttribute('data-insights-event')\n );\n if (targetWithEvent) {\n const payload = parseInsightsEvent(targetWithEvent);\n\n payload.forEach((single) => props.sendEvent!(single));\n }\n }\n\n // old way, e.g. instantsearch.insights(\"clickedObjectIDsAfterSearch\", { .. })\n const insightsTarget = findInsightsTarget(\n event.target as HTMLElement | null,\n event.currentTarget as HTMLElement | null,\n (element) => hasDataAttributes(element)\n );\n if (insightsTarget) {\n const { method, payload } = readDataAttributes(insightsTarget);\n props.insights(method, payload);\n }\n };\n\n return (\n
\n \n
\n );\n }\n\n return WithInsightsListener;\n};\n\nexport default insightsListener;\n","import { withInsights } from '../../lib/insights';\nimport type { HitsConnectorParams, HitsWidgetDescription } from './connectHits';\nimport connectHits from './connectHits';\nimport type { Connector } from '../../types';\n\n/**\n * Due to https://github.com/microsoft/web-build-tools/issues/1050, we need\n * Connector<...> imported in this file, even though it is only used implicitly.\n * This _uses_ Connector<...> so it is not accidentally removed by someone.\n */\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\ndeclare type ImportWorkaround = Connector<\n HitsWidgetDescription,\n HitsConnectorParams\n>;\n\nconst connectHitsWithInsights = withInsights(connectHits);\n\nexport default connectHitsWithInsights;\n","import {\n checkRendering,\n warning,\n createDocumentationMessageGenerator,\n noop,\n} from '../../lib/utils';\n\nimport type {\n AlgoliaSearchHelper,\n SearchParameters,\n} from 'algoliasearch-helper';\nimport type {\n Connector,\n TransformItems,\n CreateURL,\n InitOptions,\n RenderOptions,\n WidgetRenderState,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'hits-per-page',\n connector: true,\n});\n\nexport type HitsPerPageRenderStateItem = {\n /**\n * Label to display in the option.\n */\n label: string;\n\n /**\n * Number of hits to display per page.\n */\n value: number;\n\n /**\n * Indicates if it's the current refined value.\n */\n isRefined: boolean;\n};\n\nexport type HitsPerPageConnectorParamsItem = {\n /**\n * Label to display in the option.\n */\n label: string;\n\n /**\n * Number of hits to display per page.\n */\n value: number;\n\n /**\n * The default hits per page on first search.\n *\n * @default false\n */\n default?: boolean;\n};\n\nexport type HitsPerPageConnectorParams = {\n /**\n * Array of objects defining the different values and labels.\n */\n items: HitsPerPageConnectorParamsItem[];\n\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n};\n\nexport type HitsPerPageRenderState = {\n /**\n * Array of objects defining the different values and labels.\n */\n items: HitsPerPageRenderStateItem[];\n\n /**\n * Creates the URL for a single item name in the list.\n *\n * @internal\n */\n createURL: CreateURL;\n\n /**\n * Sets the number of hits per page and triggers a search.\n */\n refine: (value: number) => void;\n\n /**\n * Indicates whether or not the search has results.\n */\n hasNoResults: boolean;\n};\n\nexport type HitsPerPageWidgetDescription = {\n $$type: 'ais.hitsPerPage';\n renderState: HitsPerPageRenderState;\n indexRenderState: {\n hitsPerPage: WidgetRenderState<\n HitsPerPageRenderState,\n HitsPerPageConnectorParams\n >;\n };\n indexUiState: {\n hitsPerPage: number;\n };\n};\n\nexport type HitsPerPageConnector = Connector<\n HitsPerPageWidgetDescription,\n HitsPerPageConnectorParams\n>;\n\nconst connectHitsPerPage: HitsPerPageConnector = function connectHitsPerPage(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n items: userItems,\n transformItems = ((items) => items) as NonNullable<\n HitsPerPageConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n if (!Array.isArray(userItems)) {\n throw new Error(\n withUsage('The `items` option expects an array of objects.')\n );\n }\n\n let items = userItems;\n\n const defaultItems = items.filter((item) => item.default === true);\n\n if (defaultItems.length === 0) {\n throw new Error(\n withUsage(`A default value must be specified in \\`items\\`.`)\n );\n }\n\n if (defaultItems.length > 1) {\n throw new Error(\n withUsage('More than one default value is specified in `items`.')\n );\n }\n\n const defaultItem = defaultItems[0];\n\n const normalizeItems = ({ hitsPerPage }: SearchParameters) => {\n return items.map((item) => ({\n ...item,\n isRefined: Number(item.value) === Number(hitsPerPage),\n }));\n };\n\n type ConnectorState = {\n getRefine: (\n helper: AlgoliaSearchHelper\n ) => (value: HitsPerPageConnectorParamsItem['value']) => any;\n createURLFactory: (props: {\n state: SearchParameters;\n createURL: (InitOptions | RenderOptions)['createURL'];\n }) => HitsPerPageRenderState['createURL'];\n };\n\n const connectorState: ConnectorState = {\n getRefine: (helper) => (value) => {\n return !value && value !== 0\n ? helper.setQueryParameter('hitsPerPage', undefined).search()\n : helper.setQueryParameter('hitsPerPage', value).search();\n },\n createURLFactory:\n ({ state, createURL }) =>\n (value) =>\n createURL(\n state\n .resetPage()\n .setQueryParameter(\n 'hitsPerPage',\n !value && value !== 0 ? undefined : value\n )\n ),\n };\n\n return {\n $$type: 'ais.hitsPerPage',\n\n init(initOptions) {\n const { state, instantSearchInstance } = initOptions;\n\n const isCurrentInOptions = items.some(\n (item) => Number(state.hitsPerPage) === Number(item.value)\n );\n\n if (!isCurrentInOptions) {\n warning(\n state.hitsPerPage !== undefined,\n `\n\\`hitsPerPage\\` is not defined.\nThe option \\`hitsPerPage\\` needs to be set using the \\`configure\\` widget.\n\nLearn more: https://www.algolia.com/doc/api-reference/widgets/hits-per-page/js/\n `\n );\n\n warning(\n false,\n `\nThe \\`items\\` option of \\`hitsPerPage\\` does not contain the \"hits per page\" value coming from the state: ${state.hitsPerPage}.\n\nYou may want to add another entry to the \\`items\\` option with this value.`\n );\n\n items = [\n // The helper will convert the empty string to `undefined`.\n { value: '' as unknown as number, label: '' },\n ...items,\n ];\n }\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state.setQueryParameter('hitsPerPage', undefined);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n hitsPerPage: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ state, results, createURL, helper }) {\n return {\n items: transformItems(normalizeItems(state), { results }),\n refine: connectorState.getRefine(helper),\n createURL: connectorState.createURLFactory({ state, createURL }),\n hasNoResults: results ? results.nbHits === 0 : true,\n widgetParams,\n };\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const hitsPerPage = searchParameters.hitsPerPage;\n\n if (hitsPerPage === undefined || hitsPerPage === defaultItem.value) {\n return uiState;\n }\n\n return {\n ...uiState,\n hitsPerPage,\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n return searchParameters.setQueryParameters({\n hitsPerPage: uiState.hitsPerPage || defaultItem.value,\n });\n },\n };\n };\n};\n\nexport default connectHitsPerPage;\n","import type {\n AlgoliaSearchHelper as Helper,\n PlainSearchParameters,\n SearchParameters,\n SearchResults,\n} from 'algoliasearch-helper';\nimport type {\n Connector,\n TransformItems,\n Hit,\n WidgetRenderState,\n BaseHit,\n} from '../../types';\nimport type { SendEventForHits, BindEventForHits } from '../../lib/utils';\nimport {\n escapeHits,\n TAG_PLACEHOLDER,\n checkRendering,\n createDocumentationMessageGenerator,\n isEqual,\n addAbsolutePosition,\n addQueryID,\n noop,\n createSendEventForHits,\n createBindEventForHits,\n} from '../../lib/utils';\n\nexport type InfiniteHitsCachedHits = {\n [page: number]: Array>;\n};\n\ntype Read = ({\n state,\n}: {\n state: PlainSearchParameters;\n}) => InfiniteHitsCachedHits | null;\n\ntype Write = ({\n state,\n hits,\n}: {\n state: PlainSearchParameters;\n hits: InfiniteHitsCachedHits;\n}) => void;\n\nexport type InfiniteHitsCache = {\n read: Read;\n write: Write;\n};\n\nexport type InfiniteHitsConnectorParams = {\n /**\n * Escapes HTML entities from hits string values.\n *\n * @default `true`\n */\n escapeHTML?: boolean;\n\n /**\n * Enable the button to load previous results.\n *\n * @default `false`\n */\n showPrevious?: boolean;\n\n /**\n * Receives the items, and is called before displaying them.\n * Useful for mapping over the items to transform, and remove or reorder them.\n */\n transformItems?: TransformItems>;\n\n /**\n * Reads and writes hits from/to cache.\n * When user comes back to the search page after leaving for product page,\n * this helps restore InfiniteHits and its scroll position.\n */\n cache?: InfiniteHitsCache;\n};\n\nexport type InfiniteHitsRenderState = {\n /**\n * Loads the previous results.\n */\n showPrevious: () => void;\n\n /**\n * Loads the next page of hits.\n */\n showMore: () => void;\n\n /**\n * Indicates whether the first page of hits has been reached.\n */\n isFirstPage: boolean;\n\n /**\n * Indicates whether the last page of hits has been reached.\n */\n isLastPage: boolean;\n\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEventForHits;\n\n /**\n * Returns a string of data-insights-event attribute for insights middleware\n */\n bindEvent: BindEventForHits;\n\n /**\n * Hits for the current page\n */\n currentPageHits: Array>;\n\n /**\n * Hits for current and cached pages\n */\n hits: Array>;\n\n /**\n * The response from the Algolia API.\n */\n results?: SearchResults>;\n};\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'infinite-hits',\n connector: true,\n});\n\nexport type InfiniteHitsWidgetDescription = {\n $$type: 'ais.infiniteHits';\n renderState: InfiniteHitsRenderState;\n indexRenderState: {\n infiniteHits: WidgetRenderState<\n InfiniteHitsRenderState,\n InfiniteHitsConnectorParams\n >;\n };\n indexUiState: {\n page: number;\n };\n};\n\nexport type InfiniteHitsConnector = Connector<\n InfiniteHitsWidgetDescription,\n InfiniteHitsConnectorParams\n>;\n\nfunction getStateWithoutPage(state: PlainSearchParameters) {\n const { page, ...rest } = state || {};\n return rest;\n}\n\nfunction getInMemoryCache(): InfiniteHitsCache {\n let cachedHits: InfiniteHitsCachedHits | null = null;\n let cachedState: PlainSearchParameters | null = null;\n return {\n read({ state }) {\n return isEqual(cachedState, getStateWithoutPage(state))\n ? cachedHits\n : null;\n },\n write({ state, hits }) {\n cachedState = getStateWithoutPage(state);\n cachedHits = hits;\n },\n };\n}\n\nfunction extractHitsFromCachedHits(\n cachedHits: InfiniteHitsCachedHits\n) {\n return Object.keys(cachedHits)\n .map(Number)\n .sort((a, b) => a - b)\n .reduce((acc: Array>, page) => {\n return acc.concat(cachedHits[page]);\n }, []);\n}\n\nconst connectInfiniteHits: InfiniteHitsConnector = function connectInfiniteHits(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n // @TODO: this should be a generic, but a Connector can not yet be generic itself\n type THit = BaseHit;\n\n return (widgetParams) => {\n const {\n escapeHTML = true,\n transformItems = ((items) => items) as NonNullable<\n InfiniteHitsConnectorParams['transformItems']\n >,\n cache = getInMemoryCache(),\n } = widgetParams || {};\n let showPrevious: () => void;\n let showMore: () => void;\n let sendEvent: SendEventForHits;\n let bindEvent: BindEventForHits;\n const getFirstReceivedPage = (\n state: SearchParameters,\n cachedHits: InfiniteHitsCachedHits\n ) => {\n const { page = 0 } = state;\n const pages = Object.keys(cachedHits).map(Number);\n if (pages.length === 0) {\n return page;\n } else {\n return Math.min(page, ...pages);\n }\n };\n const getLastReceivedPage = (\n state: SearchParameters,\n cachedHits: InfiniteHitsCachedHits\n ) => {\n const { page = 0 } = state;\n const pages = Object.keys(cachedHits).map(Number);\n if (pages.length === 0) {\n return page;\n } else {\n return Math.max(page, ...pages);\n }\n };\n\n const getShowPrevious =\n (helper: Helper): (() => void) =>\n () => {\n // Using the helper's `overrideStateWithoutTriggeringChangeEvent` method\n // avoid updating the browser URL when the user displays the previous page.\n helper\n .overrideStateWithoutTriggeringChangeEvent({\n ...helper.state,\n page:\n getFirstReceivedPage(\n helper.state,\n cache.read({ state: helper.state }) || {}\n ) - 1,\n })\n .searchWithoutTriggeringOnStateChange();\n };\n\n const getShowMore =\n (helper: Helper): (() => void) =>\n () => {\n helper\n .setPage(\n getLastReceivedPage(\n helper.state,\n cache.read({ state: helper.state }) || {}\n ) + 1\n )\n .search();\n };\n\n return {\n $$type: 'ais.infiniteHits',\n\n init(initOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance: initOptions.instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n const widgetRenderState = this.getWidgetRenderState(renderOptions);\n\n renderFn(\n {\n ...widgetRenderState,\n instantSearchInstance,\n },\n false\n );\n sendEvent('view', widgetRenderState.currentPageHits);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n infiniteHits: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ results, helper, state, instantSearchInstance }) {\n let isFirstPage: boolean;\n let currentPageHits: Array> = [];\n const cachedHits = cache.read({ state }) || {};\n\n if (!results) {\n showPrevious = getShowPrevious(helper);\n showMore = getShowMore(helper);\n sendEvent = createSendEventForHits({\n instantSearchInstance,\n index: helper.getIndex(),\n widgetType: this.$$type,\n });\n bindEvent = createBindEventForHits({\n index: helper.getIndex(),\n widgetType: this.$$type,\n });\n isFirstPage =\n state.page === undefined ||\n getFirstReceivedPage(state, cachedHits) === 0;\n } else {\n const { page = 0 } = state;\n\n if (escapeHTML && results.hits.length > 0) {\n results.hits = escapeHits(results.hits);\n }\n\n const hitsWithAbsolutePosition = addAbsolutePosition(\n results.hits,\n results.page,\n results.hitsPerPage\n );\n\n const hitsWithAbsolutePositionAndQueryID = addQueryID(\n hitsWithAbsolutePosition,\n results.queryID\n );\n\n const transformedHits = transformItems(\n hitsWithAbsolutePositionAndQueryID,\n { results }\n );\n\n if (cachedHits[page] === undefined && !results.__isArtificial) {\n cachedHits[page] = transformedHits;\n cache.write({ state, hits: cachedHits });\n }\n currentPageHits = transformedHits;\n\n isFirstPage = getFirstReceivedPage(state, cachedHits) === 0;\n }\n\n const hits = extractHitsFromCachedHits(cachedHits);\n const isLastPage = results\n ? results.nbPages <= getLastReceivedPage(state, cachedHits) + 1\n : true;\n\n return {\n hits,\n currentPageHits,\n sendEvent,\n bindEvent,\n results,\n showPrevious,\n showMore,\n isFirstPage,\n isLastPage,\n widgetParams,\n };\n },\n\n dispose({ state }) {\n unmountFn();\n\n const stateWithoutPage = state.setQueryParameter('page', undefined);\n\n if (!escapeHTML) {\n return stateWithoutPage;\n }\n\n return stateWithoutPage.setQueryParameters(\n Object.keys(TAG_PLACEHOLDER).reduce(\n (acc, key) => ({\n ...acc,\n [key]: undefined,\n }),\n {}\n )\n );\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const page = searchParameters.page || 0;\n\n if (!page) {\n // return without adding `page` to uiState\n // because we don't want `page=1` in the URL\n return uiState;\n }\n\n return {\n ...uiState,\n // The page in the UI state is incremented by one\n // to expose the user value (not `0`).\n page: page + 1,\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n let widgetSearchParameters = searchParameters;\n\n if (escapeHTML) {\n widgetSearchParameters =\n searchParameters.setQueryParameters(TAG_PLACEHOLDER);\n }\n\n // The page in the search parameters is decremented by one\n // to get to the actual parameter value from the UI state.\n const page = uiState.page ? uiState.page - 1 : 0;\n\n return widgetSearchParameters.setQueryParameter('page', page);\n },\n };\n };\n};\n\nexport default connectInfiniteHits;\n","import { withInsights } from '../../lib/insights';\nimport type {\n InfiniteHitsWidgetDescription,\n InfiniteHitsConnectorParams,\n} from './connectInfiniteHits';\nimport connectInfiniteHits from './connectInfiniteHits';\nimport type { Connector } from '../../types';\n\n/**\n * Due to https://github.com/microsoft/web-build-tools/issues/1050, we need\n * Connector<...> imported in this file, even though it is only used implicitly.\n * This _uses_ Connector<...> so it is not accidentally removed by someone.\n */\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\ndeclare type ImportWorkaround = Connector<\n InfiniteHitsWidgetDescription,\n InfiniteHitsConnectorParams\n>;\n\nconst connectInfiniteHitsWithInsights = withInsights(connectInfiniteHits);\n\nexport default connectInfiniteHitsWithInsights;\n","import type { SearchResults } from 'algoliasearch-helper';\nimport type { SendEventForFacet } from '../../lib/utils';\nimport {\n checkRendering,\n createDocumentationMessageGenerator,\n createSendEventForFacet,\n noop,\n} from '../../lib/utils';\nimport type {\n Connector,\n CreateURL,\n RenderOptions,\n SortBy,\n TransformItems,\n Widget,\n WidgetRenderState,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'menu',\n connector: true,\n});\n\nconst DEFAULT_SORT = ['isRefined', 'name:asc'];\n\nexport type MenuItem = {\n /**\n * The value of the menu item.\n */\n value: string;\n /**\n * Human-readable value of the menu item.\n */\n label: string;\n /**\n * Number of results matched after refinement is applied.\n */\n count: number;\n /**\n * Indicates if the refinement is applied.\n */\n isRefined: boolean;\n};\n\nexport type MenuConnectorParams = {\n /**\n * Name of the attribute for faceting (eg. \"free_shipping\").\n */\n attribute: string;\n /**\n * How many facets values to retrieve.\n */\n limit?: number;\n /**\n * Whether to display a button that expands the number of items.\n */\n showMore?: boolean;\n /**\n * How many facets values to retrieve when `toggleShowMore` is called, this value is meant to be greater than `limit` option.\n */\n showMoreLimit?: number;\n /**\n * How to sort refinements. Possible values: `count|isRefined|name:asc|name:desc`.\n *\n * You can also use a sort function that behaves like the standard Javascript [compareFunction](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#Syntax).\n *\n * If a facetOrdering is set in the index settings, it is used when sortBy isn't passed\n */\n sortBy?: SortBy;\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n};\n\nexport type MenuRenderState = {\n /**\n * The elements that can be refined for the current search results.\n */\n items: MenuItem[];\n /**\n * Creates the URL for a single item name in the list.\n */\n createURL: CreateURL;\n /**\n * Filter the search to item value.\n */\n refine(value: string): void;\n /**\n * True if refinement can be applied.\n */\n canRefine: boolean;\n /**\n * True if the menu is displaying all the menu items.\n */\n isShowingMore: boolean;\n /**\n * Toggles the number of values displayed between `limit` and `showMore.limit`.\n */\n toggleShowMore(): void;\n /**\n * `true` if the toggleShowMore button can be activated (enough items to display more or\n * already displaying more than `limit` items)\n */\n canToggleShowMore: boolean;\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEventForFacet;\n};\n\nexport type MenuWidgetDescription = {\n $$type: 'ais.menu';\n renderState: MenuRenderState;\n indexRenderState: {\n menu: {\n [attribute: string]: WidgetRenderState<\n MenuRenderState,\n MenuConnectorParams\n >;\n };\n };\n indexUiState: {\n menu: {\n [attribute: string]: string;\n };\n };\n};\n\nexport type MenuConnector = Connector<\n MenuWidgetDescription,\n MenuConnectorParams\n>;\n\n/**\n * **Menu** connector provides the logic to build a widget that will give the user the ability to choose a single value for a specific facet. The typical usage of menu is for navigation in categories.\n *\n * This connector provides a `toggleShowMore()` function to display more or less items and a `refine()`\n * function to select an item. While selecting a new element, the `refine` will also unselect the\n * one that is currently selected.\n *\n * **Requirement:** the attribute passed as `attribute` must be present in \"attributes for faceting\" on the Algolia dashboard or configured as attributesForFaceting via a set settings call to the Algolia API.\n */\nconst connectMenu: MenuConnector = function connectMenu(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n attribute,\n limit = 10,\n showMore = false,\n showMoreLimit = 20,\n sortBy = DEFAULT_SORT,\n transformItems = ((items) => items) as NonNullable<\n MenuConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n if (!attribute) {\n throw new Error(withUsage('The `attribute` option is required.'));\n }\n\n if (showMore === true && showMoreLimit <= limit) {\n throw new Error(\n withUsage('The `showMoreLimit` option must be greater than `limit`.')\n );\n }\n\n type ThisWidget = Widget<\n MenuWidgetDescription & { widgetParams: typeof widgetParams }\n >;\n\n let sendEvent: MenuRenderState['sendEvent'] | undefined;\n let _createURL: MenuRenderState['createURL'] | undefined;\n let _refine: MenuRenderState['refine'] | undefined;\n\n // Provide the same function to the `renderFn` so that way the user\n // has to only bind it once when `isFirstRendering` for instance\n let isShowingMore = false;\n let toggleShowMore = () => {};\n function createToggleShowMore(\n renderOptions: RenderOptions,\n widget: ThisWidget\n ) {\n return () => {\n isShowingMore = !isShowingMore;\n widget.render!(renderOptions);\n };\n }\n function cachedToggleShowMore() {\n toggleShowMore();\n }\n\n function getLimit() {\n return isShowingMore ? showMoreLimit : limit;\n }\n\n return {\n $$type: 'ais.menu' as const,\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state\n .removeHierarchicalFacet(attribute)\n .setQueryParameter('maxValuesPerFacet', undefined);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n menu: {\n ...renderState.menu,\n [attribute]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState(renderOptions) {\n const { results, createURL, instantSearchInstance, helper } =\n renderOptions;\n\n let items: MenuRenderState['items'] = [];\n let canToggleShowMore = false;\n\n if (!sendEvent) {\n sendEvent = createSendEventForFacet({\n instantSearchInstance,\n helper,\n attribute,\n widgetType: this.$$type,\n });\n }\n\n if (!_createURL) {\n _createURL = (facetValue: string) =>\n createURL(\n helper.state\n .resetPage()\n .toggleFacetRefinement(attribute, facetValue)\n );\n }\n\n if (!_refine) {\n _refine = function (facetValue: string) {\n const [refinedItem] =\n helper.getHierarchicalFacetBreadcrumb(attribute);\n sendEvent!('click', facetValue ? facetValue : refinedItem);\n helper\n .toggleFacetRefinement(\n attribute,\n facetValue ? facetValue : refinedItem\n )\n .search();\n };\n }\n\n if (renderOptions.results) {\n toggleShowMore = createToggleShowMore(renderOptions, this);\n }\n\n if (results) {\n const facetValues = results.getFacetValues(attribute, {\n sortBy,\n facetOrdering: sortBy === DEFAULT_SORT,\n });\n const facetItems =\n facetValues && !Array.isArray(facetValues) && facetValues.data\n ? facetValues.data\n : [];\n\n canToggleShowMore =\n showMore && (isShowingMore || facetItems.length > getLimit());\n\n items = transformItems(\n facetItems\n .slice(0, getLimit())\n .map(({ name: label, escapedValue: value, path, ...item }) => ({\n ...item,\n label,\n value,\n })),\n { results }\n );\n }\n\n return {\n items,\n createURL: _createURL,\n refine: _refine,\n sendEvent,\n canRefine: items.length > 0,\n widgetParams,\n isShowingMore,\n toggleShowMore: cachedToggleShowMore,\n canToggleShowMore,\n };\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const [value] =\n searchParameters.getHierarchicalFacetBreadcrumb(attribute);\n\n if (!value) {\n return uiState;\n }\n\n return {\n ...uiState,\n menu: {\n ...uiState.menu,\n [attribute]: value,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n const value = uiState.menu && uiState.menu[attribute];\n\n const withFacetConfiguration = searchParameters\n .removeHierarchicalFacet(attribute)\n .addHierarchicalFacet({\n name: attribute,\n attributes: [attribute],\n });\n\n const currentMaxValuesPerFacet =\n withFacetConfiguration.maxValuesPerFacet || 0;\n\n const nextMaxValuesPerFacet = Math.max(\n currentMaxValuesPerFacet,\n showMore ? showMoreLimit : limit\n );\n\n const withMaxValuesPerFacet = withFacetConfiguration.setQueryParameter(\n 'maxValuesPerFacet',\n nextMaxValuesPerFacet\n );\n\n if (!value) {\n return withMaxValuesPerFacet.setQueryParameters({\n hierarchicalFacetsRefinements: {\n ...withMaxValuesPerFacet.hierarchicalFacetsRefinements,\n [attribute]: [],\n },\n });\n }\n\n return withMaxValuesPerFacet.addHierarchicalFacetRefinement(\n attribute,\n value\n );\n },\n };\n };\n};\n\nexport default connectMenu;\n","import type { SendEventForFacet } from '../../lib/utils';\nimport {\n checkRendering,\n createDocumentationMessageGenerator,\n isFiniteNumber,\n convertNumericRefinementsToFilters,\n noop,\n} from '../../lib/utils';\nimport type {\n Connector,\n CreateURL,\n InstantSearch,\n TransformItems,\n WidgetRenderState,\n} from '../../types';\nimport type {\n AlgoliaSearchHelper,\n SearchParameters,\n} from 'algoliasearch-helper';\nimport type { InsightsEvent } from '../../middlewares';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'numeric-menu',\n connector: true,\n});\n\nexport type NumericMenuConnectorParamsItem = {\n /**\n * Name of the option\n */\n label: string;\n\n /**\n * Higher bound of the option (<=)\n */\n start?: number;\n\n /**\n * Lower bound of the option (>=)\n */\n end?: number;\n};\n\nexport type NumericMenuRenderStateItem = {\n /**\n * Name of the option.\n */\n label: string;\n\n /**\n * URL encoded of the bounds object with the form `{start, end}`.\n * This value can be used verbatim in the webpage and can be read by `refine`\n * directly. If you want to inspect the value, you can do:\n * `JSON.parse(decodeURI(value))` to get the object.\n */\n value: string;\n\n /**\n * True if the value is selected\n */\n isRefined: boolean;\n};\n\nexport type NumericMenuConnectorParams = {\n /**\n * Name of the attribute for filtering\n */\n attribute: string;\n\n /**\n * List of all the items\n */\n items: NumericMenuConnectorParamsItem[];\n\n /**\n * Function to transform the items passed to the templates\n */\n transformItems?: TransformItems;\n};\n\nexport type NumericMenuRenderState = {\n /**\n * The list of available choices\n */\n items: NumericMenuRenderStateItem[];\n\n /**\n * Creates URLs for the next state, the string is the name of the selected option\n */\n createURL: CreateURL;\n\n /**\n * `true` if the last search contains no result\n */\n hasNoResults: boolean;\n\n /**\n * Sets the selected value and trigger a new search\n */\n refine: (facetValue: string) => void;\n\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEventForFacet;\n};\n\nexport type NumericMenuWidgetDescription = {\n $$type: 'ais.numericMenu';\n renderState: NumericMenuRenderState;\n indexRenderState: {\n numericMenu: {\n [attribute: string]: WidgetRenderState<\n NumericMenuRenderState,\n NumericMenuConnectorParams\n >;\n };\n };\n indexUiState: {\n numericMenu: {\n // @TODO: this could possibly become `${number}:${number}` later\n [attribute: string]: string;\n };\n };\n};\n\nexport type NumericMenuConnector = Connector<\n NumericMenuWidgetDescription,\n NumericMenuConnectorParams\n>;\n\nconst $$type = 'ais.numericMenu';\n\nconst createSendEvent =\n ({\n instantSearchInstance,\n helper,\n attribute,\n }: {\n instantSearchInstance: InstantSearch;\n helper: AlgoliaSearchHelper;\n attribute: string;\n }) =>\n (...args: [InsightsEvent] | [string, string, string?]) => {\n if (args.length === 1) {\n instantSearchInstance.sendEventToInsights(args[0]);\n return;\n }\n\n const [eventType, facetValue, eventName = 'Filter Applied'] = args;\n if (eventType !== 'click') {\n return;\n }\n // facetValue === \"%7B%22start%22:5,%22end%22:10%7D\"\n const filters = convertNumericRefinementsToFilters(\n getRefinedState(helper.state, attribute, facetValue),\n attribute\n );\n if (filters && filters.length > 0) {\n /*\n filters === [\"price<=10\", \"price>=5\"]\n */\n instantSearchInstance.sendEventToInsights({\n insightsMethod: 'clickedFilters',\n widgetType: $$type,\n eventType,\n payload: {\n eventName,\n index: helper.getIndex(),\n filters,\n },\n attribute,\n });\n }\n };\n\nconst connectNumericMenu: NumericMenuConnector = function connectNumericMenu(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n attribute = '',\n items = [],\n transformItems = ((item) => item) as NonNullable<\n NumericMenuConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n if (attribute === '') {\n throw new Error(withUsage('The `attribute` option is required.'));\n }\n\n if (!items || items.length === 0) {\n throw new Error(\n withUsage('The `items` option expects an array of objects.')\n );\n }\n\n type ConnectorState = {\n refine?(facetValue: string): void;\n createURL?(state: SearchParameters): (facetValue: string) => string;\n sendEvent?: SendEventForFacet;\n };\n\n const prepareItems = (state: SearchParameters) =>\n items.map(({ start, end, label }) => ({\n label,\n value: encodeURI(JSON.stringify({ start, end })),\n isRefined: isRefined(state, attribute, { start, end, label }),\n }));\n\n const connectorState: ConnectorState = {};\n\n return {\n $$type,\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n return state.clearRefinements(attribute);\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const values = searchParameters.getNumericRefinements(attribute);\n\n const equal = values['='] && values['='][0];\n\n if (equal || equal === 0) {\n return {\n ...uiState,\n numericMenu: {\n ...uiState.numericMenu,\n [attribute]: `${values['=']}`,\n },\n };\n }\n\n const min = (values['>='] && values['>='][0]) || '';\n const max = (values['<='] && values['<='][0]) || '';\n\n if (min === '' && max === '') {\n return uiState;\n }\n\n return {\n ...uiState,\n numericMenu: {\n ...uiState.numericMenu,\n [attribute]: `${min}:${max}`,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n const value = uiState.numericMenu && uiState.numericMenu[attribute];\n\n const withoutRefinements = searchParameters.clearRefinements(attribute);\n\n if (!value) {\n return withoutRefinements.setQueryParameters({\n numericRefinements: {\n ...withoutRefinements.numericRefinements,\n [attribute]: {},\n },\n });\n }\n\n const isExact = value.indexOf(':') === -1;\n\n if (isExact) {\n return withoutRefinements.addNumericRefinement(\n attribute,\n '=',\n Number(value)\n );\n }\n\n const [min, max] = value.split(':').map(parseFloat);\n\n const withMinRefinement = isFiniteNumber(min)\n ? withoutRefinements.addNumericRefinement(attribute, '>=', min)\n : withoutRefinements;\n\n const withMaxRefinement = isFiniteNumber(max)\n ? withMinRefinement.addNumericRefinement(attribute, '<=', max)\n : withMinRefinement;\n\n return withMaxRefinement;\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n numericMenu: {\n ...renderState.numericMenu,\n [attribute]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState({\n results,\n state,\n instantSearchInstance,\n helper,\n createURL,\n }) {\n if (!connectorState.refine) {\n connectorState.refine = (facetValue) => {\n const refinedState = getRefinedState(\n helper.state,\n attribute,\n facetValue\n );\n connectorState.sendEvent!('click', facetValue);\n helper.setState(refinedState).search();\n };\n }\n\n if (!connectorState.createURL) {\n connectorState.createURL = (newState) => (facetValue) =>\n createURL(getRefinedState(newState, attribute, facetValue));\n }\n\n if (!connectorState.sendEvent) {\n connectorState.sendEvent = createSendEvent({\n instantSearchInstance,\n helper,\n attribute,\n });\n }\n\n return {\n createURL: connectorState.createURL(state),\n items: transformItems(prepareItems(state), { results }),\n hasNoResults: results ? results.nbHits === 0 : true,\n refine: connectorState.refine,\n sendEvent: connectorState.sendEvent,\n widgetParams,\n };\n },\n };\n };\n};\n\nfunction isRefined(\n state: SearchParameters,\n attribute: string,\n option: NumericMenuConnectorParamsItem\n) {\n // @TODO: same as another spot, why is this mixing arrays & elements?\n const currentRefinements = state.getNumericRefinements(attribute);\n\n if (option.start !== undefined && option.end !== undefined) {\n if (option.start === option.end) {\n return hasNumericRefinement(currentRefinements, '=', option.start);\n } else {\n return (\n hasNumericRefinement(currentRefinements, '>=', option.start) &&\n hasNumericRefinement(currentRefinements, '<=', option.end)\n );\n }\n }\n\n if (option.start !== undefined) {\n return hasNumericRefinement(currentRefinements, '>=', option.start);\n }\n\n if (option.end !== undefined) {\n return hasNumericRefinement(currentRefinements, '<=', option.end);\n }\n\n if (option.start === undefined && option.end === undefined) {\n return (\n Object.keys(currentRefinements) as SearchParameters.Operator[]\n ).every((operator) => (currentRefinements[operator] || []).length === 0);\n }\n\n return false;\n}\n\nfunction getRefinedState(\n state: SearchParameters,\n attribute: string,\n facetValue: string\n) {\n let resolvedState = state;\n\n const refinedOption = JSON.parse(decodeURI(facetValue));\n\n // @TODO: why is array / element mixed here & hasRefinements; seems wrong?\n const currentRefinements = resolvedState.getNumericRefinements(attribute);\n\n if (refinedOption.start === undefined && refinedOption.end === undefined) {\n return resolvedState.removeNumericRefinement(attribute);\n }\n\n if (!isRefined(resolvedState, attribute, refinedOption)) {\n resolvedState = resolvedState.removeNumericRefinement(attribute);\n }\n\n if (refinedOption.start !== undefined && refinedOption.end !== undefined) {\n if (refinedOption.start > refinedOption.end) {\n throw new Error('option.start should be > to option.end');\n }\n\n if (refinedOption.start === refinedOption.end) {\n if (hasNumericRefinement(currentRefinements, '=', refinedOption.start)) {\n resolvedState = resolvedState.removeNumericRefinement(\n attribute,\n '=',\n refinedOption.start\n );\n } else {\n resolvedState = resolvedState.addNumericRefinement(\n attribute,\n '=',\n refinedOption.start\n );\n }\n return resolvedState;\n }\n }\n\n if (refinedOption.start !== undefined) {\n if (hasNumericRefinement(currentRefinements, '>=', refinedOption.start)) {\n resolvedState = resolvedState.removeNumericRefinement(\n attribute,\n '>=',\n refinedOption.start\n );\n }\n resolvedState = resolvedState.addNumericRefinement(\n attribute,\n '>=',\n refinedOption.start\n );\n }\n\n if (refinedOption.end !== undefined) {\n if (hasNumericRefinement(currentRefinements, '<=', refinedOption.end)) {\n resolvedState = resolvedState.removeNumericRefinement(\n attribute,\n '<=',\n refinedOption.end\n );\n }\n resolvedState = resolvedState.addNumericRefinement(\n attribute,\n '<=',\n refinedOption.end\n );\n }\n\n if (typeof resolvedState.page === 'number') {\n resolvedState.page = 0;\n }\n\n return resolvedState;\n}\n\nfunction hasNumericRefinement(\n currentRefinements: SearchParameters.OperatorList,\n operator: SearchParameters.Operator,\n value: number\n) {\n return (\n currentRefinements[operator] !== undefined &&\n currentRefinements[operator]!.includes(value)\n );\n}\n\nexport default connectNumericMenu;\n","import { range } from '../../lib/utils';\n\nclass Paginator {\n public currentPage: number;\n public total: number;\n public padding: number;\n\n public constructor(params: {\n currentPage: number;\n total: number;\n padding: number;\n }) {\n this.currentPage = params.currentPage;\n this.total = params.total;\n this.padding = params.padding;\n }\n\n public pages() {\n const { total, currentPage, padding } = this;\n\n if (total === 0) return [0];\n\n const totalDisplayedPages = this.nbPagesDisplayed(padding, total);\n if (totalDisplayedPages === total) {\n return range({ end: total });\n }\n\n const paddingLeft = this.calculatePaddingLeft(\n currentPage,\n padding,\n total,\n totalDisplayedPages\n );\n const paddingRight = totalDisplayedPages - paddingLeft;\n\n const first = currentPage - paddingLeft;\n const last = currentPage + paddingRight;\n\n return range({ start: first, end: last });\n }\n\n public nbPagesDisplayed(padding: number, total: number) {\n return Math.min(2 * padding + 1, total);\n }\n\n public calculatePaddingLeft(\n current: number,\n padding: number,\n total: number,\n totalDisplayedPages: number\n ) {\n if (current <= padding) {\n return current;\n }\n\n if (current >= total - padding) {\n return totalDisplayedPages - (total - current);\n }\n\n return padding;\n }\n\n public isLastPage() {\n return this.currentPage === this.total - 1 || this.total === 0;\n }\n\n public isFirstPage() {\n return this.currentPage === 0;\n }\n}\n\nexport default Paginator;\n","import {\n checkRendering,\n createDocumentationMessageGenerator,\n noop,\n} from '../../lib/utils';\nimport Paginator from './Paginator';\nimport type { Connector, CreateURL, WidgetRenderState } from '../../types';\nimport type { SearchParameters } from 'algoliasearch-helper';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'pagination',\n connector: true,\n});\n\nexport type PaginationConnectorParams = {\n /**\n * The total number of pages to browse.\n */\n totalPages?: number;\n\n /**\n * The padding of pages to show around the current page\n * @default 3\n */\n padding?: number;\n};\n\nexport type PaginationRenderState = {\n /** Creates URLs for the next state, the number is the page to generate the URL for. */\n createURL: CreateURL;\n\n /** Sets the current page and triggers a search. */\n refine(page: number): void;\n\n /** true if this search returned more than one page */\n canRefine: boolean;\n\n /** The number of the page currently displayed. */\n currentRefinement: number;\n\n /** The number of hits computed for the last query (can be approximated). */\n nbHits: number;\n\n /** The number of pages for the result set. */\n nbPages: number;\n\n /** The actual pages relevant to the current situation and padding. */\n pages: number[];\n\n /** true if the current page is also the first page. */\n isFirstPage: boolean;\n\n /** true if the current page is also the last page. */\n isLastPage: boolean;\n};\n\nexport type PaginationWidgetDescription = {\n $$type: 'ais.pagination';\n renderState: PaginationRenderState;\n indexRenderState: {\n pagination: WidgetRenderState<\n PaginationRenderState,\n PaginationConnectorParams\n >;\n };\n indexUiState: {\n page: number;\n };\n};\n\nexport type PaginationConnector = Connector<\n PaginationWidgetDescription,\n PaginationConnectorParams\n>;\n\n/**\n * **Pagination** connector provides the logic to build a widget that will let the user\n * choose the current page of the results.\n *\n * When using the pagination with Algolia, you should be aware that the engine won't provide you pages\n * beyond the 1000th hits by default. You can find more information on the [Algolia documentation](https://www.algolia.com/doc/guides/searching/pagination/#pagination-limitations).\n */\nconst connectPagination: PaginationConnector = function connectPagination(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const { totalPages, padding = 3 } = widgetParams || {};\n\n const pager = new Paginator({\n currentPage: 0,\n total: 0,\n padding,\n });\n\n type ConnectorState = {\n refine?(page: number): void;\n createURL?(state: SearchParameters): (page: number) => string;\n };\n\n const connectorState: ConnectorState = {};\n\n function getMaxPage({ nbPages }: { nbPages: number }) {\n return totalPages !== undefined ? Math.min(totalPages, nbPages) : nbPages;\n }\n\n return {\n $$type: 'ais.pagination',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state.setQueryParameter('page', undefined);\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const page = searchParameters.page || 0;\n\n if (!page) {\n return uiState;\n }\n\n return {\n ...uiState,\n page: page + 1,\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n const page = uiState.page ? uiState.page - 1 : 0;\n\n return searchParameters.setQueryParameter('page', page);\n },\n\n getWidgetRenderState({ results, helper, state, createURL }) {\n if (!connectorState.refine) {\n connectorState.refine = (page) => {\n helper.setPage(page);\n helper.search();\n };\n }\n\n if (!connectorState.createURL) {\n connectorState.createURL = (helperState) => (page) =>\n createURL(helperState.setPage(page));\n }\n\n const page = state.page || 0;\n const nbPages = getMaxPage(results || { nbPages: 0 });\n pager.currentPage = page;\n pager.total = nbPages;\n\n return {\n createURL: connectorState.createURL(state),\n refine: connectorState.refine,\n canRefine: nbPages > 1,\n currentRefinement: page,\n nbHits: results?.nbHits || 0,\n nbPages,\n pages: results ? pager.pages() : [],\n isFirstPage: pager.isFirstPage(),\n isLastPage: pager.isLastPage(),\n widgetParams,\n };\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n pagination: this.getWidgetRenderState(renderOptions),\n };\n },\n };\n };\n};\n\nexport default connectPagination;\n","import type {\n AlgoliaSearchHelper,\n SearchParameters,\n SearchResults,\n} from 'algoliasearch-helper';\nimport type { SendEventForFacet } from '../../lib/utils';\nimport {\n checkRendering,\n createDocumentationMessageGenerator,\n convertNumericRefinementsToFilters,\n isFiniteNumber,\n find,\n noop,\n} from '../../lib/utils';\nimport type { InsightsEvent } from '../../middlewares';\nimport type { Connector, InstantSearch, WidgetRenderState } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator(\n { name: 'range-input', connector: true },\n { name: 'range-slider', connector: true }\n);\n\nconst $$type = 'ais.range';\n\nexport type RangeMin = number | undefined;\nexport type RangeMax = number | undefined;\n\n// @MAJOR: potentially we should consolidate these types\nexport type RangeBoundaries = [RangeMin, RangeMax];\nexport type Range = {\n min: RangeMin;\n max: RangeMax;\n};\n\nexport type RangeRenderState = {\n /**\n * Sets a range to filter the results on. Both values\n * are optional, and will default to the higher and lower bounds. You can use `undefined` to remove a\n * previously set bound or to set an infinite bound.\n * @param rangeValue tuple of [min, max] bounds\n */\n refine(rangeValue: RangeBoundaries): void;\n\n /**\n * Indicates whether this widget can be refined\n */\n canRefine: boolean;\n\n /**\n * Send an event to the insights middleware\n */\n sendEvent: SendEventForFacet;\n\n /**\n * Maximum range possible for this search\n */\n range: Range;\n\n /**\n * Current refinement of the search\n */\n start: RangeBoundaries;\n\n /**\n * Transform for the rendering `from` and/or `to` values.\n * Both functions take a `number` as input and should output a `string`.\n */\n format: {\n from(fromValue: number): string;\n to(toValue: number): string;\n };\n};\n\nexport type RangeConnectorParams = {\n /**\n * Name of the attribute for faceting.\n */\n attribute: string;\n\n /**\n * Minimal range value, default to automatically computed from the result set.\n */\n min?: number;\n\n /**\n * Maximal range value, default to automatically computed from the result set.\n */\n max?: number;\n\n /**\n * Number of digits after decimal point to use.\n */\n precision?: number;\n};\n\nexport type RangeWidgetDescription = {\n $$type: 'ais.range';\n renderState: RangeRenderState;\n indexRenderState: {\n range: {\n [attribute: string]: WidgetRenderState<\n RangeRenderState,\n RangeConnectorParams\n >;\n };\n };\n indexUiState: {\n range: {\n // @TODO: this could possibly become `${number}:${number}` later\n [attribute: string]: string;\n };\n };\n};\n\nexport type RangeConnector = Connector<\n RangeWidgetDescription,\n RangeConnectorParams\n>;\n\nfunction toPrecision({\n min,\n max,\n precision,\n}: {\n min?: number;\n max?: number;\n precision: number;\n}) {\n const pow = Math.pow(10, precision);\n\n return {\n min: min ? Math.floor(min * pow) / pow : min,\n max: max ? Math.ceil(max * pow) / pow : max,\n };\n}\n\n/**\n * **Range** connector provides the logic to create custom widget that will let\n * the user refine results using a numeric range.\n *\n * This connectors provides a `refine()` function that accepts bounds. It will also provide\n * information about the min and max bounds for the current result set.\n */\nconst connectRange: RangeConnector = function connectRange(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n attribute = '',\n min: minBound,\n max: maxBound,\n precision = 0,\n } = widgetParams || {};\n\n if (!attribute) {\n throw new Error(withUsage('The `attribute` option is required.'));\n }\n\n if (\n isFiniteNumber(minBound) &&\n isFiniteNumber(maxBound) &&\n minBound > maxBound\n ) {\n throw new Error(withUsage(\"The `max` option can't be lower than `min`.\"));\n }\n\n const formatToNumber = (v: string | number) =>\n Number(Number(v).toFixed(precision));\n\n const rangeFormatter = {\n from: (v: number) => v.toLocaleString(),\n to: (v: number) => formatToNumber(v).toLocaleString(),\n };\n\n // eslint-disable-next-line complexity\n const getRefinedState = (\n helper: AlgoliaSearchHelper,\n currentRange: Range,\n nextMin: RangeMin | string,\n nextMax: RangeMax | string\n ) => {\n let resolvedState = helper.state;\n const { min: currentRangeMin, max: currentRangeMax } = currentRange;\n\n const [min] = resolvedState.getNumericRefinement(attribute, '>=') || [];\n const [max] = resolvedState.getNumericRefinement(attribute, '<=') || [];\n\n const isResetMin = nextMin === undefined || nextMin === '';\n const isResetMax = nextMax === undefined || nextMax === '';\n\n const { min: nextMinAsNumber, max: nextMaxAsNumber } = toPrecision({\n min: !isResetMin ? parseFloat(nextMin as string) : undefined,\n max: !isResetMax ? parseFloat(nextMax as string) : undefined,\n precision,\n });\n\n let newNextMin: RangeMin;\n if (!isFiniteNumber(minBound) && currentRangeMin === nextMinAsNumber) {\n newNextMin = undefined;\n } else if (isFiniteNumber(minBound) && isResetMin) {\n newNextMin = minBound;\n } else {\n newNextMin = nextMinAsNumber;\n }\n\n let newNextMax: RangeMax;\n if (!isFiniteNumber(maxBound) && currentRangeMax === nextMaxAsNumber) {\n newNextMax = undefined;\n } else if (isFiniteNumber(maxBound) && isResetMax) {\n newNextMax = maxBound;\n } else {\n newNextMax = nextMaxAsNumber;\n }\n\n const isResetNewNextMin = newNextMin === undefined;\n\n const isGreaterThanCurrentRange =\n isFiniteNumber(currentRangeMin) && currentRangeMin <= newNextMin!;\n const isMinValid =\n isResetNewNextMin ||\n (isFiniteNumber(newNextMin) &&\n (!isFiniteNumber(currentRangeMin) || isGreaterThanCurrentRange));\n\n const isResetNewNextMax = newNextMax === undefined;\n const isLowerThanRange =\n isFiniteNumber(newNextMax) && currentRangeMax! >= newNextMax;\n const isMaxValid =\n isResetNewNextMax ||\n (isFiniteNumber(newNextMax) &&\n (!isFiniteNumber(currentRangeMax) || isLowerThanRange));\n\n const hasMinChange = min !== newNextMin;\n const hasMaxChange = max !== newNextMax;\n\n if ((hasMinChange || hasMaxChange) && isMinValid && isMaxValid) {\n resolvedState = resolvedState.removeNumericRefinement(attribute);\n\n if (isFiniteNumber(newNextMin)) {\n resolvedState = resolvedState.addNumericRefinement(\n attribute,\n '>=',\n newNextMin\n );\n }\n\n if (isFiniteNumber(newNextMax)) {\n resolvedState = resolvedState.addNumericRefinement(\n attribute,\n '<=',\n newNextMax\n );\n }\n\n return resolvedState.resetPage();\n }\n\n return null;\n };\n\n const sendEventWithRefinedState = (\n refinedState: SearchParameters | null,\n instantSearchInstance: InstantSearch,\n helper: AlgoliaSearchHelper,\n eventName = 'Filter Applied'\n ) => {\n const filters = convertNumericRefinementsToFilters(\n refinedState,\n attribute\n );\n if (filters && filters.length > 0) {\n instantSearchInstance.sendEventToInsights({\n insightsMethod: 'clickedFilters',\n widgetType: $$type,\n eventType: 'click',\n payload: {\n eventName,\n index: helper.getIndex(),\n filters,\n },\n attribute,\n });\n }\n };\n\n const createSendEvent =\n (\n instantSearchInstance: InstantSearch,\n helper: AlgoliaSearchHelper,\n currentRange: Range\n ) =>\n (...args: [InsightsEvent] | [string, string, string?]) => {\n if (args.length === 1) {\n instantSearchInstance.sendEventToInsights(args[0]);\n return;\n }\n\n const [eventType, facetValue, eventName] = args;\n if (eventType !== 'click') {\n return;\n }\n const [nextMin, nextMax] = facetValue;\n const refinedState = getRefinedState(\n helper,\n currentRange,\n nextMin,\n nextMax\n );\n sendEventWithRefinedState(\n refinedState,\n instantSearchInstance,\n helper,\n eventName\n );\n };\n\n function _getCurrentRange(\n stats: Partial>\n ) {\n let min: number;\n if (isFiniteNumber(minBound)) {\n min = minBound;\n } else if (isFiniteNumber(stats.min)) {\n min = stats.min;\n } else {\n min = 0;\n }\n\n let max: number;\n if (isFiniteNumber(maxBound)) {\n max = maxBound;\n } else if (isFiniteNumber(stats.max)) {\n max = stats.max;\n } else {\n max = 0;\n }\n\n return toPrecision({ min, max, precision });\n }\n\n function _getCurrentRefinement(\n helper: AlgoliaSearchHelper\n ): RangeBoundaries {\n const [minValue] = helper.getNumericRefinement(attribute, '>=') || [];\n\n const [maxValue] = helper.getNumericRefinement(attribute, '<=') || [];\n\n const min = isFiniteNumber(minValue) ? minValue : -Infinity;\n const max = isFiniteNumber(maxValue) ? maxValue : Infinity;\n\n return [min, max];\n }\n\n function _refine(\n instantSearchInstance: InstantSearch,\n helper: AlgoliaSearchHelper,\n currentRange: Range\n ) {\n return ([nextMin, nextMax]: RangeBoundaries = [undefined, undefined]) => {\n const refinedState = getRefinedState(\n helper,\n currentRange,\n nextMin,\n nextMax\n );\n if (refinedState) {\n sendEventWithRefinedState(\n refinedState,\n instantSearchInstance,\n helper\n );\n helper.setState(refinedState).search();\n }\n };\n }\n\n return {\n $$type,\n\n init(initOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance: initOptions.instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n range: {\n ...renderState.range,\n [attribute]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState({ results, helper, instantSearchInstance }) {\n const facetsFromResults = (results && results.disjunctiveFacets) || [];\n const facet = find(\n facetsFromResults,\n (facetResult) => facetResult.name === attribute\n );\n const stats = (facet && facet.stats) || {\n min: undefined,\n max: undefined,\n };\n\n const currentRange = _getCurrentRange(stats);\n const start = _getCurrentRefinement(helper);\n\n let refine: ReturnType;\n\n if (!results) {\n // On first render pass an empty range\n // to be able to bypass the validation\n // related to it\n refine = _refine(instantSearchInstance, helper, {\n min: undefined,\n max: undefined,\n });\n } else {\n refine = _refine(instantSearchInstance, helper, currentRange);\n }\n\n return {\n refine,\n canRefine: currentRange.min !== currentRange.max,\n format: rangeFormatter,\n range: currentRange,\n sendEvent: createSendEvent(\n instantSearchInstance,\n helper,\n currentRange\n ),\n widgetParams: {\n ...widgetParams,\n precision,\n },\n start,\n };\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state\n .removeDisjunctiveFacet(attribute)\n .removeNumericRefinement(attribute);\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const { '>=': min = [], '<=': max = [] } =\n searchParameters.getNumericRefinements(attribute);\n\n if (min.length === 0 && max.length === 0) {\n return uiState;\n }\n\n return {\n ...uiState,\n range: {\n ...uiState.range,\n [attribute]: `${min}:${max}`,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n let widgetSearchParameters = searchParameters\n .addDisjunctiveFacet(attribute)\n .setQueryParameters({\n numericRefinements: {\n ...searchParameters.numericRefinements,\n [attribute]: {},\n },\n });\n\n if (isFiniteNumber(minBound)) {\n widgetSearchParameters = widgetSearchParameters.addNumericRefinement(\n attribute,\n '>=',\n minBound\n );\n }\n\n if (isFiniteNumber(maxBound)) {\n widgetSearchParameters = widgetSearchParameters.addNumericRefinement(\n attribute,\n '<=',\n maxBound\n );\n }\n\n const value = uiState.range && uiState.range[attribute];\n\n if (!value || value.indexOf(':') === -1) {\n return widgetSearchParameters;\n }\n\n const [lowerBound, upperBound] = value.split(':').map(parseFloat);\n\n if (\n isFiniteNumber(lowerBound) &&\n (!isFiniteNumber(minBound) || minBound < lowerBound)\n ) {\n widgetSearchParameters =\n widgetSearchParameters.removeNumericRefinement(attribute, '>=');\n widgetSearchParameters = widgetSearchParameters.addNumericRefinement(\n attribute,\n '>=',\n lowerBound\n );\n }\n\n if (\n isFiniteNumber(upperBound) &&\n (!isFiniteNumber(maxBound) || upperBound < maxBound)\n ) {\n widgetSearchParameters =\n widgetSearchParameters.removeNumericRefinement(attribute, '<=');\n widgetSearchParameters = widgetSearchParameters.addNumericRefinement(\n attribute,\n '<=',\n upperBound\n );\n }\n\n return widgetSearchParameters;\n },\n };\n };\n};\n\nexport default connectRange;\n","import type { AlgoliaSearchHelper, SearchResults } from 'algoliasearch-helper';\nimport type { SendEventForFacet } from '../../lib/utils';\nimport {\n escapeFacets,\n TAG_PLACEHOLDER,\n TAG_REPLACEMENT,\n checkRendering,\n createDocumentationMessageGenerator,\n createSendEventForFacet,\n noop,\n} from '../../lib/utils';\nimport type {\n Connector,\n TransformItems,\n SortBy,\n RenderOptions,\n Widget,\n InitOptions,\n FacetHit,\n CreateURL,\n WidgetRenderState,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'refinement-list',\n connector: true,\n});\n\nconst DEFAULT_SORT = ['isRefined', 'count:desc', 'name:asc'];\n\nexport type RefinementListItem = {\n /**\n * The value of the refinement list item.\n */\n value: string;\n /**\n * Human-readable value of the refinement list item.\n */\n label: string;\n /**\n * Human-readable value of the searched refinement list item.\n */\n highlighted?: string;\n /**\n * Number of matched results after refinement is applied.\n */\n count: number;\n /**\n * Indicates if the list item is refined.\n */\n isRefined: boolean;\n};\n\nexport type RefinementListConnectorParams = {\n /**\n * The name of the attribute in the records.\n */\n attribute: string;\n /**\n * How the filters are combined together.\n */\n operator?: 'and' | 'or';\n /**\n * The max number of items to display when\n * `showMoreLimit` is not set or if the widget is showing less value.\n */\n limit?: number;\n /**\n * Whether to display a button that expands the number of items.\n */\n showMore?: boolean;\n /**\n * The max number of items to display if the widget\n * is showing more items.\n */\n showMoreLimit?: number;\n /**\n * How to sort refinements. Possible values: `count|isRefined|name:asc|name:desc`.\n *\n * You can also use a sort function that behaves like the standard Javascript [compareFunction](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#Syntax).\n *\n * If a facetOrdering is set in the index settings, it is used when sortBy isn't passed\n */\n sortBy?: SortBy;\n /**\n * Escapes the content of the facet values.\n */\n escapeFacetValues?: boolean;\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n};\n\nexport type RefinementListRenderState = {\n /**\n * The list of filtering values returned from Algolia API.\n */\n items: RefinementListItem[];\n /**\n * indicates whether the results are exhaustive (complete)\n */\n hasExhaustiveItems: boolean;\n /**\n * Creates the next state url for a selected refinement.\n */\n createURL: CreateURL;\n /**\n * Action to apply selected refinements.\n */\n refine(value: string): void;\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEventForFacet;\n /**\n * Searches for values inside the list.\n */\n searchForItems(query: string): void;\n /**\n * `true` if the values are from an index search.\n */\n isFromSearch: boolean;\n /**\n * `true` if a refinement can be applied.\n */\n canRefine: boolean;\n /**\n * `true` if the toggleShowMore button can be activated (enough items to display more or\n * already displaying more than `limit` items)\n */\n canToggleShowMore: boolean;\n /**\n * True if the menu is displaying all the menu items.\n */\n isShowingMore: boolean;\n /**\n * Toggles the number of values displayed between `limit` and `showMoreLimit`.\n */\n toggleShowMore(): void;\n};\n\nexport type RefinementListWidgetDescription = {\n $$type: 'ais.refinementList';\n renderState: RefinementListRenderState;\n indexRenderState: {\n refinementList: {\n [attribute: string]: WidgetRenderState<\n RefinementListRenderState,\n RefinementListConnectorParams\n >;\n };\n };\n indexUiState: {\n refinementList: {\n [attribute: string]: string[];\n };\n };\n};\n\nexport type RefinementListConnector = Connector<\n RefinementListWidgetDescription,\n RefinementListConnectorParams\n>;\n\n/**\n * **RefinementList** connector provides the logic to build a custom widget that\n * will let the user filter the results based on the values of a specific facet.\n *\n * **Requirement:** the attribute passed as `attribute` must be present in\n * attributesForFaceting of the searched index.\n *\n * This connector provides:\n * - a `refine()` function to select an item.\n * - a `toggleShowMore()` function to display more or less items\n * - a `searchForItems()` function to search within the items.\n */\nconst connectRefinementList: RefinementListConnector =\n function connectRefinementList(renderFn, unmountFn = noop) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n attribute,\n operator = 'or',\n limit = 10,\n showMore = false,\n showMoreLimit = 20,\n sortBy = DEFAULT_SORT,\n escapeFacetValues = true,\n transformItems = ((items) => items) as NonNullable<\n RefinementListConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n type ThisWidget = Widget<\n RefinementListWidgetDescription & { widgetParams: typeof widgetParams }\n >;\n\n if (!attribute) {\n throw new Error(withUsage('The `attribute` option is required.'));\n }\n\n if (!/^(and|or)$/.test(operator)) {\n throw new Error(\n withUsage(\n `The \\`operator\\` must one of: \\`\"and\"\\`, \\`\"or\"\\` (got \"${operator}\").`\n )\n );\n }\n\n if (showMore === true && showMoreLimit <= limit) {\n throw new Error(\n withUsage('`showMoreLimit` should be greater than `limit`.')\n );\n }\n\n const formatItems = ({\n name: label,\n escapedValue: value,\n ...item\n }: SearchResults.FacetValue): RefinementListItem => ({\n ...item,\n value,\n label,\n highlighted: label,\n });\n\n let lastResultsFromMainSearch: SearchResults;\n let lastItemsFromMainSearch: RefinementListItem[] = [];\n let hasExhaustiveItems = true;\n let triggerRefine: RefinementListRenderState['refine'] | undefined;\n let sendEvent: RefinementListRenderState['sendEvent'] | undefined;\n\n let isShowingMore = false;\n // Provide the same function to the `renderFn` so that way the user\n // has to only bind it once when `isFirstRendering` for instance\n let toggleShowMore = () => {};\n function cachedToggleShowMore() {\n toggleShowMore();\n }\n\n function createToggleShowMore(\n renderOptions: RenderOptions,\n widget: ThisWidget\n ) {\n return () => {\n isShowingMore = !isShowingMore;\n widget.render!(renderOptions);\n };\n }\n\n function getLimit() {\n return isShowingMore ? showMoreLimit : limit;\n }\n\n let searchForFacetValues: (\n renderOptions: RenderOptions | InitOptions\n ) => RefinementListRenderState['searchForItems'] = () => () => {};\n\n const createSearchForFacetValues = function (\n helper: AlgoliaSearchHelper,\n widget: ThisWidget\n ) {\n return (renderOptions: RenderOptions | InitOptions) =>\n (query: string) => {\n const { instantSearchInstance, results: searchResults } =\n renderOptions;\n if (query === '' && lastItemsFromMainSearch) {\n // render with previous data from the helper.\n renderFn(\n {\n ...widget.getWidgetRenderState({\n ...renderOptions,\n results: lastResultsFromMainSearch,\n }),\n instantSearchInstance,\n },\n false\n );\n } else {\n const tags = {\n highlightPreTag: escapeFacetValues\n ? TAG_PLACEHOLDER.highlightPreTag\n : TAG_REPLACEMENT.highlightPreTag,\n highlightPostTag: escapeFacetValues\n ? TAG_PLACEHOLDER.highlightPostTag\n : TAG_REPLACEMENT.highlightPostTag,\n };\n\n helper\n .searchForFacetValues(\n attribute,\n query,\n // We cap the `maxFacetHits` value to 100 because the Algolia API\n // doesn't support a greater number.\n // See https://www.algolia.com/doc/api-reference/api-parameters/maxFacetHits/\n Math.min(getLimit(), 100),\n tags\n )\n .then((results) => {\n const facetValues = escapeFacetValues\n ? escapeFacets(results.facetHits)\n : results.facetHits;\n\n const normalizedFacetValues = transformItems(\n facetValues.map(({ escapedValue, value, ...item }) => ({\n ...item,\n value: escapedValue,\n label: value,\n })),\n { results: searchResults }\n );\n\n renderFn(\n {\n ...widget.getWidgetRenderState({\n ...renderOptions,\n results: lastResultsFromMainSearch,\n }),\n items: normalizedFacetValues,\n canToggleShowMore: false,\n canRefine: true,\n isFromSearch: true,\n instantSearchInstance,\n },\n false\n );\n });\n }\n };\n };\n\n return {\n $$type: 'ais.refinementList' as const,\n\n init(initOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance: initOptions.instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n refinementList: {\n ...renderState.refinementList,\n [attribute]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState(renderOptions) {\n const { results, state, createURL, instantSearchInstance, helper } =\n renderOptions;\n let items: RefinementListItem[] = [];\n let facetValues: SearchResults.FacetValue[] | FacetHit[] = [];\n\n if (!sendEvent || !triggerRefine || !searchForFacetValues) {\n sendEvent = createSendEventForFacet({\n instantSearchInstance,\n helper,\n attribute,\n widgetType: this.$$type,\n });\n\n triggerRefine = (facetValue) => {\n sendEvent!('click', facetValue);\n helper.toggleFacetRefinement(attribute, facetValue).search();\n };\n\n searchForFacetValues = createSearchForFacetValues(helper, this);\n }\n\n if (results) {\n const values = results.getFacetValues(attribute, {\n sortBy,\n facetOrdering: sortBy === DEFAULT_SORT,\n });\n facetValues = values && Array.isArray(values) ? values : [];\n items = transformItems(\n facetValues.slice(0, getLimit()).map(formatItems),\n { results }\n );\n\n const maxValuesPerFacetConfig = state.maxValuesPerFacet;\n const currentLimit = getLimit();\n // If the limit is the max number of facet retrieved it is impossible to know\n // if the facets are exhaustive. The only moment we are sure it is exhaustive\n // is when it is strictly under the number requested unless we know that another\n // widget has requested more values (maxValuesPerFacet > getLimit()).\n // Because this is used for making the search of facets unable or not, it is important\n // to be conservative here.\n hasExhaustiveItems =\n maxValuesPerFacetConfig! > currentLimit\n ? facetValues.length <= currentLimit\n : facetValues.length < currentLimit;\n\n lastResultsFromMainSearch = results;\n lastItemsFromMainSearch = items;\n\n if (renderOptions.results) {\n toggleShowMore = createToggleShowMore(renderOptions, this);\n }\n }\n\n // Do not mistake searchForFacetValues and searchFacetValues which is the actual search\n // function\n const searchFacetValues =\n searchForFacetValues && searchForFacetValues(renderOptions);\n\n const canShowLess =\n isShowingMore && lastItemsFromMainSearch.length > limit;\n const canShowMore = showMore && !hasExhaustiveItems;\n\n const canToggleShowMore = canShowLess || canShowMore;\n\n return {\n createURL: (facetValue) =>\n createURL(\n state.resetPage().toggleFacetRefinement(attribute, facetValue)\n ),\n items,\n refine: triggerRefine,\n searchForItems: searchFacetValues,\n isFromSearch: false,\n canRefine: items.length > 0,\n widgetParams,\n isShowingMore,\n canToggleShowMore,\n toggleShowMore: cachedToggleShowMore,\n sendEvent,\n hasExhaustiveItems,\n };\n },\n\n dispose({ state }) {\n unmountFn();\n\n const withoutMaxValuesPerFacet = state.setQueryParameter(\n 'maxValuesPerFacet',\n undefined\n );\n if (operator === 'and') {\n return withoutMaxValuesPerFacet.removeFacet(attribute);\n }\n return withoutMaxValuesPerFacet.removeDisjunctiveFacet(attribute);\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const values =\n operator === 'or'\n ? searchParameters.getDisjunctiveRefinements(attribute)\n : searchParameters.getConjunctiveRefinements(attribute);\n\n if (!values.length) {\n return uiState;\n }\n\n return {\n ...uiState,\n refinementList: {\n ...uiState.refinementList,\n [attribute]: values,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n const isDisjunctive = operator === 'or';\n const values =\n uiState.refinementList && uiState.refinementList[attribute];\n\n const withoutRefinements =\n searchParameters.clearRefinements(attribute);\n const withFacetConfiguration = isDisjunctive\n ? withoutRefinements.addDisjunctiveFacet(attribute)\n : withoutRefinements.addFacet(attribute);\n\n const currentMaxValuesPerFacet =\n withFacetConfiguration.maxValuesPerFacet || 0;\n\n const nextMaxValuesPerFacet = Math.max(\n currentMaxValuesPerFacet,\n showMore ? showMoreLimit : limit\n );\n\n const withMaxValuesPerFacet =\n withFacetConfiguration.setQueryParameter(\n 'maxValuesPerFacet',\n nextMaxValuesPerFacet\n );\n\n if (!values) {\n const key = isDisjunctive\n ? 'disjunctiveFacetsRefinements'\n : 'facetsRefinements';\n\n return withMaxValuesPerFacet.setQueryParameters({\n [key]: {\n ...withMaxValuesPerFacet[key],\n [attribute]: [],\n },\n });\n }\n\n return values.reduce(\n (parameters, value) =>\n isDisjunctive\n ? parameters.addDisjunctiveFacetRefinement(attribute, value)\n : parameters.addFacetRefinement(attribute, value),\n withMaxValuesPerFacet\n );\n },\n };\n };\n };\n\nexport default connectRefinementList;\n","import {\n checkRendering,\n createDocumentationMessageGenerator,\n noop,\n} from '../../lib/utils';\nimport type { Connector, WidgetRenderState } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'search-box',\n connector: true,\n});\n\nexport type SearchBoxConnectorParams = {\n /**\n * A function that will be called every time\n * a new value for the query is set. The first parameter is the query and the second is a\n * function to actually trigger the search. The function takes the query as the parameter.\n *\n * This queryHook can be used to debounce the number of searches done from the searchBox.\n */\n queryHook?: (query: string, hook: (value: string) => void) => void;\n};\n\n/**\n * @typedef {Object} CustomSearchBoxWidgetParams\n * @property {function(string, function(string))} [queryHook = undefined] A function that will be called every time\n * a new value for the query is set. The first parameter is the query and the second is a\n * function to actually trigger the search. The function takes the query as the parameter.\n *\n * This queryHook can be used to debounce the number of searches done from the searchBox.\n */\n\nexport type SearchBoxRenderState = {\n /**\n * The query from the last search.\n */\n query: string;\n /**\n * Sets a new query and searches.\n */\n refine: (value: string) => void;\n /**\n * Remove the query and perform search.\n */\n clear: () => void;\n /**\n * `true` if the search results takes more than a certain time to come back\n * from Algolia servers. This can be configured on the InstantSearch constructor with the attribute\n * `stalledSearchDelay` which is 200ms, by default.\n */\n isSearchStalled: boolean;\n};\n\nexport type SearchBoxWidgetDescription = {\n $$type: 'ais.searchBox';\n renderState: SearchBoxRenderState;\n indexRenderState: {\n searchBox: WidgetRenderState<\n SearchBoxRenderState,\n SearchBoxConnectorParams\n >;\n };\n indexUiState: {\n query: string;\n };\n};\n\nexport type SearchBoxConnector = Connector<\n SearchBoxWidgetDescription,\n SearchBoxConnectorParams\n>;\n\nconst defaultQueryHook: SearchBoxConnectorParams['queryHook'] = (query, hook) =>\n hook(query);\n\n/**\n * **SearchBox** connector provides the logic to build a widget that will let the user search for a query.\n *\n * The connector provides to the rendering: `refine()` to set the query. The behaviour of this function\n * may be impacted by the `queryHook` widget parameter.\n */\nconst connectSearchBox: SearchBoxConnector = function connectSearchBox(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const { queryHook = defaultQueryHook } = widgetParams || {};\n\n let _refine: SearchBoxRenderState['refine'];\n let _clear: SearchBoxRenderState['clear'];\n\n return {\n $$type: 'ais.searchBox',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state.setQueryParameter('query', undefined);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n searchBox: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ helper, searchMetadata, state }) {\n if (!_refine) {\n _refine = (query) => {\n queryHook(query, (q) => helper.setQuery(q).search());\n };\n\n _clear = () => {\n helper.setQuery('').search();\n };\n }\n\n return {\n query: state.query || '',\n refine: _refine,\n clear: _clear,\n widgetParams,\n isSearchStalled: searchMetadata.isSearchStalled,\n };\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const query = searchParameters.query || '';\n\n if (query === '' || (uiState && uiState.query === query)) {\n return uiState;\n }\n\n return {\n ...uiState,\n query,\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n return searchParameters.setQueryParameter('query', uiState.query || '');\n },\n };\n };\n};\n\nexport default connectSearchBox;\n","import {\n checkRendering,\n createDocumentationMessageGenerator,\n find,\n warning,\n noop,\n} from '../../lib/utils';\nimport type { Connector, TransformItems, WidgetRenderState } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'sort-by',\n connector: true,\n});\n\n/**\n * The **SortBy** connector provides the logic to build a custom widget that will display a\n * list of indices. With Algolia, this is most commonly used for changing ranking strategy. This allows\n * a user to change how the hits are being sorted.\n */\n\nexport type SortByItem = {\n /**\n * The name of the index to target.\n */\n value: string;\n /**\n * The label of the index to display.\n */\n label: string;\n};\n\nexport type SortByConnectorParams = {\n /**\n * Array of objects defining the different indices to choose from.\n */\n items: SortByItem[];\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n};\n\nexport type SortByRenderState = {\n /**\n * The initially selected index.\n */\n initialIndex?: string;\n /**\n * The currently selected index.\n */\n currentRefinement: string;\n /**\n * All the available indices\n */\n options: SortByItem[];\n /**\n * Switches indices and triggers a new search.\n */\n refine: (value: string) => void;\n /**\n * `true` if the last search contains no result.\n */\n hasNoResults: boolean;\n};\n\nexport type SortByWidgetDescription = {\n $$type: 'ais.sortBy';\n renderState: SortByRenderState;\n indexRenderState: {\n sortBy: WidgetRenderState;\n };\n indexUiState: {\n sortBy: string;\n };\n};\n\nexport type SortByConnector = Connector<\n SortByWidgetDescription,\n SortByConnectorParams\n>;\n\nconst connectSortBy: SortByConnector = function connectSortBy(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n const connectorState: ConnectorState = {};\n\n type ConnectorState = {\n setIndex?(indexName: string): void;\n initialIndex?: string;\n };\n\n return (widgetParams) => {\n const {\n items,\n transformItems = ((x) => x) as NonNullable<\n SortByConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n if (!Array.isArray(items)) {\n throw new Error(\n withUsage('The `items` option expects an array of objects.')\n );\n }\n\n return {\n $$type: 'ais.sortBy',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n const widgetRenderState = this.getWidgetRenderState(initOptions);\n const currentIndex = widgetRenderState.currentRefinement;\n const isCurrentIndexInItems = find(\n items,\n (item) => item.value === currentIndex\n );\n\n warning(\n isCurrentIndexInItems !== undefined,\n `The index named \"${currentIndex}\" is not listed in the \\`items\\` of \\`sortBy\\`.`\n );\n\n renderFn(\n {\n ...widgetRenderState,\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return connectorState.initialIndex\n ? state.setIndex(connectorState.initialIndex)\n : state;\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n sortBy: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ results, helper, state, parent }) {\n if (!connectorState.initialIndex && parent) {\n connectorState.initialIndex = parent.getIndexName();\n }\n if (!connectorState.setIndex) {\n connectorState.setIndex = (indexName) => {\n helper.setIndex(indexName).search();\n };\n }\n\n return {\n currentRefinement: state.index,\n options: transformItems(items, { results }),\n refine: connectorState.setIndex,\n hasNoResults: results ? results.nbHits === 0 : true,\n widgetParams,\n };\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const currentIndex = searchParameters.index;\n\n return {\n ...uiState,\n sortBy:\n currentIndex !== connectorState.initialIndex\n ? currentIndex\n : undefined,\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n return searchParameters.setQueryParameter(\n 'index',\n uiState.sortBy ||\n connectorState.initialIndex ||\n searchParameters.index\n );\n },\n };\n };\n};\n\nexport default connectSortBy;\n","import type {\n AlgoliaSearchHelper,\n SearchParameters,\n SearchResults,\n} from 'algoliasearch-helper';\nimport {\n checkRendering,\n createDocumentationLink,\n createDocumentationMessageGenerator,\n noop,\n warning,\n} from '../../lib/utils';\nimport type {\n Connector,\n InstantSearch,\n CreateURL,\n WidgetRenderState,\n} from '../../types';\nimport type { InsightsEvent } from '../../middlewares';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'rating-menu',\n connector: true,\n});\n\nconst $$type = 'ais.ratingMenu';\n\nconst MAX_VALUES_PER_FACET_API_LIMIT = 1000;\nconst STEP = 1;\n\ntype SendEvent = (...args: [InsightsEvent] | [string, string, string?]) => void;\n\ntype CreateSendEvent = (createSendEventArgs: {\n instantSearchInstance: InstantSearch;\n helper: AlgoliaSearchHelper;\n getRefinedStar: () => number | number[] | undefined;\n attribute: string;\n}) => SendEvent;\n\nconst createSendEvent: CreateSendEvent =\n ({ instantSearchInstance, helper, getRefinedStar, attribute }) =>\n (...args) => {\n if (args.length === 1) {\n instantSearchInstance.sendEventToInsights(args[0]);\n return;\n }\n const [eventType, facetValue, eventName = 'Filter Applied'] = args;\n if (eventType !== 'click') {\n return;\n }\n const isRefined = getRefinedStar() === Number(facetValue);\n if (!isRefined) {\n instantSearchInstance.sendEventToInsights({\n insightsMethod: 'clickedFilters',\n widgetType: $$type,\n eventType,\n payload: {\n eventName,\n index: helper.getIndex(),\n filters: [`${attribute}>=${facetValue}`],\n },\n attribute,\n });\n }\n };\n\ntype StarRatingItems = {\n /**\n * Name corresponding to the number of stars.\n */\n name: string;\n /**\n * Human-readable name corresponding to the number of stars.\n */\n label: string;\n /**\n * Number of stars as string.\n */\n value: string;\n /**\n * Count of matched results corresponding to the number of stars.\n */\n count: number;\n /**\n * Array of length of maximum rating value with stars to display or not.\n */\n stars: boolean[];\n /**\n * Indicates if star rating refinement is applied.\n */\n isRefined: boolean;\n};\n\nexport type RatingMenuConnectorParams = {\n /**\n * Name of the attribute for faceting (eg. \"free_shipping\").\n */\n attribute: string;\n\n /**\n * The maximum rating value.\n */\n max?: number;\n};\n\nexport type RatingMenuRenderState = {\n /**\n * Possible star ratings the user can apply.\n */\n items: StarRatingItems[];\n\n /**\n * Creates an URL for the next state (takes the item value as parameter). Takes the value of an item as parameter.\n */\n createURL: CreateURL;\n\n /**\n * Indicates if search state can be refined.\n */\n canRefine: boolean;\n\n /**\n * Selects a rating to filter the results (takes the filter value as parameter). Takes the value of an item as parameter.\n */\n refine: (value: string) => void;\n\n /**\n * `true` if the last search contains no result.\n */\n hasNoResults: boolean;\n\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEvent;\n};\n\nexport type RatingMenuWidgetDescription = {\n $$type: 'ais.ratingMenu';\n renderState: RatingMenuRenderState;\n indexRenderState: {\n ratingMenu: {\n [attribute: string]: WidgetRenderState<\n RatingMenuRenderState,\n RatingMenuConnectorParams\n >;\n };\n };\n indexUiState: {\n ratingMenu: {\n [attribute: string]: number;\n };\n };\n};\n\nexport type RatingMenuConnector = Connector<\n RatingMenuWidgetDescription,\n RatingMenuConnectorParams\n>;\n\n/**\n * **StarRating** connector provides the logic to build a custom widget that will let\n * the user refine search results based on ratings.\n *\n * The connector provides to the rendering: `refine()` to select a value and\n * `items` that are the values that can be selected. `refine` should be used\n * with `items.value`.\n */\nconst connectRatingMenu: RatingMenuConnector = function connectRatingMenu(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const { attribute, max = 5 } = widgetParams || {};\n let sendEvent: SendEvent;\n\n if (!attribute) {\n throw new Error(withUsage('The `attribute` option is required.'));\n }\n\n const getRefinedStar = (state: SearchParameters) => {\n const values = state.getNumericRefinements(attribute);\n\n if (!values['>=']?.length) {\n return undefined;\n }\n\n return values['>='][0];\n };\n\n const getFacetsMaxDecimalPlaces = (\n facetResults: SearchResults.FacetValue[]\n ) => {\n let maxDecimalPlaces = 0;\n facetResults.forEach((facetResult) => {\n const [, decimal = ''] = facetResult.name.split('.');\n maxDecimalPlaces = Math.max(maxDecimalPlaces, decimal.length);\n });\n return maxDecimalPlaces;\n };\n\n const getFacetValuesWarningMessage = ({\n maxDecimalPlaces,\n maxFacets,\n maxValuesPerFacet,\n }: {\n maxDecimalPlaces: number;\n maxFacets: number;\n maxValuesPerFacet: number;\n }) => {\n const maxDecimalPlacesInRange = Math.max(\n 0,\n Math.floor(Math.log10(MAX_VALUES_PER_FACET_API_LIMIT / max))\n );\n const maxFacetsInRange = Math.min(\n MAX_VALUES_PER_FACET_API_LIMIT,\n Math.pow(10, maxDecimalPlacesInRange) * max\n );\n\n const solutions: string[] = [];\n\n if (maxFacets > MAX_VALUES_PER_FACET_API_LIMIT) {\n solutions.push(\n `- Update your records to lower the precision of the values in the \"${attribute}\" attribute (for example: ${(5.123456789).toPrecision(\n maxDecimalPlaces + 1\n )} to ${(5.123456789).toPrecision(maxDecimalPlacesInRange + 1)})`\n );\n }\n if (maxValuesPerFacet < maxFacetsInRange) {\n solutions.push(\n `- Increase the maximum number of facet values to ${maxFacetsInRange} using the \"configure\" widget ${createDocumentationLink(\n { name: 'configure' }\n )} and the \"maxValuesPerFacet\" parameter https://www.algolia.com/doc/api-reference/api-parameters/maxValuesPerFacet/`\n );\n }\n\n return `The ${attribute} attribute can have ${maxFacets} different values (0 to ${max} with a maximum of ${maxDecimalPlaces} decimals = ${maxFacets}) but you retrieved only ${maxValuesPerFacet} facet values. Therefore the number of results that match the refinements can be incorrect.\n ${\n solutions.length\n ? `To resolve this problem you can:\\n${solutions.join('\\n')}`\n : ``\n }`;\n };\n\n function getRefinedState(state: SearchParameters, facetValue: string) {\n const isRefined = getRefinedStar(state) === Number(facetValue);\n\n const emptyState = state.resetPage().removeNumericRefinement(attribute);\n\n if (!isRefined) {\n return emptyState\n .addNumericRefinement(attribute, '<=', max)\n .addNumericRefinement(attribute, '>=', Number(facetValue));\n }\n return emptyState;\n }\n\n const toggleRefinement = (\n helper: AlgoliaSearchHelper,\n facetValue: string\n ) => {\n sendEvent('click', facetValue);\n helper.setState(getRefinedState(helper.state, facetValue)).search();\n };\n\n type ConnectorState = {\n toggleRefinementFactory: (\n helper: AlgoliaSearchHelper\n ) => (facetValue: string) => void;\n createURLFactory: ({\n state,\n createURL,\n }: {\n state: SearchParameters;\n createURL: (createURLState: SearchParameters) => string;\n }) => (value: string) => string;\n };\n\n const connectorState: ConnectorState = {\n toggleRefinementFactory: (helper) => toggleRefinement.bind(null, helper),\n createURLFactory:\n ({ state, createURL }) =>\n (value) =>\n createURL(getRefinedState(state, value)),\n };\n\n return {\n $$type,\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n ratingMenu: {\n ...renderState.ratingMenu,\n [attribute]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState({\n helper,\n results,\n state,\n instantSearchInstance,\n createURL,\n }) {\n let facetValues: StarRatingItems[] = [];\n\n if (!sendEvent) {\n sendEvent = createSendEvent({\n instantSearchInstance,\n helper,\n getRefinedStar: () => getRefinedStar(helper.state),\n attribute,\n });\n }\n\n if (results) {\n const facetResults = results.getFacetValues(\n attribute,\n {}\n ) as SearchResults.FacetValue[];\n const maxValuesPerFacet = facetResults.length;\n\n const maxDecimalPlaces = getFacetsMaxDecimalPlaces(facetResults);\n const maxFacets = Math.pow(10, maxDecimalPlaces) * max;\n\n warning(\n maxFacets <= maxValuesPerFacet,\n getFacetValuesWarningMessage({\n maxDecimalPlaces,\n maxFacets,\n maxValuesPerFacet,\n })\n );\n\n const refinedStar = getRefinedStar(state);\n\n for (let star = STEP; star < max; star += STEP) {\n const isRefined = refinedStar === star;\n\n const count = facetResults\n .filter((f) => Number(f.name) >= star && Number(f.name) <= max)\n .map((f) => f.count)\n .reduce((sum, current) => sum + current, 0);\n\n if (refinedStar && !isRefined && count === 0) {\n // skip count==0 when at least 1 refinement is enabled\n // eslint-disable-next-line no-continue\n continue;\n }\n\n const stars = [...new Array(Math.floor(max / STEP))].map(\n (_v, i) => i * STEP < star\n );\n\n facetValues.push({\n stars,\n name: String(star),\n label: String(star),\n value: String(star),\n count,\n isRefined,\n });\n }\n }\n facetValues = facetValues.reverse();\n\n return {\n items: facetValues,\n hasNoResults: results ? results.nbHits === 0 : true,\n canRefine: facetValues.length > 0,\n refine: connectorState.toggleRefinementFactory(helper),\n sendEvent,\n createURL: connectorState.createURLFactory({ state, createURL }),\n widgetParams,\n };\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state.removeNumericRefinement(attribute);\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const value = getRefinedStar(searchParameters);\n\n if (typeof value !== 'number') {\n return uiState;\n }\n\n return {\n ...uiState,\n ratingMenu: {\n ...uiState.ratingMenu,\n [attribute]: value,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n const value = uiState.ratingMenu && uiState.ratingMenu[attribute];\n\n const withoutRefinements = searchParameters.clearRefinements(attribute);\n const withDisjunctiveFacet =\n withoutRefinements.addDisjunctiveFacet(attribute);\n\n if (!value) {\n return withDisjunctiveFacet.setQueryParameters({\n numericRefinements: {\n ...withDisjunctiveFacet.numericRefinements,\n [attribute]: {},\n },\n });\n }\n\n return withDisjunctiveFacet\n .addNumericRefinement(attribute, '<=', max)\n .addNumericRefinement(attribute, '>=', value);\n },\n };\n };\n};\n\nexport default connectRatingMenu;\n","import {\n checkRendering,\n createDocumentationMessageGenerator,\n noop,\n} from '../../lib/utils';\nimport type { Connector, WidgetRenderState } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'stats',\n connector: true,\n});\n\n/**\n * **Stats** connector provides the logic to build a custom widget that will displays\n * search statistics (hits number and processing time).\n */\n\nexport type StatsRenderState = {\n /**\n * The maximum number of hits per page returned by Algolia.\n */\n hitsPerPage?: number;\n /**\n * The number of hits in the result set.\n */\n nbHits: number;\n /**\n * The number of sorted hits in the result set (when using Relevant sort).\n */\n nbSortedHits?: number;\n /**\n * Indicates whether the index is currently using Relevant sort and is displaying only sorted hits.\n */\n areHitsSorted: boolean;\n /**\n * The number of pages computed for the result set.\n */\n nbPages: number;\n /**\n * The current page.\n */\n page: number;\n /**\n * The time taken to compute the results inside the Algolia engine.\n */\n processingTimeMS: number;\n /**\n * The query used for the current search.\n */\n query: string;\n};\n\nexport type StatsConnectorParams = Record;\n\nexport type StatsWidgetDescription = {\n $$type: 'ais.stats';\n renderState: StatsRenderState;\n indexRenderState: {\n stats: WidgetRenderState;\n };\n};\n\nexport type StatsConnector = Connector<\n StatsWidgetDescription,\n StatsConnectorParams\n>;\n\nconst connectStats: StatsConnector = function connectStats(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => ({\n $$type: 'ais.stats',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose() {\n unmountFn();\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n stats: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ results, state }) {\n if (!results) {\n return {\n hitsPerPage: state.hitsPerPage,\n nbHits: 0,\n nbSortedHits: undefined,\n areHitsSorted: false,\n nbPages: 0,\n page: state.page || 0,\n processingTimeMS: -1,\n query: state.query || '',\n widgetParams,\n };\n }\n\n return {\n hitsPerPage: results.hitsPerPage,\n nbHits: results.nbHits,\n nbSortedHits: results.nbSortedHits,\n areHitsSorted:\n typeof results.appliedRelevancyStrictness !== 'undefined' &&\n results.appliedRelevancyStrictness > 0 &&\n results.nbSortedHits !== results.nbHits,\n nbPages: results.nbPages,\n page: results.page,\n processingTimeMS: results.processingTimeMS,\n query: results.query,\n widgetParams,\n };\n },\n });\n};\n\nexport default connectStats;\n","import type {\n AlgoliaSearchHelper,\n SearchParameters,\n SearchResults,\n} from 'algoliasearch-helper';\nimport {\n checkRendering,\n escapeFacetValue,\n createDocumentationMessageGenerator,\n find,\n noop,\n toArray,\n} from '../../lib/utils';\nimport type {\n Connector,\n CreateURL,\n InstantSearch,\n WidgetRenderState,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'toggle-refinement',\n connector: true,\n});\n\nconst $$type = 'ais.toggleRefinement';\n\ntype BuiltInSendEventForToggle = (\n eventType: string,\n isRefined: boolean,\n eventName?: string\n) => void;\ntype CustomSendEventForToggle = (customPayload: any) => void;\n\nexport type SendEventForToggle = BuiltInSendEventForToggle &\n CustomSendEventForToggle;\n\nconst createSendEvent = ({\n instantSearchInstance,\n helper,\n attribute,\n on,\n}: {\n instantSearchInstance: InstantSearch;\n helper: AlgoliaSearchHelper;\n attribute: string;\n on: string[] | undefined;\n}) => {\n const sendEventForToggle: SendEventForToggle = (...args: any[]) => {\n if (args.length === 1) {\n instantSearchInstance.sendEventToInsights(args[0]);\n return;\n }\n const [eventType, isRefined, eventName = 'Filter Applied'] = args;\n if (eventType !== 'click' || on === undefined) {\n return;\n }\n\n // only send an event when the refinement gets applied,\n // not when it gets removed\n if (!isRefined) {\n instantSearchInstance.sendEventToInsights({\n insightsMethod: 'clickedFilters',\n widgetType: $$type,\n eventType,\n payload: {\n eventName,\n index: helper.getIndex(),\n filters: on.map((value) => `${attribute}:${value}`),\n },\n attribute,\n });\n }\n };\n return sendEventForToggle;\n};\n\nexport type ToggleRefinementValue = {\n /**\n * Whether this option is enabled.\n */\n isRefined: boolean;\n /**\n * Number of result if this option is toggled.\n */\n count: number | null;\n};\n\nexport type ToggleRefinementConnectorParams = {\n /**\n * Name of the attribute for faceting (e.g., \"free_shipping\").\n */\n attribute: string;\n /**\n * Value to filter on when toggled.\n * @default \"true\"\n */\n on?: string | string[] | boolean | boolean[] | number | number[];\n /**\n * Value to filter on when not toggled.\n */\n off?: string | string[] | boolean | boolean[] | number | number[];\n};\n\nexport type ToggleRefinementRenderState = {\n /** The current toggle value */\n value: {\n /**\n * The attribute name of this toggle.\n */\n name: string;\n /**\n * Whether the current option is \"on\" (true) or \"off\" (false)\n */\n isRefined: boolean;\n /**\n * Number of results if this option is toggled.\n */\n count: number | null;\n /**\n * Information about the \"on\" toggle.\n */\n onFacetValue: ToggleRefinementValue;\n /**\n * Information about the \"off\" toggle.\n */\n offFacetValue: ToggleRefinementValue;\n };\n /**\n * Creates an URL for the next state.\n */\n createURL: CreateURL;\n /**\n * Send a \"Facet Clicked\" Insights event.\n */\n sendEvent: SendEventForToggle;\n /**\n * Indicates if search state can be refined.\n */\n canRefine: boolean;\n /**\n * Updates to the next state by applying the toggle refinement.\n */\n refine: (value?: { isRefined: boolean }) => void;\n};\n\nexport type ToggleRefinementWidgetDescription = {\n $$type: 'ais.toggleRefinement';\n renderState: ToggleRefinementRenderState;\n indexRenderState: {\n toggleRefinement: {\n [attribute: string]: WidgetRenderState<\n ToggleRefinementRenderState,\n ToggleRefinementConnectorParams\n >;\n };\n };\n indexUiState: {\n toggle: {\n [attribute: string]: boolean;\n };\n };\n};\n\nexport type ToggleRefinementConnector = Connector<\n ToggleRefinementWidgetDescription,\n ToggleRefinementConnectorParams\n>;\n\n/**\n * **Toggle** connector provides the logic to build a custom widget that will provide\n * an on/off filtering feature based on an attribute value or values.\n *\n * Two modes are implemented in the custom widget:\n * - with or without the value filtered\n * - switch between two values.\n */\nconst connectToggleRefinement: ToggleRefinementConnector =\n function connectToggleRefinement(renderFn, unmountFn = noop) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const { attribute, on: userOn = true, off: userOff } = widgetParams || {};\n\n if (!attribute) {\n throw new Error(withUsage('The `attribute` option is required.'));\n }\n\n const hasAnOffValue = userOff !== undefined;\n const on = toArray(userOn).map(escapeFacetValue);\n const off = hasAnOffValue\n ? toArray(userOff).map(escapeFacetValue)\n : undefined;\n\n let sendEvent: SendEventForToggle;\n\n const toggleRefinementFactory =\n (helper: AlgoliaSearchHelper) =>\n (\n {\n isRefined,\n }: {\n isRefined: boolean;\n } = { isRefined: false }\n ) => {\n if (!isRefined) {\n sendEvent('click', isRefined);\n if (hasAnOffValue) {\n off!.forEach((v) =>\n helper.removeDisjunctiveFacetRefinement(attribute, v)\n );\n }\n\n on.forEach((v) =>\n helper.addDisjunctiveFacetRefinement(attribute, v)\n );\n } else {\n on.forEach((v) =>\n helper.removeDisjunctiveFacetRefinement(attribute, v)\n );\n\n if (hasAnOffValue) {\n off!.forEach((v) =>\n helper.addDisjunctiveFacetRefinement(attribute, v)\n );\n }\n }\n\n helper.search();\n };\n\n const connectorState = {\n createURLFactory:\n (\n isRefined: boolean,\n {\n state,\n createURL,\n }: {\n state: SearchParameters;\n createURL(parameters: SearchParameters): string;\n }\n ) =>\n () => {\n state = state.resetPage();\n\n const valuesToRemove = isRefined ? on : off;\n if (valuesToRemove) {\n valuesToRemove.forEach((v) => {\n state = state.removeDisjunctiveFacetRefinement(attribute, v);\n });\n }\n\n const valuesToAdd = isRefined ? off : on;\n if (valuesToAdd) {\n valuesToAdd.forEach((v) => {\n state = state.addDisjunctiveFacetRefinement(attribute, v);\n });\n }\n\n return createURL(state);\n },\n };\n\n return {\n $$type,\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state.removeDisjunctiveFacet(attribute);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n toggleRefinement: {\n ...renderState.toggleRefinement,\n [attribute]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState({\n state,\n helper,\n results,\n createURL,\n instantSearchInstance,\n }) {\n const isRefined = results\n ? on.every((v) => state.isDisjunctiveFacetRefined(attribute, v))\n : on.every((v) => state.isDisjunctiveFacetRefined(attribute, v));\n\n let onFacetValue: ToggleRefinementValue = {\n isRefined,\n count: 0,\n };\n\n let offFacetValue: ToggleRefinementValue = {\n isRefined: hasAnOffValue && !isRefined,\n count: 0,\n };\n\n if (results) {\n const offValue = toArray(off || false);\n const allFacetValues = (results.getFacetValues(attribute, {}) ||\n []) as SearchResults.FacetValue[];\n\n const onData = on\n .map((v) =>\n find(\n allFacetValues,\n ({ escapedValue }) =>\n escapedValue === escapeFacetValue(String(v))\n )\n )\n .filter((v): v is SearchResults.FacetValue => v !== undefined);\n\n const offData = hasAnOffValue\n ? offValue\n .map((v) =>\n find(\n allFacetValues,\n ({ escapedValue }) =>\n escapedValue === escapeFacetValue(String(v))\n )\n )\n .filter((v): v is SearchResults.FacetValue => v !== undefined)\n : [];\n\n onFacetValue = {\n isRefined: onData.length\n ? onData.every((v) => v.isRefined)\n : false,\n count: onData.reduce((acc, v) => acc + v.count, 0) || null,\n };\n\n offFacetValue = {\n isRefined: offData.length\n ? offData.every((v) => v.isRefined)\n : false,\n count:\n offData.reduce((acc, v) => acc + v.count, 0) ||\n allFacetValues.reduce((total, { count }) => total + count, 0),\n };\n }\n\n if (!sendEvent) {\n sendEvent = createSendEvent({\n instantSearchInstance,\n attribute,\n on,\n helper,\n });\n }\n const nextRefinement = isRefined ? offFacetValue : onFacetValue;\n\n return {\n value: {\n name: attribute,\n isRefined,\n count: results ? nextRefinement.count : null,\n onFacetValue,\n offFacetValue,\n },\n createURL: connectorState.createURLFactory(isRefined, {\n state,\n createURL,\n }),\n sendEvent,\n canRefine: Boolean(results ? nextRefinement.count : null),\n refine: toggleRefinementFactory(helper),\n widgetParams,\n };\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const isRefined =\n on &&\n on.every((v) =>\n searchParameters.isDisjunctiveFacetRefined(attribute, v)\n );\n\n if (!isRefined) {\n return uiState;\n }\n\n return {\n ...uiState,\n toggle: {\n ...uiState.toggle,\n [attribute]: isRefined,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n let withFacetConfiguration = searchParameters\n .clearRefinements(attribute)\n .addDisjunctiveFacet(attribute);\n\n const isRefined = Boolean(\n uiState.toggle && uiState.toggle[attribute]\n );\n\n if (isRefined) {\n if (on) {\n on.forEach((v) => {\n withFacetConfiguration =\n withFacetConfiguration.addDisjunctiveFacetRefinement(\n attribute,\n v\n );\n });\n }\n\n return withFacetConfiguration;\n }\n\n // It's not refined with an `off` value\n if (hasAnOffValue) {\n if (off) {\n off.forEach((v) => {\n withFacetConfiguration =\n withFacetConfiguration.addDisjunctiveFacetRefinement(\n attribute,\n v\n );\n });\n }\n return withFacetConfiguration;\n }\n\n // It's not refined without an `off` value\n return withFacetConfiguration.setQueryParameters({\n disjunctiveFacetsRefinements: {\n ...searchParameters.disjunctiveFacetsRefinements,\n [attribute]: [],\n },\n });\n },\n };\n };\n };\n\nexport default connectToggleRefinement;\n","import {\n checkRendering,\n warning,\n createDocumentationMessageGenerator,\n isEqual,\n noop,\n} from '../../lib/utils';\nimport type { SearchParameters, SearchResults } from 'algoliasearch-helper';\nimport type {\n Connector,\n TransformItems,\n CreateURL,\n WidgetRenderState,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'breadcrumb',\n connector: true,\n});\n\nexport type BreadcrumbConnectorParamsItem = {\n /**\n * Label of the category or subcategory.\n */\n label: string;\n\n /**\n * Value of breadcrumb item.\n */\n value: string | null;\n};\n\nexport type BreadcrumbConnectorParams = {\n /**\n * Attributes to use to generate the hierarchy of the breadcrumb.\n */\n attributes: string[];\n\n /**\n * Prefix path to use if the first level is not the root level.\n */\n rootPath?: string;\n\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n\n /**\n * The level separator used in the records.\n *\n * @default '>'\n */\n separator?: string;\n};\n\nexport type BreadcrumbRenderState = {\n /**\n * Creates the URL for a single item name in the list.\n */\n createURL: CreateURL;\n\n /**\n * Array of objects defining the different values and labels.\n */\n items: BreadcrumbConnectorParamsItem[];\n\n /**\n * Sets the path of the hierarchical filter and triggers a new search.\n */\n refine: (value: BreadcrumbConnectorParamsItem['value']) => void;\n\n /**\n * True if refinement can be applied.\n */\n canRefine: boolean;\n};\n\nexport type BreadcrumbWidgetDescription = {\n $$type: 'ais.breadcrumb';\n renderState: BreadcrumbRenderState;\n indexRenderState: {\n breadcrumb: {\n [rootAttribute: string]: WidgetRenderState<\n BreadcrumbRenderState,\n BreadcrumbConnectorParams\n >;\n };\n };\n};\n\nexport type BreadcrumbConnector = Connector<\n BreadcrumbWidgetDescription,\n BreadcrumbConnectorParams\n>;\n\nconst connectBreadcrumb: BreadcrumbConnector = function connectBreadcrumb(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n type ConnectorState = {\n refine?: BreadcrumbRenderState['refine'];\n createURL?: BreadcrumbRenderState['createURL'];\n };\n\n const connectorState: ConnectorState = {};\n\n return (widgetParams) => {\n const {\n attributes,\n separator = ' > ',\n rootPath = null,\n transformItems = ((items) => items) as NonNullable<\n BreadcrumbConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n if (!attributes || !Array.isArray(attributes) || attributes.length === 0) {\n throw new Error(\n withUsage('The `attributes` option expects an array of strings.')\n );\n }\n\n const [hierarchicalFacetName] = attributes;\n\n function getRefinedState(\n state: SearchParameters,\n facetValue: string | null\n ) {\n if (!facetValue) {\n const breadcrumb = state.getHierarchicalFacetBreadcrumb(\n hierarchicalFacetName\n );\n if (breadcrumb.length === 0) {\n return state;\n } else {\n return state\n .resetPage()\n .toggleFacetRefinement(hierarchicalFacetName, breadcrumb[0]);\n }\n }\n return state\n .resetPage()\n .toggleFacetRefinement(hierarchicalFacetName, facetValue);\n }\n\n return {\n $$type: 'ais.breadcrumb',\n\n init(initOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance: initOptions.instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n },\n\n dispose() {\n unmountFn();\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n breadcrumb: {\n ...renderState.breadcrumb,\n [hierarchicalFacetName]: this.getWidgetRenderState(renderOptions),\n },\n };\n },\n\n getWidgetRenderState({ helper, createURL, results, state }) {\n function getItems() {\n // The hierarchicalFacets condition is required for flavors\n // that render immediately with empty results, without relying\n // on init() (like React InstantSearch Hooks).\n if (!results || state.hierarchicalFacets.length === 0) {\n return [];\n }\n\n const [{ name: facetName }] = state.hierarchicalFacets;\n\n const facetValues = results.getFacetValues(\n facetName,\n {}\n ) as SearchResults.HierarchicalFacet;\n const data = Array.isArray(facetValues.data) ? facetValues.data : [];\n const items = transformItems(shiftItemsValues(prepareItems(data)), {\n results,\n });\n\n return items;\n }\n\n const items = getItems();\n\n if (!connectorState.createURL) {\n connectorState.createURL = (facetValue) => {\n return createURL(getRefinedState(helper.state, facetValue));\n };\n }\n\n if (!connectorState.refine) {\n connectorState.refine = (facetValue) => {\n helper.setState(getRefinedState(helper.state, facetValue)).search();\n };\n }\n\n return {\n canRefine: items.length > 0,\n createURL: connectorState.createURL,\n items,\n refine: connectorState.refine,\n widgetParams,\n };\n },\n\n getWidgetSearchParameters(searchParameters) {\n if (searchParameters.isHierarchicalFacet(hierarchicalFacetName)) {\n const facet = searchParameters.getHierarchicalFacetByName(\n hierarchicalFacetName\n );\n\n warning(\n isEqual(facet.attributes, attributes) &&\n facet.separator === separator &&\n facet.rootPath === rootPath,\n 'Using Breadcrumb and HierarchicalMenu on the same facet with different options overrides the configuration of the HierarchicalMenu.'\n );\n\n return searchParameters;\n }\n\n return searchParameters.addHierarchicalFacet({\n name: hierarchicalFacetName,\n attributes,\n separator,\n rootPath,\n });\n },\n };\n };\n};\n\nfunction prepareItems(data: SearchResults.HierarchicalFacet[]) {\n return data.reduce((result, currentItem) => {\n if (currentItem.isRefined) {\n result.push({\n label: currentItem.name,\n value: currentItem.escapedValue,\n });\n if (Array.isArray(currentItem.data)) {\n result = result.concat(prepareItems(currentItem.data));\n }\n }\n return result;\n }, []);\n}\n\nfunction shiftItemsValues(array: BreadcrumbConnectorParamsItem[]) {\n return array.map((x, idx) => ({\n label: x.label,\n value: idx + 1 === array.length ? null : array[idx + 1].value,\n }));\n}\n\nexport default connectBreadcrumb;\n","import type {\n AlgoliaSearchHelper,\n SearchParameters,\n} from 'algoliasearch-helper';\nimport type { SendEventForHits } from '../../lib/utils';\nimport {\n checkRendering,\n aroundLatLngToPosition,\n insideBoundingBoxToBoundingBox,\n createDocumentationMessageGenerator,\n createSendEventForHits,\n noop,\n} from '../../lib/utils';\nimport type {\n Connector,\n GeoLoc,\n Hit,\n InitOptions,\n RenderOptions,\n TransformItems,\n WidgetRenderState,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'geo-search',\n connector: true,\n});\n\n// in this connector, we assume insideBoundingBox is only a string,\n// even though in the helper it's defined as number[][] alone.\n// This can be done, since the connector assumes \"control\" of the parameter\nfunction getBoundingBoxAsString(state: SearchParameters) {\n return (state.insideBoundingBox as unknown as string) || '';\n}\nfunction setBoundingBoxAsString(state: SearchParameters, value: string) {\n return state.setQueryParameter(\n 'insideBoundingBox',\n value as unknown as number[][]\n );\n}\n\nexport type GeoHit = Hit & Required>;\n\ntype Bounds = {\n /**\n * The top right corner of the map view.\n */\n northEast: GeoLoc;\n /**\n * The bottom left corner of the map view.\n */\n southWest: GeoLoc;\n};\n\nexport type GeoSearchRenderState = {\n /**\n * Reset the current bounding box refinement.\n */\n clearMapRefinement(): void;\n /**\n * The current bounding box of the search.\n */\n currentRefinement?: Bounds;\n /**\n * Return true if the map has move since the last refinement.\n */\n hasMapMoveSinceLastRefine(): boolean;\n /**\n * Return true if the current refinement is set with the map bounds.\n */\n isRefinedWithMap(): boolean;\n /**\n * Return true if the user is able to refine on map move.\n */\n isRefineOnMapMove(): boolean;\n /**\n * The matched hits from Algolia API.\n */\n items: GeoHit[];\n /**\n * The current position of the search.\n */\n position?: GeoLoc;\n /**\n * Sets a bounding box to filter the results from the given map bounds.\n */\n refine(bounds: Bounds): void;\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEventForHits;\n /**\n * Set the fact that the map has moved since the last refinement, should be\n * called on each map move. The call to the function triggers a new rendering\n * only when the value change.\n */\n setMapMoveSinceLastRefine(): void;\n /**\n * Toggle the fact that the user is able to refine on map move.\n */\n toggleRefineOnMapMove(): void;\n};\n\nexport type GeoSearchConnectorParams = {\n /**\n * If true, refine will be triggered as you move the map.\n * @default true\n */\n enableRefineOnMapMove?: boolean;\n /**\n * Function to transform the items passed to the templates.\n * @default items => items\n */\n transformItems?: TransformItems;\n};\n\nconst $$type = 'ais.geoSearch';\n\nexport type GeoSearchWidgetDescription = {\n $$type: 'ais.geoSearch';\n renderState: GeoSearchRenderState;\n indexRenderState: {\n geoSearch: WidgetRenderState<\n GeoSearchRenderState,\n GeoSearchConnectorParams\n >;\n };\n indexUiState: {\n geoSearch: {\n /**\n * The rectangular area in geo coordinates.\n * The rectangle is defined by two diagonally opposite points,\n * hence by 4 floats separated by commas.\n *\n * @example '47.3165,4.9665,47.3424,5.0201'\n */\n boundingBox: string;\n };\n };\n};\n\nexport type GeoSearchConnector = Connector<\n GeoSearchWidgetDescription,\n GeoSearchConnectorParams\n>;\n\n/**\n * The **GeoSearch** connector provides the logic to build a widget that will display the results on a map. It also provides a way to search for results based on their position. The connector provides functions to manage the search experience (search on map interaction or control the interaction for example).\n *\n * @requirements\n *\n * Note that the GeoSearch connector uses the [geosearch](https://www.algolia.com/doc/guides/searching/geo-search) capabilities of Algolia. Your hits **must** have a `_geoloc` attribute in order to be passed to the rendering function.\n *\n * Currently, the feature is not compatible with multiple values in the _geoloc attribute.\n */\nconst connectGeoSearch: GeoSearchConnector = (renderFn, unmountFn = noop) => {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n enableRefineOnMapMove = true,\n transformItems = ((items) => items) as NonNullable<\n GeoSearchConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n const widgetState = {\n isRefineOnMapMove: enableRefineOnMapMove,\n // @MAJOR hasMapMoveSinceLastRefine -> hasMapMovedSinceLastRefine\n hasMapMoveSinceLastRefine: false,\n lastRefinePosition: '',\n lastRefineBoundingBox: '',\n internalToggleRefineOnMapMove: noop,\n internalSetMapMoveSinceLastRefine: noop,\n };\n\n const getPositionFromState = (state: SearchParameters) =>\n state.aroundLatLng\n ? aroundLatLngToPosition(state.aroundLatLng)\n : undefined;\n\n const getCurrentRefinementFromState = (state: SearchParameters) =>\n state.insideBoundingBox &&\n insideBoundingBoxToBoundingBox(state.insideBoundingBox);\n\n const refine =\n (helper: AlgoliaSearchHelper) =>\n ({ northEast: ne, southWest: sw }: Bounds) => {\n const boundingBox = [ne.lat, ne.lng, sw.lat, sw.lng].join();\n\n helper\n .setState(\n setBoundingBoxAsString(helper.state, boundingBox).resetPage()\n )\n .search();\n\n widgetState.hasMapMoveSinceLastRefine = false;\n widgetState.lastRefineBoundingBox = boundingBox;\n };\n\n const clearMapRefinement = (helper: AlgoliaSearchHelper) => () => {\n helper.setQueryParameter('insideBoundingBox', undefined).search();\n };\n\n const isRefinedWithMap = (state: SearchParameters) => () =>\n Boolean(state.insideBoundingBox);\n\n const toggleRefineOnMapMove = () =>\n widgetState.internalToggleRefineOnMapMove();\n const createInternalToggleRefinementOnMapMove =\n (\n renderOptions: TRenderOptions,\n // false positive eslint because of generics\n // eslint-disable-next-line no-shadow\n render: (renderOptions: TRenderOptions) => void\n ) =>\n () => {\n widgetState.isRefineOnMapMove = !widgetState.isRefineOnMapMove;\n\n render(renderOptions);\n };\n\n const isRefineOnMapMove = () => widgetState.isRefineOnMapMove;\n\n const setMapMoveSinceLastRefine = () =>\n widgetState.internalSetMapMoveSinceLastRefine();\n const createInternalSetMapMoveSinceLastRefine =\n (\n renderOptions: TRenderOptions,\n // false positive eslint because of generics\n // eslint-disable-next-line no-shadow\n render: (renderOptions: TRenderOptions) => void\n ) =>\n () => {\n const shouldTriggerRender =\n widgetState.hasMapMoveSinceLastRefine !== true;\n\n widgetState.hasMapMoveSinceLastRefine = true;\n\n if (shouldTriggerRender) {\n render(renderOptions);\n }\n };\n\n const hasMapMoveSinceLastRefine = () =>\n widgetState.hasMapMoveSinceLastRefine;\n\n let sendEvent: SendEventForHits;\n\n return {\n $$type,\n\n init(initArgs) {\n const { instantSearchInstance } = initArgs;\n const isFirstRendering = true;\n\n widgetState.internalToggleRefineOnMapMove =\n createInternalToggleRefinementOnMapMove(initArgs, noop);\n\n widgetState.internalSetMapMoveSinceLastRefine =\n createInternalSetMapMoveSinceLastRefine(initArgs, noop);\n\n renderFn(\n {\n ...this.getWidgetRenderState(initArgs),\n instantSearchInstance,\n },\n isFirstRendering\n );\n },\n\n render(renderArgs) {\n const { helper, instantSearchInstance } = renderArgs;\n const isFirstRendering = false;\n // We don't use the state provided by the render function because we need\n // to be sure that the state is the latest one for the following condition\n const state = helper.state;\n\n const positionChangedSinceLastRefine =\n Boolean(state.aroundLatLng) &&\n Boolean(widgetState.lastRefinePosition) &&\n state.aroundLatLng !== widgetState.lastRefinePosition;\n\n const boundingBoxChangedSinceLastRefine =\n !state.insideBoundingBox &&\n Boolean(widgetState.lastRefineBoundingBox) &&\n state.insideBoundingBox !== widgetState.lastRefineBoundingBox;\n\n if (\n positionChangedSinceLastRefine ||\n boundingBoxChangedSinceLastRefine\n ) {\n widgetState.hasMapMoveSinceLastRefine = false;\n }\n\n widgetState.lastRefinePosition = state.aroundLatLng || '';\n\n widgetState.lastRefineBoundingBox = getBoundingBoxAsString(state);\n\n widgetState.internalToggleRefineOnMapMove =\n createInternalToggleRefinementOnMapMove(\n renderArgs,\n this.render!.bind(this)\n );\n\n widgetState.internalSetMapMoveSinceLastRefine =\n createInternalSetMapMoveSinceLastRefine(\n renderArgs,\n this.render!.bind(this)\n );\n\n const widgetRenderState = this.getWidgetRenderState(renderArgs);\n\n sendEvent('view', widgetRenderState.items);\n\n renderFn(\n {\n ...widgetRenderState,\n instantSearchInstance,\n },\n isFirstRendering\n );\n },\n\n getWidgetRenderState(renderOptions) {\n const { helper, results, instantSearchInstance } = renderOptions;\n const state = helper.state;\n\n const items = results\n ? transformItems(\n results.hits.filter((hit) => hit._geoloc),\n { results }\n )\n : [];\n\n if (!sendEvent) {\n sendEvent = createSendEventForHits({\n instantSearchInstance,\n index: helper.getIndex(),\n widgetType: $$type,\n });\n }\n\n return {\n items,\n position: getPositionFromState(state),\n currentRefinement: getCurrentRefinementFromState(state),\n refine: refine(helper),\n sendEvent,\n clearMapRefinement: clearMapRefinement(helper),\n isRefinedWithMap: isRefinedWithMap(state),\n toggleRefineOnMapMove,\n isRefineOnMapMove,\n setMapMoveSinceLastRefine,\n hasMapMoveSinceLastRefine,\n widgetParams,\n };\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n geoSearch: this.getWidgetRenderState(renderOptions),\n };\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state.setQueryParameter('insideBoundingBox', undefined);\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const boundingBox = getBoundingBoxAsString(searchParameters);\n\n if (\n !boundingBox ||\n (uiState &&\n uiState.geoSearch &&\n uiState.geoSearch.boundingBox === boundingBox)\n ) {\n return uiState;\n }\n\n return {\n ...uiState,\n geoSearch: {\n boundingBox,\n },\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n if (!uiState || !uiState.geoSearch) {\n return searchParameters.setQueryParameter(\n 'insideBoundingBox',\n undefined\n );\n }\n return setBoundingBoxAsString(\n searchParameters,\n uiState.geoSearch.boundingBox\n );\n },\n };\n };\n};\n\nexport default connectGeoSearch;\n","import {\n safelyRunOnBrowser,\n checkRendering,\n createDocumentationMessageGenerator,\n noop,\n} from '../../lib/utils';\n\nimport type { Connector, WidgetRenderState } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'powered-by',\n connector: true,\n});\n\nexport type PoweredByRenderState = {\n /** the url to redirect to on click */\n url: string;\n};\n\nexport type PoweredByConnectorParams = {\n /** the url to redirect to on click */\n url?: string;\n};\n\nexport type PoweredByWidgetDescription = {\n $$type: 'ais.poweredBy';\n renderState: PoweredByRenderState;\n indexRenderState: {\n poweredBy: WidgetRenderState<\n PoweredByRenderState,\n PoweredByConnectorParams\n >;\n };\n};\n\nexport type PoweredByConnector = Connector<\n PoweredByWidgetDescription,\n PoweredByConnectorParams\n>;\n\n/**\n * **PoweredBy** connector provides the logic to build a custom widget that will displays\n * the logo to redirect to Algolia.\n */\nconst connectPoweredBy: PoweredByConnector = function connectPoweredBy(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n const defaultUrl =\n 'https://www.algolia.com/?' +\n 'utm_source=instantsearch.js&' +\n 'utm_medium=website&' +\n `utm_content=${safelyRunOnBrowser(\n ({ window }) => window.location?.hostname || '',\n { fallback: () => '' }\n )}&` +\n 'utm_campaign=poweredby';\n\n return (widgetParams) => {\n const { url = defaultUrl } = widgetParams || {};\n\n return {\n $$type: 'ais.poweredBy',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n poweredBy: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState() {\n return {\n url,\n widgetParams,\n };\n },\n\n dispose() {\n unmountFn();\n },\n };\n };\n};\n\nexport default connectPoweredBy;\n","import type {\n SearchParameters,\n PlainSearchParameters,\n AlgoliaSearchHelper,\n} from 'algoliasearch-helper';\nimport algoliasearchHelper from 'algoliasearch-helper';\nimport type { Connector, WidgetRenderState } from '../../types';\nimport {\n createDocumentationMessageGenerator,\n isPlainObject,\n mergeSearchParameters,\n noop,\n} from '../../lib/utils';\n\n/**\n * Refine the given search parameters.\n */\ntype Refine = (searchParameters: PlainSearchParameters) => void;\n\nexport type ConfigureConnectorParams = {\n /**\n * A list of [search parameters](https://www.algolia.com/doc/api-reference/search-api-parameters/)\n * to enable when the widget mounts.\n */\n searchParameters: PlainSearchParameters;\n};\n\nexport type ConfigureRenderState = {\n /**\n * Refine the given search parameters.\n */\n refine: Refine;\n};\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'configure',\n connector: true,\n});\n\nfunction getInitialSearchParameters(\n state: SearchParameters,\n widgetParams: ConfigureConnectorParams\n): SearchParameters {\n // We leverage the helper internals to remove the `widgetParams` from\n // the state. The function `setQueryParameters` omits the values that\n // are `undefined` on the next state.\n return state.setQueryParameters(\n Object.keys(widgetParams.searchParameters).reduce(\n (acc, key) => ({\n ...acc,\n [key]: undefined,\n }),\n {}\n )\n );\n}\n\nexport type ConfigureWidgetDescription = {\n $$type: 'ais.configure';\n renderState: ConfigureRenderState;\n indexRenderState: {\n configure: WidgetRenderState<\n ConfigureRenderState,\n ConfigureConnectorParams\n >;\n };\n indexUiState: {\n configure: PlainSearchParameters;\n };\n};\n\nexport type ConfigureConnector = Connector<\n ConfigureWidgetDescription,\n ConfigureConnectorParams\n>;\n\nconst connectConfigure: ConfigureConnector = function connectConfigure(\n renderFn = noop,\n unmountFn = noop\n) {\n return (widgetParams) => {\n if (!widgetParams || !isPlainObject(widgetParams.searchParameters)) {\n throw new Error(\n withUsage('The `searchParameters` option expects an object.')\n );\n }\n\n type ConnectorState = {\n refine?: Refine;\n };\n\n const connectorState: ConnectorState = {};\n\n function refine(helper: AlgoliaSearchHelper): Refine {\n return (searchParameters: PlainSearchParameters) => {\n // Merge new `searchParameters` with the ones set from other widgets\n const actualState = getInitialSearchParameters(\n helper.state,\n widgetParams\n );\n const nextSearchParameters = mergeSearchParameters(\n actualState,\n new algoliasearchHelper.SearchParameters(searchParameters)\n );\n\n // Update original `widgetParams.searchParameters` to the new refined one\n widgetParams.searchParameters = searchParameters;\n\n // Trigger a search with the resolved search parameters\n helper.setState(nextSearchParameters).search();\n };\n }\n\n return {\n $$type: 'ais.configure',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return getInitialSearchParameters(state, widgetParams);\n },\n\n getRenderState(renderState, renderOptions) {\n const widgetRenderState = this.getWidgetRenderState(renderOptions);\n return {\n ...renderState,\n configure: {\n ...widgetRenderState,\n widgetParams: {\n ...widgetRenderState.widgetParams,\n searchParameters: mergeSearchParameters(\n new algoliasearchHelper.SearchParameters(\n renderState.configure?.widgetParams.searchParameters\n ),\n new algoliasearchHelper.SearchParameters(\n widgetRenderState.widgetParams.searchParameters\n )\n ).getQueryParams(),\n },\n },\n };\n },\n\n getWidgetRenderState({ helper }) {\n if (!connectorState.refine) {\n connectorState.refine = refine(helper);\n }\n\n return {\n refine: connectorState.refine,\n widgetParams,\n };\n },\n\n getWidgetSearchParameters(state, { uiState }) {\n return mergeSearchParameters(\n state,\n new algoliasearchHelper.SearchParameters({\n ...uiState.configure,\n ...widgetParams.searchParameters,\n })\n );\n },\n\n getWidgetUiState(uiState) {\n return {\n ...uiState,\n configure: {\n ...uiState.configure,\n ...widgetParams.searchParameters,\n },\n };\n },\n };\n };\n};\n\nexport default connectConfigure;\n","import type {\n SearchParameters,\n PlainSearchParameters,\n} from 'algoliasearch-helper';\nimport algoliasearchHelper from 'algoliasearch-helper';\nimport type { AlgoliaHit, Connector } from '../../types';\nimport {\n createDocumentationMessageGenerator,\n getObjectType,\n warning,\n getPropertyByPath,\n} from '../../lib/utils';\nimport type { ConfigureWidgetDescription } from '../configure/connectConfigure';\nimport connectConfigure from '../configure/connectConfigure';\n\nexport type MatchingPatterns = {\n [attribute: string]: {\n /**\n * The score of the optional filter.\n *\n * @see https://www.algolia.com/doc/guides/managing-results/rules/merchandising-and-promoting/in-depth/optional-filters/\n */\n score: number;\n };\n};\n\nexport type TransformSearchParameters = (\n searchParameters: SearchParameters\n) => PlainSearchParameters;\n\nexport type ConfigureRelatedItemsConnectorParams = {\n /**\n * The reference hit to extract the filters from.\n */\n hit: AlgoliaHit;\n /**\n * The schema to create the optional filters.\n * Each key represents an attribute from the hit.\n */\n matchingPatterns: MatchingPatterns;\n /**\n * Function to transform the generated search parameters.\n */\n transformSearchParameters?: TransformSearchParameters;\n};\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'configure-related-items',\n connector: true,\n});\n\nfunction createOptionalFilter({\n attributeName,\n attributeValue,\n attributeScore,\n}: {\n attributeName: string;\n attributeValue: string;\n attributeScore: number;\n}) {\n return `${attributeName}:${attributeValue}`;\n}\n\nexport type ConfigureRelatedItemsWidgetDescription = {\n $$type: 'ais.configureRelatedItems';\n} & Omit;\n\nexport type ConfigureRelatedItemsConnector = Connector<\n ConfigureRelatedItemsWidgetDescription,\n ConfigureRelatedItemsConnectorParams\n>;\n\nconst connectConfigureRelatedItems: ConfigureRelatedItemsConnector =\n function connectConfigureRelatedItems(renderFn, unmountFn) {\n return (widgetParams) => {\n const {\n hit,\n matchingPatterns,\n transformSearchParameters = ((x) => x) as TransformSearchParameters,\n } = widgetParams || {};\n\n if (!hit) {\n throw new Error(withUsage('The `hit` option is required.'));\n }\n\n if (!matchingPatterns) {\n throw new Error(\n withUsage('The `matchingPatterns` option is required.')\n );\n }\n\n const optionalFilters = Object.keys(matchingPatterns).reduce<\n Array\n >((acc, attributeName) => {\n const attribute = matchingPatterns[attributeName];\n const attributeValue = getPropertyByPath(hit, attributeName);\n const attributeScore = attribute.score;\n\n if (Array.isArray(attributeValue)) {\n return [\n ...acc,\n attributeValue.map((attributeSubValue) => {\n return createOptionalFilter({\n attributeName,\n attributeValue: attributeSubValue,\n attributeScore,\n });\n }),\n ];\n }\n\n if (typeof attributeValue === 'string') {\n return [\n ...acc,\n createOptionalFilter({\n attributeName,\n attributeValue,\n attributeScore,\n }),\n ];\n }\n\n warning(\n false,\n `\nThe \\`matchingPatterns\\` option returned a value of type ${getObjectType(\n attributeValue\n )} for the \"${attributeName}\" key. This value was not sent to Algolia because \\`optionalFilters\\` only supports strings and array of strings.\n\nYou can remove the \"${attributeName}\" key from the \\`matchingPatterns\\` option.\n\nSee https://www.algolia.com/doc/api-reference/api-parameters/optionalFilters/\n `\n );\n\n return acc;\n }, []);\n\n const searchParameters: PlainSearchParameters = {\n ...transformSearchParameters(\n new algoliasearchHelper.SearchParameters({\n sumOrFiltersScores: true,\n facetFilters: [`objectID:-${hit.objectID}`],\n optionalFilters,\n })\n ),\n };\n\n const makeWidget = connectConfigure(renderFn, unmountFn);\n\n return {\n // required, since widget parameters differ between these connectors\n // and we don't want to have the parameters of configure here\n ...makeWidget({ searchParameters } as any),\n $$type: 'ais.configureRelatedItems',\n };\n };\n };\n\nexport default connectConfigureRelatedItems;\n","import type { SearchResults } from 'algoliasearch-helper';\nimport type { SendEventForHits } from '../../lib/utils';\nimport {\n escapeHits,\n TAG_PLACEHOLDER,\n checkRendering,\n createDocumentationMessageGenerator,\n createSendEventForHits,\n noop,\n warning,\n} from '../../lib/utils';\nimport type { Hits, Connector, WidgetRenderState } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'autocomplete',\n connector: true,\n});\n\nexport type AutocompleteConnectorParams = {\n /**\n * Escapes HTML entities from hits string values.\n *\n * @default `true`\n */\n escapeHTML?: boolean;\n};\n\nexport type AutocompleteRenderState = {\n /**\n * The current value of the query.\n */\n currentRefinement: string;\n\n /**\n * The indices this widget has access to.\n */\n indices: Array<{\n /**\n * The name of the index\n */\n indexName: string;\n\n /**\n * The resolved hits from the index matching the query.\n */\n hits: Hits;\n\n /**\n * The full results object from the Algolia API.\n */\n results: SearchResults;\n\n /**\n * Send event to insights middleware\n */\n sendEvent: SendEventForHits;\n }>;\n\n /**\n * Searches into the indices with the provided query.\n */\n refine: (query: string) => void;\n};\n\nexport type AutocompleteWidgetDescription = {\n $$type: 'ais.autocomplete';\n renderState: AutocompleteRenderState;\n indexRenderState: {\n autocomplete: WidgetRenderState<\n AutocompleteRenderState,\n AutocompleteConnectorParams\n >;\n };\n indexUiState: { query: string };\n};\n\nexport type AutocompleteConnector = Connector<\n AutocompleteWidgetDescription,\n AutocompleteConnectorParams\n>;\n\nconst connectAutocomplete: AutocompleteConnector = function connectAutocomplete(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const { escapeHTML = true } = widgetParams || {};\n\n warning(\n !(widgetParams as any).indices,\n `\nThe option \\`indices\\` has been removed from the Autocomplete connector.\n\nThe indices to target are now inferred from the widgets tree.\n${\n Array.isArray((widgetParams as any).indices)\n ? `\nAn alternative would be:\n\nconst autocomplete = connectAutocomplete(renderer);\n\nsearch.addWidgets([\n ${(widgetParams as any).indices\n .map(({ value }: { value: string }) => `index({ indexName: '${value}' }),`)\n .join('\\n ')}\n autocomplete()\n]);\n`\n : ''\n}\n `\n );\n\n type ConnectorState = {\n refine?(query: string): void;\n };\n\n const connectorState: ConnectorState = {};\n\n return {\n $$type: 'ais.autocomplete',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n const renderState = this.getWidgetRenderState(renderOptions);\n\n renderState.indices.forEach(({ sendEvent, hits }) => {\n sendEvent('view', hits);\n });\n\n renderFn(\n {\n ...renderState,\n instantSearchInstance,\n },\n false\n );\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n autocomplete: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({\n helper,\n state,\n scopedResults,\n instantSearchInstance,\n }) {\n if (!connectorState.refine) {\n connectorState.refine = (query: string) => {\n helper.setQuery(query).search();\n };\n }\n\n const indices = scopedResults.map((scopedResult) => {\n // We need to escape the hits because highlighting\n // exposes HTML tags to the end-user.\n scopedResult.results.hits = escapeHTML\n ? escapeHits(scopedResult.results.hits)\n : scopedResult.results.hits;\n\n const sendEvent = createSendEventForHits({\n instantSearchInstance,\n index: scopedResult.results.index,\n widgetType: this.$$type,\n });\n\n return {\n indexId: scopedResult.indexId,\n indexName: scopedResult.results.index,\n hits: scopedResult.results.hits,\n results: scopedResult.results,\n sendEvent,\n };\n });\n\n return {\n currentRefinement: state.query || '',\n indices,\n refine: connectorState.refine,\n widgetParams,\n };\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const query = searchParameters.query || '';\n\n if (query === '' || (uiState && uiState.query === query)) {\n return uiState;\n }\n\n return {\n ...uiState,\n query,\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n const parameters = {\n query: uiState.query || '',\n };\n\n if (!escapeHTML) {\n return searchParameters.setQueryParameters(parameters);\n }\n\n return searchParameters.setQueryParameters({\n ...parameters,\n ...TAG_PLACEHOLDER,\n });\n },\n\n dispose({ state }) {\n unmountFn();\n\n const stateWithoutQuery = state.setQueryParameter('query', undefined);\n\n if (!escapeHTML) {\n return stateWithoutQuery;\n }\n\n return stateWithoutQuery.setQueryParameters(\n Object.keys(TAG_PLACEHOLDER).reduce(\n (acc, key) => ({\n ...acc,\n [key]: undefined,\n }),\n {}\n )\n );\n },\n };\n };\n};\n\nexport default connectAutocomplete;\n","import type {\n AlgoliaSearchHelper as Helper,\n SearchParameters,\n} from 'algoliasearch-helper';\nimport type { Connector, TransformItems, WidgetRenderState } from '../../types';\nimport {\n checkRendering,\n createDocumentationMessageGenerator,\n warning,\n getRefinements,\n isEqual,\n noop,\n} from '../../lib/utils';\nimport type {\n Refinement as InternalRefinement,\n NumericRefinement as InternalNumericRefinement,\n} from '../../lib/utils/getRefinements';\n\ntype TrackedFilterRefinement = string | number | boolean;\n\nexport type ParamTrackedFilters = {\n [facetName: string]: (\n facetValues: TrackedFilterRefinement[]\n ) => TrackedFilterRefinement[];\n};\nexport type ParamTransformRuleContexts = (ruleContexts: string[]) => string[];\n\nexport type QueryRulesConnectorParams = {\n trackedFilters?: ParamTrackedFilters;\n transformRuleContexts?: ParamTransformRuleContexts;\n transformItems?: TransformItems;\n};\n\nexport type QueryRulesRenderState = {\n items: any[];\n};\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'query-rules',\n connector: true,\n});\n\nfunction hasStateRefinements(state: SearchParameters): boolean {\n return [\n state.disjunctiveFacetsRefinements,\n state.facetsRefinements,\n state.hierarchicalFacetsRefinements,\n state.numericRefinements,\n ].some((refinement) =>\n Boolean(refinement && Object.keys(refinement).length > 0)\n );\n}\n\n// A context rule must consist only of alphanumeric characters, hyphens, and underscores.\n// See https://www.algolia.com/doc/guides/managing-results/refine-results/merchandising-and-promoting/in-depth/implementing-query-rules/#context\nfunction escapeRuleContext(ruleName: string): string {\n return ruleName.replace(/[^a-z0-9-_]+/gi, '_');\n}\n\nfunction getRuleContextsFromTrackedFilters({\n helper,\n sharedHelperState,\n trackedFilters,\n}: {\n helper: Helper;\n sharedHelperState: SearchParameters;\n trackedFilters: ParamTrackedFilters;\n}): string[] {\n const ruleContexts = Object.keys(trackedFilters).reduce(\n (facets, facetName) => {\n const facetRefinements: TrackedFilterRefinement[] = getRefinements(\n helper.lastResults || {},\n sharedHelperState,\n true\n )\n .filter(\n (refinement: InternalRefinement) => refinement.attribute === facetName\n )\n .map(\n (refinement: InternalRefinement) =>\n (refinement as InternalNumericRefinement).numericValue ||\n refinement.name\n );\n\n const getTrackedFacetValues = trackedFilters[facetName];\n const trackedFacetValues = getTrackedFacetValues(facetRefinements);\n\n return [\n ...facets,\n ...facetRefinements\n .filter((facetRefinement) =>\n trackedFacetValues.includes(facetRefinement)\n )\n .map((facetValue) =>\n escapeRuleContext(`ais-${facetName}-${facetValue}`)\n ),\n ];\n },\n []\n );\n\n return ruleContexts;\n}\n\nfunction applyRuleContexts(\n this: {\n helper: Helper;\n initialRuleContexts: string[];\n trackedFilters: ParamTrackedFilters;\n transformRuleContexts: ParamTransformRuleContexts;\n },\n event: { state: SearchParameters }\n): void {\n const { helper, initialRuleContexts, trackedFilters, transformRuleContexts } =\n this;\n\n const sharedHelperState = event.state;\n const previousRuleContexts: string[] = sharedHelperState.ruleContexts || [];\n const newRuleContexts = getRuleContextsFromTrackedFilters({\n helper,\n sharedHelperState,\n trackedFilters,\n });\n const nextRuleContexts = [...initialRuleContexts, ...newRuleContexts];\n\n warning(\n nextRuleContexts.length <= 10,\n `\nThe maximum number of \\`ruleContexts\\` is 10. They have been sliced to that limit.\nConsider using \\`transformRuleContexts\\` to minimize the number of rules sent to Algolia.\n`\n );\n\n const ruleContexts = transformRuleContexts(nextRuleContexts).slice(0, 10);\n\n if (!isEqual(previousRuleContexts, ruleContexts)) {\n helper.overrideStateWithoutTriggeringChangeEvent({\n ...sharedHelperState,\n ruleContexts,\n });\n }\n}\n\nexport type QueryRulesWidgetDescription = {\n $$type: 'ais.queryRules';\n renderState: QueryRulesRenderState;\n indexRenderState: {\n queryRules: WidgetRenderState<\n QueryRulesRenderState,\n QueryRulesConnectorParams\n >;\n };\n};\n\nexport type QueryRulesConnector = Connector<\n QueryRulesWidgetDescription,\n QueryRulesConnectorParams\n>;\n\nconst connectQueryRules: QueryRulesConnector = function connectQueryRules(\n render,\n unmount = noop\n) {\n checkRendering(render, withUsage());\n\n return (widgetParams) => {\n const {\n trackedFilters = {} as ParamTrackedFilters,\n transformRuleContexts = ((rules) => rules) as ParamTransformRuleContexts,\n transformItems = ((items) => items) as NonNullable<\n QueryRulesConnectorParams['transformItems']\n >,\n } = widgetParams || {};\n\n Object.keys(trackedFilters).forEach((facetName) => {\n if (typeof trackedFilters[facetName] !== 'function') {\n throw new Error(\n withUsage(\n `'The \"${facetName}\" filter value in the \\`trackedFilters\\` option expects a function.`\n )\n );\n }\n });\n\n const hasTrackedFilters = Object.keys(trackedFilters).length > 0;\n\n // We store the initial rule contexts applied before creating the widget\n // so that we do not override them with the rules created from `trackedFilters`.\n let initialRuleContexts: string[] = [];\n let onHelperChange: (event: { state: SearchParameters }) => void;\n\n return {\n $$type: 'ais.queryRules',\n\n init(initOptions) {\n const { helper, state, instantSearchInstance } = initOptions;\n\n initialRuleContexts = state.ruleContexts || [];\n onHelperChange = applyRuleContexts.bind({\n helper,\n initialRuleContexts,\n trackedFilters,\n transformRuleContexts,\n });\n\n if (hasTrackedFilters) {\n // We need to apply the `ruleContexts` based on the `trackedFilters`\n // before the helper changes state in some cases:\n // - Some filters are applied on the first load (e.g. using `configure`)\n // - The `transformRuleContexts` option sets initial `ruleContexts`.\n if (\n hasStateRefinements(state) ||\n Boolean(widgetParams.transformRuleContexts)\n ) {\n onHelperChange({ state });\n }\n\n // We track every change in the helper to override its state and add\n // any `ruleContexts` needed based on the `trackedFilters`.\n helper.on('change', onHelperChange);\n }\n\n render(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n render(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n getWidgetRenderState({ results }) {\n const { userData = [] } = results || {};\n const items = transformItems(userData, { results });\n\n return {\n items,\n widgetParams,\n };\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n queryRules: this.getWidgetRenderState(renderOptions),\n };\n },\n\n dispose({ helper, state }) {\n unmount();\n\n if (hasTrackedFilters) {\n helper.removeListener('change', onHelperChange);\n\n return state.setQueryParameter('ruleContexts', initialRuleContexts);\n }\n\n return state;\n },\n };\n };\n};\n\nexport default connectQueryRules;\n","// `SpeechRecognition` is an API used on the browser so we can safely disable\n// the `window` check.\n/* eslint-disable no-restricted-globals */\n/* global SpeechRecognition SpeechRecognitionEvent */\nimport type {\n CreateVoiceSearchHelper,\n Status,\n VoiceListeningState,\n} from './types';\n\nconst createVoiceSearchHelper: CreateVoiceSearchHelper =\n function createVoiceSearchHelper({\n searchAsYouSpeak,\n language,\n onQueryChange,\n onStateChange,\n }) {\n const SpeechRecognitionAPI: new () => SpeechRecognition =\n (window as any).webkitSpeechRecognition ||\n (window as any).SpeechRecognition;\n const getDefaultState = (status: Status): VoiceListeningState => ({\n status,\n transcript: '',\n isSpeechFinal: false,\n errorCode: undefined,\n });\n let state: VoiceListeningState = getDefaultState('initial');\n let recognition: SpeechRecognition | undefined;\n\n const isBrowserSupported = (): boolean => Boolean(SpeechRecognitionAPI);\n\n const isListening = (): boolean =>\n state.status === 'askingPermission' ||\n state.status === 'waiting' ||\n state.status === 'recognizing';\n\n const setState = (newState: Partial = {}): void => {\n state = { ...state, ...newState };\n onStateChange();\n };\n\n const getState = (): VoiceListeningState => state;\n\n const resetState = (status: Status = 'initial'): void => {\n setState(getDefaultState(status));\n };\n\n const onStart = (): void => {\n setState({\n status: 'waiting',\n });\n };\n\n const onError = (event: Event): void => {\n setState({ status: 'error', errorCode: (event as any).error });\n };\n\n const onResult = (event: SpeechRecognitionEvent): void => {\n setState({\n status: 'recognizing',\n transcript:\n (event.results[0] &&\n event.results[0][0] &&\n event.results[0][0].transcript) ||\n '',\n isSpeechFinal: event.results[0] && event.results[0].isFinal,\n });\n if (searchAsYouSpeak && state.transcript) {\n onQueryChange(state.transcript);\n }\n };\n\n const onEnd = (): void => {\n if (!state.errorCode && state.transcript && !searchAsYouSpeak) {\n onQueryChange(state.transcript);\n }\n if (state.status !== 'error') {\n setState({ status: 'finished' });\n }\n };\n\n const startListening = (): void => {\n recognition = new SpeechRecognitionAPI();\n if (!recognition) {\n return;\n }\n resetState('askingPermission');\n recognition.interimResults = true;\n\n if (language) {\n recognition.lang = language;\n }\n\n recognition.addEventListener('start', onStart);\n recognition.addEventListener('error', onError);\n recognition.addEventListener('result', onResult);\n recognition.addEventListener('end', onEnd);\n recognition.start();\n };\n\n const dispose = (): void => {\n if (!recognition) {\n return;\n }\n recognition.stop();\n recognition.removeEventListener('start', onStart);\n recognition.removeEventListener('error', onError);\n recognition.removeEventListener('result', onResult);\n recognition.removeEventListener('end', onEnd);\n recognition = undefined;\n };\n\n const stopListening = (): void => {\n dispose();\n // Because `dispose` removes event listeners, `end` listener is not called.\n // So we're setting the `status` as `finished` here.\n // If we don't do it, it will be still `waiting` or `recognizing`.\n resetState('finished');\n };\n\n return {\n getState,\n isBrowserSupported,\n isListening,\n startListening,\n stopListening,\n dispose,\n };\n };\n\nexport default createVoiceSearchHelper;\n","import type { PlainSearchParameters } from 'algoliasearch-helper';\nimport {\n checkRendering,\n createDocumentationMessageGenerator,\n noop,\n} from '../../lib/utils';\nimport type { Connector, WidgetRenderState } from '../../types';\nimport builtInCreateVoiceSearchHelper from '../../lib/voiceSearchHelper';\nimport type {\n CreateVoiceSearchHelper,\n VoiceListeningState,\n} from '../../lib/voiceSearchHelper/types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'voice-search',\n connector: true,\n});\n\nexport type VoiceSearchConnectorParams = {\n searchAsYouSpeak?: boolean;\n language?: string;\n additionalQueryParameters?: (params: {\n query: string;\n }) => PlainSearchParameters | void;\n createVoiceSearchHelper?: CreateVoiceSearchHelper;\n};\n\nexport type VoiceSearchRenderState = {\n isBrowserSupported: boolean;\n isListening: boolean;\n toggleListening: () => void;\n voiceListeningState: VoiceListeningState;\n};\n\nexport type VoiceSearchWidgetDescription = {\n $$type: 'ais.voiceSearch';\n renderState: VoiceSearchRenderState;\n indexRenderState: {\n voiceSearch: WidgetRenderState<\n VoiceSearchRenderState,\n VoiceSearchConnectorParams\n >;\n };\n indexUiState: {\n query: string;\n };\n};\n\nexport type VoiceSearchConnector = Connector<\n VoiceSearchWidgetDescription,\n VoiceSearchConnectorParams\n>;\n\nconst connectVoiceSearch: VoiceSearchConnector = function connectVoiceSearch(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n searchAsYouSpeak = false,\n language,\n additionalQueryParameters,\n createVoiceSearchHelper = builtInCreateVoiceSearchHelper,\n } = widgetParams;\n\n return {\n $$type: 'ais.voiceSearch',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n voiceSearch: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState(renderOptions) {\n const { helper, instantSearchInstance } = renderOptions;\n if (!(this as any)._refine) {\n (this as any)._refine = (query: string): void => {\n if (query !== helper.state.query) {\n const queryLanguages = language\n ? [language.split('-')[0]]\n : undefined;\n helper.setQueryParameter('queryLanguages', queryLanguages);\n\n if (typeof additionalQueryParameters === 'function') {\n helper.setState(\n helper.state.setQueryParameters({\n ignorePlurals: true,\n removeStopWords: true,\n // @ts-ignore (optionalWords only allows array in v3, while string is also valid)\n optionalWords: query,\n ...additionalQueryParameters({ query }),\n })\n );\n }\n\n helper.setQuery(query).search();\n }\n };\n }\n\n if (!(this as any)._voiceSearchHelper) {\n (this as any)._voiceSearchHelper = createVoiceSearchHelper({\n searchAsYouSpeak,\n language,\n onQueryChange: (query) => (this as any)._refine(query),\n onStateChange: () => {\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n });\n }\n\n const {\n isBrowserSupported,\n isListening,\n startListening,\n stopListening,\n getState,\n } = (this as any)._voiceSearchHelper;\n\n return {\n isBrowserSupported: isBrowserSupported(),\n isListening: isListening(),\n toggleListening() {\n if (!isBrowserSupported()) {\n return;\n }\n if (isListening()) {\n stopListening();\n } else {\n startListening();\n }\n },\n voiceListeningState: getState(),\n widgetParams,\n };\n },\n\n dispose({ state }) {\n (this as any)._voiceSearchHelper.dispose();\n\n unmountFn();\n\n let newState = state;\n if (typeof additionalQueryParameters === 'function') {\n const additional = additionalQueryParameters({ query: '' });\n const toReset = additional\n ? (\n Object.keys(additional) as Array\n ).reduce((acc, current) => {\n // @ts-ignore search parameters is typed as readonly in v4\n acc[current] = undefined;\n return acc;\n }, {})\n : {};\n newState = state.setQueryParameters({\n // @ts-ignore (queryLanguages is not added to algoliasearch v3)\n queryLanguages: undefined,\n ignorePlurals: undefined,\n removeStopWords: undefined,\n optionalWords: undefined,\n ...toReset,\n });\n }\n\n return newState.setQueryParameter('query', undefined);\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n const query = searchParameters.query || '';\n\n if (!query) {\n return uiState;\n }\n\n return {\n ...uiState,\n query,\n };\n },\n\n getWidgetSearchParameters(searchParameters, { uiState }) {\n return searchParameters.setQueryParameter('query', uiState.query || '');\n },\n };\n };\n};\n\nexport default connectVoiceSearch;\n","import {\n checkRendering,\n createDocumentationMessageGenerator,\n createConcurrentSafePromise,\n addQueryID,\n debounce,\n addAbsolutePosition,\n noop,\n escapeHits,\n} from '../../lib/utils';\nimport type { DebouncedFunction } from '../../lib/utils/debounce';\nimport type {\n Connector,\n Hits,\n Hit,\n FindAnswersOptions,\n FindAnswersParameters,\n FindAnswersResponse,\n WidgetRenderState,\n} from '../../types';\n\ntype IndexWithAnswers = {\n readonly findAnswers: any;\n};\n\nfunction hasFindAnswersMethod(\n answersIndex: IndexWithAnswers | any\n): answersIndex is IndexWithAnswers {\n return typeof (answersIndex as IndexWithAnswers).findAnswers === 'function';\n}\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'answers',\n connector: true,\n});\n\nexport type AnswersRenderState = {\n /**\n * The matched hits from Algolia API.\n */\n hits: Hits;\n\n /**\n * Whether it's still loading the results from the Answers API.\n */\n isLoading: boolean;\n};\n\nexport type AnswersConnectorParams = {\n /**\n * Attributes to use for predictions.\n * If empty, we use all `searchableAttributes` to find answers.\n * All your `attributesForPrediction` must be part of your `searchableAttributes`.\n */\n attributesForPrediction?: string[];\n\n /**\n * The languages in the query. Currently only supports `en`.\n */\n queryLanguages: ['en'];\n\n /**\n * Maximum number of answers to retrieve from the Answers Engine.\n * Cannot be greater than 1000.\n * @default 1\n */\n nbHits?: number;\n\n /**\n * Debounce time in milliseconds to debounce render\n * @default 100\n */\n renderDebounceTime?: number;\n\n /**\n * Debounce time in milliseconds to debounce search\n * @default 100\n */\n searchDebounceTime?: number;\n\n /**\n * Whether to escape HTML tags from hits string values.\n *\n * @default true\n */\n escapeHTML?: boolean;\n\n /**\n * Extra parameters to pass to findAnswers method.\n * @default {}\n */\n extraParameters?: FindAnswersOptions;\n};\n\nexport type AnswersWidgetDescription = {\n $$type: 'ais.answers';\n renderState: AnswersRenderState;\n indexRenderState: {\n answers: WidgetRenderState;\n };\n};\n\nexport type AnswersConnector = Connector<\n AnswersWidgetDescription,\n AnswersConnectorParams\n>;\n\nconst connectAnswers: AnswersConnector = function connectAnswers(\n renderFn,\n unmountFn = noop\n) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n queryLanguages,\n attributesForPrediction,\n nbHits = 1,\n renderDebounceTime = 100,\n searchDebounceTime = 100,\n escapeHTML = true,\n extraParameters = {},\n } = widgetParams || {};\n\n // @ts-expect-error checking for the wrong value\n if (!queryLanguages || queryLanguages.length === 0) {\n throw new Error(\n withUsage('The `queryLanguages` expects an array of strings.')\n );\n }\n\n const runConcurrentSafePromise =\n createConcurrentSafePromise>();\n\n let lastHits: FindAnswersResponse['hits'] = [];\n let isLoading = false;\n const debouncedRender = debounce(renderFn, renderDebounceTime);\n\n // this does not directly use DebouncedFunction, since then the generic will disappear\n let debouncedRefine: DebouncedFunction<\n (...params: FindAnswersParameters) => Promise>\n >;\n\n return {\n $$type: 'ais.answers',\n\n init(initOptions) {\n const { state, instantSearchInstance } = initOptions;\n const answersIndex = instantSearchInstance.client.initIndex!(\n state.index\n );\n if (!hasFindAnswersMethod(answersIndex)) {\n throw new Error(withUsage('`algoliasearch` >= 4.8.0 required.'));\n }\n debouncedRefine = debounce(\n answersIndex.findAnswers,\n searchDebounceTime\n );\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance: initOptions.instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const query = renderOptions.state.query;\n if (!query) {\n // renders nothing with empty query\n lastHits = [];\n isLoading = false;\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n return;\n }\n\n // render the loader\n lastHits = [];\n isLoading = true;\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n\n // call /answers API\n runConcurrentSafePromise(\n debouncedRefine(query, queryLanguages, {\n ...extraParameters,\n nbHits,\n attributesForPrediction,\n })\n ).then((result) => {\n if (!result) {\n // It's undefined when it's debounced.\n return;\n }\n\n if (escapeHTML && result.hits.length > 0) {\n result.hits = escapeHits(result.hits);\n }\n\n const hitsWithAbsolutePosition = addAbsolutePosition(\n result.hits,\n 0,\n nbHits\n );\n\n const hitsWithAbsolutePositionAndQueryID = addQueryID(\n hitsWithAbsolutePosition,\n result.queryID\n );\n\n lastHits = hitsWithAbsolutePositionAndQueryID;\n isLoading = false;\n debouncedRender(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n });\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n answers: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState() {\n return {\n hits: lastHits,\n isLoading,\n widgetParams,\n };\n },\n\n dispose({ state }) {\n unmountFn();\n return state;\n },\n\n getWidgetSearchParameters(state) {\n return state;\n },\n };\n };\n};\n\nexport default connectAnswers;\n","import type { Connector, WidgetRenderState } from '../../types';\nimport { noop } from '../../lib/utils';\n\nexport type RelevantSortConnectorParams = Record;\n\ntype Refine = (relevancyStrictness: number) => void;\n\nexport type RelevantSortRenderState = {\n /**\n * Indicates if it has appliedRelevancyStrictness greater than zero\n */\n isRelevantSorted: boolean;\n\n /**\n * Indicates if the results come from a virtual replica\n */\n isVirtualReplica: boolean;\n\n /**\n * Indicates if search state can be refined\n */\n canRefine: boolean;\n\n /**\n * Sets the value as relevancyStrictness and trigger a new search\n */\n refine: Refine;\n};\n\nexport type RelevantSortWidgetDescription = {\n $$type: 'ais.relevantSort';\n renderState: RelevantSortRenderState;\n indexRenderState: {\n relevantSort: WidgetRenderState<\n RelevantSortRenderState,\n RelevantSortConnectorParams\n >;\n };\n indexUiState: {\n relevantSort: number;\n };\n};\n\nexport type RelevantSortConnector = Connector<\n RelevantSortWidgetDescription,\n RelevantSortConnectorParams\n>;\n\nconst connectRelevantSort: RelevantSortConnector = function connectRelevantSort(\n renderFn = noop,\n unmountFn = noop\n) {\n return (widgetParams) => {\n type ConnectorState = {\n refine?: Refine;\n };\n\n const connectorState: ConnectorState = {};\n\n return {\n $$type: 'ais.relevantSort',\n\n init(initOptions) {\n const { instantSearchInstance } = initOptions;\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance,\n },\n true\n );\n },\n\n render(renderOptions) {\n const { instantSearchInstance } = renderOptions;\n\n renderFn(\n {\n ...this.getWidgetRenderState(renderOptions),\n instantSearchInstance,\n },\n false\n );\n },\n\n dispose({ state }) {\n unmountFn();\n\n return state.setQueryParameter('relevancyStrictness', undefined);\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n relevantSort: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState({ results, helper }) {\n if (!connectorState.refine) {\n connectorState.refine = (relevancyStrictness: number | undefined) => {\n helper\n .setQueryParameter('relevancyStrictness', relevancyStrictness)\n .search();\n };\n }\n\n const { appliedRelevancyStrictness } = results || {};\n\n const isVirtualReplica = appliedRelevancyStrictness !== undefined;\n\n return {\n isRelevantSorted:\n typeof appliedRelevancyStrictness !== 'undefined' &&\n appliedRelevancyStrictness > 0,\n isVirtualReplica,\n canRefine: isVirtualReplica,\n refine: connectorState.refine,\n widgetParams,\n };\n },\n\n getWidgetSearchParameters(state, { uiState }) {\n return state.setQueryParameter(\n 'relevancyStrictness',\n uiState.relevantSort ?? state.relevancyStrictness\n );\n },\n\n getWidgetUiState(uiState, { searchParameters }) {\n return {\n ...uiState,\n relevantSort:\n searchParameters.relevancyStrictness || uiState.relevantSort,\n };\n },\n };\n };\n};\n\nexport default connectRelevantSort;\n","import {\n checkRendering,\n createDocumentationMessageGenerator,\n getWidgetAttribute,\n noop,\n warning,\n} from '../../lib/utils';\nimport type {\n Connector,\n TransformItems,\n TransformItemsMetadata,\n Widget,\n} from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'dynamic-widgets',\n connector: true,\n});\n\nexport type DynamicWidgetsRenderState = {\n attributesToRender: string[];\n};\n\nexport type DynamicWidgetsConnectorParams = {\n /**\n * An array of widgets, displayed in the order defined by `facetOrdering`.\n */\n widgets: Widget[];\n\n /**\n * Function to return a fallback widget when an attribute isn't found in\n * `widgets`.\n */\n fallbackWidget?(args: {\n /** The attribute name to create a widget for. */\n attribute: string;\n }): Widget;\n\n /**\n * Function to transform the items to render.\n * The function also exposes the full search response.\n */\n transformItems?: TransformItems<\n string,\n Omit & {\n results: NonNullable;\n }\n >;\n\n /**\n * To prevent unneeded extra network requests when widgets mount or unmount,\n * we request all facet values.\n *\n * @default ['*']\n */\n facets?: ['*'] | never[];\n\n /**\n * If you have more than 20 facet values pinned, you need to increase the\n * maxValuesPerFacet to at least that value.\n *\n * @default 20\n */\n maxValuesPerFacet?: number;\n};\n\nexport type DynamicWidgetsWidgetDescription = {\n $$type: 'ais.dynamicWidgets';\n renderState: DynamicWidgetsRenderState;\n indexRenderState: {\n dynamicWidgets: DynamicWidgetsRenderState;\n };\n};\n\nexport type DynamicWidgetsConnector = Connector<\n DynamicWidgetsWidgetDescription,\n DynamicWidgetsConnectorParams\n>;\n\nconst MAX_WILDCARD_FACETS = 20;\n\nconst connectDynamicWidgets: DynamicWidgetsConnector =\n function connectDynamicWidgets(renderFn, unmountFn = noop) {\n checkRendering(renderFn, withUsage());\n\n return (widgetParams) => {\n const {\n widgets,\n maxValuesPerFacet = 20,\n facets = ['*'],\n transformItems = (items) => items,\n fallbackWidget,\n } = widgetParams;\n\n if (\n !(\n widgets &&\n Array.isArray(widgets) &&\n widgets.every((widget) => typeof widget === 'object')\n )\n ) {\n throw new Error(\n withUsage('The `widgets` option expects an array of widgets.')\n );\n }\n\n if (\n !(\n Array.isArray(facets) &&\n facets.length <= 1 &&\n (facets[0] === '*' || facets[0] === undefined)\n )\n ) {\n throw new Error(\n withUsage(\n `The \\`facets\\` option only accepts [] or [\"*\"], you passed ${JSON.stringify(\n facets\n )}`\n )\n );\n }\n\n const localWidgets: Map =\n new Map();\n\n return {\n $$type: 'ais.dynamicWidgets',\n init(initOptions) {\n widgets.forEach((widget) => {\n const attribute = getWidgetAttribute(widget, initOptions);\n localWidgets.set(attribute, { widget, isMounted: false });\n });\n\n renderFn(\n {\n ...this.getWidgetRenderState(initOptions),\n instantSearchInstance: initOptions.instantSearchInstance,\n },\n true\n );\n },\n render(renderOptions) {\n const { parent } = renderOptions;\n const renderState = this.getWidgetRenderState(renderOptions);\n\n const widgetsToUnmount: Widget[] = [];\n const widgetsToMount: Widget[] = [];\n\n if (fallbackWidget) {\n renderState.attributesToRender.forEach((attribute) => {\n if (!localWidgets.has(attribute)) {\n const widget = fallbackWidget({ attribute });\n localWidgets.set(attribute, { widget, isMounted: false });\n }\n });\n }\n\n localWidgets.forEach(({ widget, isMounted }, attribute) => {\n const shouldMount =\n renderState.attributesToRender.indexOf(attribute) > -1;\n\n if (!isMounted && shouldMount) {\n widgetsToMount.push(widget);\n localWidgets.set(attribute, {\n widget,\n isMounted: true,\n });\n } else if (isMounted && !shouldMount) {\n widgetsToUnmount.push(widget);\n localWidgets.set(attribute, {\n widget,\n isMounted: false,\n });\n }\n });\n\n parent!.addWidgets(widgetsToMount);\n // make sure this only happens after the regular render, otherwise it\n // happens too quick, since render is \"deferred\" for the next microtask,\n // so this needs to be a whole task later\n setTimeout(() => parent!.removeWidgets(widgetsToUnmount), 0);\n\n renderFn(\n {\n ...renderState,\n instantSearchInstance: renderOptions.instantSearchInstance,\n },\n false\n );\n },\n dispose({ parent }) {\n const toRemove: Widget[] = [];\n localWidgets.forEach(({ widget, isMounted }) => {\n if (isMounted) {\n toRemove.push(widget);\n }\n });\n parent.removeWidgets(toRemove);\n\n unmountFn();\n },\n getWidgetSearchParameters(state) {\n // broadening the scope of facets to avoid conflict between never and *\n return (facets as string[]).reduce(\n (acc, curr) => acc.addFacet(curr),\n state.setQueryParameters({\n maxValuesPerFacet: Math.max(\n maxValuesPerFacet || 0,\n state.maxValuesPerFacet || 0\n ),\n })\n );\n },\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n dynamicWidgets: this.getWidgetRenderState(renderOptions),\n };\n },\n getWidgetRenderState({ results, state }) {\n if (!results) {\n return { attributesToRender: [], widgetParams };\n }\n\n const attributesToRender = transformItems(\n results.renderingContent?.facetOrdering?.facets?.order ?? [],\n { results }\n );\n\n if (!Array.isArray(attributesToRender)) {\n throw new Error(\n withUsage(\n 'The `transformItems` option expects a function that returns an Array.'\n )\n );\n }\n\n warning(\n maxValuesPerFacet >= (state.maxValuesPerFacet || 0),\n `The maxValuesPerFacet set by dynamic widgets (${maxValuesPerFacet}) is smaller than one of the limits set by a widget (${state.maxValuesPerFacet}). This causes a mismatch in query parameters and thus an extra network request when that widget is mounted.`\n );\n\n warning(\n attributesToRender.length <= MAX_WILDCARD_FACETS ||\n widgetParams.facets !== undefined,\n `More than ${MAX_WILDCARD_FACETS} facets are requested to be displayed without explicitly setting which facets to retrieve. This could have a performance impact. Set \"facets\" to [] to do two smaller network requests, or explicitly to ['*'] to avoid this warning.`\n );\n\n return {\n attributesToRender,\n widgetParams,\n };\n },\n };\n };\n };\n\nexport default connectDynamicWidgets;\n","export { default as connectClearRefinements } from './clear-refinements/connectClearRefinements';\nexport { default as connectCurrentRefinements } from './current-refinements/connectCurrentRefinements';\nexport { default as connectHierarchicalMenu } from './hierarchical-menu/connectHierarchicalMenu';\nexport { default as connectHits } from './hits/connectHits';\nexport { default as connectHitsWithInsights } from './hits/connectHitsWithInsights';\nexport { default as connectHitsPerPage } from './hits-per-page/connectHitsPerPage';\nexport { default as connectInfiniteHits } from './infinite-hits/connectInfiniteHits';\nexport { default as connectInfiniteHitsWithInsights } from './infinite-hits/connectInfiniteHitsWithInsights';\nexport { default as connectMenu } from './menu/connectMenu';\nexport { default as connectNumericMenu } from './numeric-menu/connectNumericMenu';\nexport { default as connectPagination } from './pagination/connectPagination';\nexport { default as connectRange } from './range/connectRange';\nexport { default as connectRefinementList } from './refinement-list/connectRefinementList';\nexport { default as connectSearchBox } from './search-box/connectSearchBox';\nexport { default as connectSortBy } from './sort-by/connectSortBy';\nexport { default as connectRatingMenu } from './rating-menu/connectRatingMenu';\nexport { default as connectStats } from './stats/connectStats';\nexport { default as connectToggleRefinement } from './toggle-refinement/connectToggleRefinement';\nexport { default as connectBreadcrumb } from './breadcrumb/connectBreadcrumb';\nexport { default as connectGeoSearch } from './geo-search/connectGeoSearch';\nexport { default as connectPoweredBy } from './powered-by/connectPoweredBy';\nexport { default as connectConfigure } from './configure/connectConfigure';\nexport { default as EXPERIMENTAL_connectConfigureRelatedItems } from './configure-related-items/connectConfigureRelatedItems';\nexport { default as connectAutocomplete } from './autocomplete/connectAutocomplete';\nexport { default as connectQueryRules } from './query-rules/connectQueryRules';\nexport { default as connectVoiceSearch } from './voice-search/connectVoiceSearch';\nexport { default as EXPERIMENTAL_connectAnswers } from './answers/connectAnswers';\nexport { default as connectRelevantSort } from './relevant-sort/connectRelevantSort';\n\nimport connectDynamicWidgets from './dynamic-widgets/connectDynamicWidgets';\nexport { connectDynamicWidgets };\nimport { deprecate } from '../lib/utils';\n/** @deprecated use connectDynamicWidgets */\nexport const EXPERIMENTAL_connectDynamicWidgets = deprecate(\n connectDynamicWidgets,\n 'use connectDynamicWidgets'\n);\n","import type { SearchParameters, SearchResults } from 'algoliasearch-helper';\nimport { createDocumentationMessageGenerator, warning } from '../../lib/utils';\nimport type { WidgetFactory, WidgetRenderState } from '../../types';\n\nexport type AnalyticsWidgetParamsPushFunction = (\n /**\n * Contains the search parameters, serialized as a query string.\n */\n formattedParameters: string,\n\n /**\n * Contains the whole search state.\n */\n state: SearchParameters,\n\n /**\n * The last received results.\n */\n results: SearchResults\n) => void;\n\nexport type AnalyticsWidgetParams = {\n /**\n * A function that is called every time the query or refinements changes. You\n * need to add the logic to push the data to your analytics platform.\n */\n pushFunction: AnalyticsWidgetParamsPushFunction;\n\n /**\n * The number of milliseconds between the last search keystroke and calling `pushFunction`.\n *\n * @default 3000\n */\n delay?: number;\n\n /**\n * Triggers `pushFunction` after click on page or redirecting the page. This is useful if\n * you want the pushFunction to be called for the last actions before the user leaves the\n * current page, even if the delay wasn’t reached.\n *\n * @default false\n */\n triggerOnUIInteraction?: boolean;\n\n /**\n * Triggers `pushFunction` when InstantSearch is initialized. This means\n * the `pushFunction` might be called even though the user didn’t perfom\n * any search-related action.\n *\n * @default true\n */\n pushInitialSearch?: boolean;\n\n /**\n * Triggers `pushFunction` when the page changes, either through the UI or programmatically.\n *\n * @default false\n */\n pushPagination?: boolean;\n};\n\nconst withUsage = createDocumentationMessageGenerator({ name: 'analytics' });\n\nexport type AnalyticsWidgetDescription = {\n $$type: 'ais.analytics';\n $$widgetType: 'ais.analytics';\n renderState: Record;\n indexRenderState: {\n analytics: WidgetRenderState<\n Record,\n AnalyticsWidgetParams\n >;\n };\n};\n\nexport type AnalyticsWidget = WidgetFactory<\n AnalyticsWidgetDescription,\n AnalyticsWidgetParams,\n AnalyticsWidgetParams\n>;\n\n// @major this widget will be removed from the next major version.\nconst analytics: AnalyticsWidget = function analytics(widgetParams) {\n const {\n pushFunction,\n delay = 3000,\n triggerOnUIInteraction = false,\n pushInitialSearch = true,\n pushPagination = false,\n } = widgetParams || {};\n\n if (!pushFunction) {\n throw new Error(withUsage('The `pushFunction` option is required.'));\n }\n\n warning(\n false,\n `\\`analytics\\` widget has been deprecated. It is still supported in 4.x releases, but not further. It is replaced by the \\`insights\\` middleware.\n\nFor the migration, visit https://www.algolia.com/doc/guides/building-search-ui/upgrade-guides/js/#analytics-widget`\n );\n\n type AnalyticsState = {\n results: SearchResults;\n state: SearchParameters;\n } | null;\n\n let cachedState: AnalyticsState = null;\n\n type RefinementParameters = {\n [key: string]: string[];\n };\n\n const serializeRefinements = function (\n parameters: RefinementParameters\n ): string {\n const refinements: string[] = [];\n\n for (const parameter in parameters) {\n if (parameters.hasOwnProperty(parameter)) {\n const values = parameters[parameter].join('+');\n\n refinements.push(\n `${encodeURIComponent(parameter)}=${encodeURIComponent(\n parameter\n )}_${encodeURIComponent(values)}`\n );\n }\n }\n\n return refinements.join('&');\n };\n\n const serializeNumericRefinements = function (\n numericRefinements: SearchParameters['numericRefinements']\n ): string {\n const refinements: string[] = [];\n\n for (const attribute in numericRefinements) {\n if (numericRefinements.hasOwnProperty(attribute)) {\n const filter = numericRefinements[attribute];\n\n if (filter.hasOwnProperty('>=') && filter.hasOwnProperty('<=')) {\n if (\n filter['>='] &&\n filter['>='][0] === filter['<='] &&\n filter['<='][0]\n ) {\n refinements.push(`${attribute}=${attribute}_${filter['>=']}`);\n } else {\n refinements.push(\n `${attribute}=${attribute}_${filter['>=']}to${filter['<=']}`\n );\n }\n } else if (filter.hasOwnProperty('>=')) {\n refinements.push(`${attribute}=${attribute}_from${filter['>=']}`);\n } else if (filter.hasOwnProperty('<=')) {\n refinements.push(`${attribute}=${attribute}_to${filter['<=']}`);\n } else if (filter.hasOwnProperty('=')) {\n const equals: string[] = [];\n\n for (const equal in filter['=']) {\n // eslint-disable-next-line max-depth\n if (filter['='].hasOwnProperty(equal)) {\n // @ts-ignore somehow 'equal' is a string, even though it's a number?\n equals.push(filter['='][equal]);\n }\n }\n\n refinements.push(`${attribute}=${attribute}_${equals.join('-')}`);\n }\n }\n }\n\n return refinements.join('&');\n };\n\n let lastSentData = '';\n\n const sendAnalytics = function (analyticsState: AnalyticsState | null): void {\n if (analyticsState === null) {\n return;\n }\n\n const serializedParams: string[] = [];\n\n const serializedRefinements = serializeRefinements({\n ...analyticsState.state.disjunctiveFacetsRefinements,\n ...analyticsState.state.facetsRefinements,\n ...analyticsState.state.hierarchicalFacetsRefinements,\n });\n\n const serializedNumericRefinements = serializeNumericRefinements(\n analyticsState.state.numericRefinements\n );\n\n if (serializedRefinements !== '') {\n serializedParams.push(serializedRefinements);\n }\n\n if (serializedNumericRefinements !== '') {\n serializedParams.push(serializedNumericRefinements);\n }\n\n const stringifiedParams = serializedParams.join('&');\n\n let dataToSend = `Query: ${\n analyticsState.state.query || ''\n }, ${stringifiedParams}`;\n if (pushPagination === true) {\n dataToSend += `, Page: ${analyticsState.state.page || 0}`;\n }\n\n if (lastSentData !== dataToSend) {\n pushFunction(\n stringifiedParams,\n analyticsState.state,\n analyticsState.results\n );\n\n lastSentData = dataToSend;\n }\n };\n\n let pushTimeout: number;\n let isInitialSearch = true;\n\n if (pushInitialSearch === true) {\n isInitialSearch = false;\n }\n\n const onClick = (): void => {\n sendAnalytics(cachedState);\n };\n\n const onUnload = (): void => {\n sendAnalytics(cachedState);\n };\n\n return {\n $$type: 'ais.analytics',\n $$widgetType: 'ais.analytics',\n\n init() {\n if (triggerOnUIInteraction === true) {\n document.addEventListener('click', onClick);\n window.addEventListener('beforeunload', onUnload);\n }\n },\n\n render({ results, state }) {\n if (isInitialSearch === true) {\n isInitialSearch = false;\n\n return;\n }\n\n cachedState = { results, state };\n\n if (pushTimeout) {\n clearTimeout(pushTimeout);\n }\n\n pushTimeout = window.setTimeout(() => sendAnalytics(cachedState), delay);\n },\n\n dispose() {\n if (triggerOnUIInteraction === true) {\n document.removeEventListener('click', onClick);\n window.removeEventListener('beforeunload', onUnload);\n }\n },\n\n getRenderState(renderState, renderOptions) {\n return {\n ...renderState,\n analytics: this.getWidgetRenderState(renderOptions),\n };\n },\n\n getWidgetRenderState() {\n return {\n widgetParams,\n };\n },\n };\n};\n\nexport default analytics;\n","/*!\n Copyright (c) 2017 Jed Watson.\n Licensed under the MIT License (MIT), see\n http://jedwatson.github.io/classnames\n*/\n/* global define */\n\n(function () {\n\t'use strict';\n\n\tvar hasOwn = {}.hasOwnProperty;\n\n\tfunction classNames () {\n\t\tvar classes = [];\n\n\t\tfor (var i = 0; i < arguments.length; i++) {\n\t\t\tvar arg = arguments[i];\n\t\t\tif (!arg) continue;\n\n\t\t\tvar argType = typeof arg;\n\n\t\t\tif (argType === 'string' || argType === 'number') {\n\t\t\t\tclasses.push(arg);\n\t\t\t} else if (Array.isArray(arg) && arg.length) {\n\t\t\t\tvar inner = classNames.apply(null, arg);\n\t\t\t\tif (inner) {\n\t\t\t\t\tclasses.push(inner);\n\t\t\t\t}\n\t\t\t} else if (argType === 'object') {\n\t\t\t\tfor (var key in arg) {\n\t\t\t\t\tif (hasOwn.call(arg, key) && arg[key]) {\n\t\t\t\t\t\tclasses.push(key);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn classes.join(' ');\n\t}\n\n\tif (typeof module !== 'undefined' && module.exports) {\n\t\tclassNames.default = classNames;\n\t\tmodule.exports = classNames;\n\t} else if (typeof define === 'function' && typeof define.amd === 'object' && define.amd) {\n\t\t// register as 'classnames', consistent with npm package name\n\t\tdefine('classnames', [], function () {\n\t\t\treturn classNames;\n\t\t});\n\t} else {\n\t\twindow.classNames = classNames;\n\t}\n}());\n","/** @jsx h */\n\nimport type { JSX } from 'preact';\nimport { h, Component } from 'preact';\nimport { renderTemplate, isEqual } from '../../lib/utils';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\nimport type { Templates } from '../../types';\n\nconst defaultProps = {\n data: {},\n rootTagName: 'div',\n useCustomCompileOptions: {},\n templates: {},\n templatesConfig: {},\n};\n\ntype TemplateProps = {\n data?: Record;\n rootProps?: Record;\n rootTagName?: keyof JSX.IntrinsicElements;\n templateKey: string;\n bindEvent?: (...args: any[]) => string;\n} & PreparedTemplateProps &\n Readonly;\n\n// @TODO: Template should be a generic and receive TData to pass to Templates (to avoid TTemplateData to be set as `any`)\nclass Template extends Component {\n public static readonly defaultProps = defaultProps;\n\n public shouldComponentUpdate(nextProps: TemplateProps) {\n return (\n !isEqual(this.props.data, nextProps.data) ||\n this.props.templateKey !== nextProps.templateKey ||\n !isEqual(this.props.rootProps, nextProps.rootProps)\n );\n }\n\n public render() {\n const RootTagName = this.props.rootTagName;\n\n const useCustomCompileOptions =\n this.props.useCustomCompileOptions[this.props.templateKey];\n const compileOptions = useCustomCompileOptions\n ? this.props.templatesConfig.compileOptions\n : {};\n\n const content = renderTemplate({\n templates: this.props.templates,\n templateKey: this.props.templateKey,\n compileOptions,\n helpers: this.props.templatesConfig.helpers,\n data: this.props.data,\n bindEvent: this.props.bindEvent,\n });\n\n if (content === null) {\n // Adds a noscript to the DOM but virtual DOM is null\n // See http://facebook.github.io/react/docs/component-specs.html#render\n return null;\n }\n\n return (\n \n );\n }\n}\n\nexport default Template;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport cx from 'classnames';\nimport Template from '../Template/Template';\nimport type {\n BreadcrumbCSSClasses,\n BreadcrumbTemplates,\n} from '../../widgets/breadcrumb/breadcrumb';\nimport type { ComponentCSSClasses } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\nimport type { BreadcrumbConnectorParamsItem } from '../../connectors/breadcrumb/connectBreadcrumb';\n\nexport type BreadcrumbComponentCSSClasses =\n ComponentCSSClasses;\n\nexport type BreadcrumbComponentTemplates = Required;\n\nexport type BreadcrumbProps = {\n items: BreadcrumbConnectorParamsItem[];\n cssClasses: BreadcrumbComponentCSSClasses;\n templateProps: PreparedTemplateProps;\n createURL(value?: string | null): string;\n refine(value?: string | null): void;\n canRefine?: boolean;\n};\n\nconst Breadcrumb = ({\n items,\n cssClasses,\n templateProps,\n createURL,\n refine,\n}: BreadcrumbProps) => (\n \n
    \n \n {\n event.preventDefault();\n refine(undefined);\n },\n }}\n />\n \n\n {items.map((item, idx) => {\n const isLast = idx === items.length - 1;\n\n return (\n \n \n {isLast ? (\n item.label\n ) : (\n {\n event.preventDefault();\n refine(item.value);\n }}\n >\n {item.label}\n \n )}\n \n );\n })}\n
\n \n);\n\nexport default Breadcrumb;\n","import type { BreadcrumbComponentTemplates } from '../../components/Breadcrumb/Breadcrumb';\n\nconst defaultTemplates: BreadcrumbComponentTemplates = {\n home: 'Home',\n separator: '>',\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport type {\n BreadcrumbComponentCSSClasses,\n BreadcrumbComponentTemplates,\n} from '../../components/Breadcrumb/Breadcrumb';\nimport Breadcrumb from '../../components/Breadcrumb/Breadcrumb';\nimport type {\n BreadcrumbWidgetDescription,\n BreadcrumbConnectorParams,\n BreadcrumbRenderState,\n} from '../../connectors/breadcrumb/connectBreadcrumb';\nimport connectBreadcrumb from '../../connectors/breadcrumb/connectBreadcrumb';\nimport defaultTemplates from './defaultTemplates';\nimport {\n getContainerNode,\n prepareTemplateProps,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type { WidgetFactory, Template, Renderer } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nconst withUsage = createDocumentationMessageGenerator({ name: 'breadcrumb' });\nconst suit = component('Breadcrumb');\n\nconst renderer =\n ({\n containerNode,\n cssClasses,\n renderState,\n templates,\n }: {\n containerNode: HTMLElement;\n cssClasses: BreadcrumbComponentCSSClasses;\n renderState: {\n templateProps?: PreparedTemplateProps;\n };\n templates: BreadcrumbTemplates;\n }): Renderer> =>\n (\n { canRefine, createURL, instantSearchInstance, items, refine },\n isFirstRendering\n ) => {\n if (isFirstRendering) {\n renderState.templateProps = prepareTemplateProps({\n defaultTemplates,\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n\n return;\n }\n\n render(\n ,\n containerNode\n );\n };\n\nexport type BreadcrumbCSSClasses = Partial<{\n /**\n * CSS class to add to the root element of the widget.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the root element of the widget if there are no refinements.\n */\n noRefinementRoot: string | string[];\n\n /**\n * CSS class to add to the list element.\n */\n list: string | string[];\n\n /**\n * CSS class to add to the items of the list. The items contains the link and the separator.\n */\n item: string | string[];\n\n /**\n * CSS class to add to the selected item in the list: the last one or the home if there are no refinements.\n */\n selectedItem: string | string[];\n\n /**\n * CSS class to add to the separator.\n */\n separator: string | string[];\n\n /**\n * CSS class to add to the links in the items.\n */\n link: string | string[];\n}>;\n\nexport type BreadcrumbTemplates = Partial<{\n /**\n * Label of the breadcrumb's first element.\n */\n home: Template;\n\n /**\n * Symbol used to separate the elements of the breadcrumb.\n */\n separator: Template;\n}>;\n\nexport type BreadcrumbWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * Templates to use for the widget.\n */\n templates?: BreadcrumbTemplates;\n\n /**\n * CSS classes to add to the wrapping elements.\n */\n cssClasses?: BreadcrumbCSSClasses;\n};\n\nexport type BreadcrumbWidget = WidgetFactory<\n BreadcrumbWidgetDescription & { $$widgetType: 'ais.breadcrumb' },\n BreadcrumbConnectorParams,\n BreadcrumbWidgetParams\n>;\n\nconst breadcrumb: BreadcrumbWidget = function breadcrumb(widgetParams) {\n const {\n container,\n attributes,\n separator,\n rootPath,\n transformItems,\n templates = {},\n cssClasses: userCssClasses = {},\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n noRefinementRoot: cx(\n suit({ modifierName: 'noRefinement' }),\n userCssClasses.noRefinementRoot\n ),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n selectedItem: cx(\n suit({ descendantName: 'item', modifierName: 'selected' }),\n userCssClasses.selectedItem\n ),\n separator: cx(\n suit({ descendantName: 'separator' }),\n userCssClasses.separator\n ),\n link: cx(suit({ descendantName: 'link' }), userCssClasses.link),\n };\n\n const specializedRenderer = renderer({\n containerNode,\n cssClasses,\n renderState: {},\n templates,\n });\n\n const makeWidget = connectBreadcrumb(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({ attributes, separator, rootPath, transformItems }),\n $$widgetType: 'ais.breadcrumb',\n };\n};\n\nexport default breadcrumb;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport cx from 'classnames';\nimport Template from '../Template/Template';\nimport type { ClearRefinementsRenderState } from '../../connectors/clear-refinements/connectClearRefinements';\nimport type {\n ClearRefinementsCSSClasses,\n ClearRefinementsTemplates,\n} from '../../widgets/clear-refinements/clear-refinements';\nimport type { ComponentCSSClasses } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nexport type ClearRefinementsComponentCSSClasses =\n ComponentCSSClasses;\n\nexport type ClearRefinementsComponentTemplates =\n Required;\n\nexport type ClearRefinementsProps = {\n refine: ClearRefinementsRenderState['refine'];\n cssClasses: ClearRefinementsComponentCSSClasses;\n hasRefinements: ClearRefinementsRenderState['hasRefinements'];\n templateProps: PreparedTemplateProps;\n};\n\nconst ClearRefinements = ({\n hasRefinements,\n refine,\n cssClasses,\n templateProps,\n}: ClearRefinementsProps) => (\n
\n \n
\n);\n\nexport default ClearRefinements;\n","import type { ClearRefinementsComponentTemplates } from '../../components/ClearRefinements/ClearRefinements';\n\nconst defaultTemplates: ClearRefinementsComponentTemplates = {\n resetLabel: 'Clear refinements',\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport type {\n ClearRefinementsComponentCSSClasses,\n ClearRefinementsComponentTemplates,\n} from '../../components/ClearRefinements/ClearRefinements';\nimport ClearRefinements from '../../components/ClearRefinements/ClearRefinements';\nimport cx from 'classnames';\nimport type {\n ClearRefinementsConnectorParams,\n ClearRefinementsRenderState,\n ClearRefinementsWidgetDescription,\n} from '../../connectors/clear-refinements/connectClearRefinements';\nimport connectClearRefinements from '../../connectors/clear-refinements/connectClearRefinements';\nimport defaultTemplates from './defaultTemplates';\nimport {\n getContainerNode,\n prepareTemplateProps,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type { WidgetFactory, Template, Renderer } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'clear-refinements',\n});\nconst suit = component('ClearRefinements');\n\nconst renderer =\n ({\n containerNode,\n cssClasses,\n renderState,\n templates,\n }: {\n containerNode: HTMLElement;\n cssClasses: ClearRefinementsComponentCSSClasses;\n renderState: {\n templateProps?: PreparedTemplateProps;\n };\n templates: ClearRefinementsTemplates;\n }): Renderer<\n ClearRefinementsRenderState,\n Partial\n > =>\n ({ refine, hasRefinements, instantSearchInstance }, isFirstRendering) => {\n if (isFirstRendering) {\n renderState.templateProps = prepareTemplateProps({\n defaultTemplates,\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n return;\n }\n\n render(\n ,\n containerNode\n );\n };\n\nexport type ClearRefinementsCSSClasses = Partial<{\n /**\n * CSS class to add to the wrapper element.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the button of the widget.\n */\n button: string | string[];\n\n /**\n * CSS class to add to the button when there are no refinements.\n */\n disabledButton: string | string[];\n}>;\n\nexport type ClearRefinementsTemplates = Partial<{\n /**\n * Template for the content of the button\n */\n resetLabel: Template;\n}>;\n\nexport type ClearRefinementsWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * Templates to use for the widget.\n */\n templates?: ClearRefinementsTemplates;\n\n /**\n * CSS classes to be added.\n */\n cssClasses?: ClearRefinementsCSSClasses;\n};\n\nexport type ClearRefinementsWidget = WidgetFactory<\n ClearRefinementsWidgetDescription & { $$widgetType: 'ais.clearRefinements' },\n ClearRefinementsConnectorParams,\n ClearRefinementsWidgetParams\n>;\n\nconst clearRefinements: ClearRefinementsWidget = (widgetParams) => {\n const {\n container,\n templates = {},\n includedAttributes,\n excludedAttributes,\n transformItems,\n cssClasses: userCssClasses = {},\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n button: cx(suit({ descendantName: 'button' }), userCssClasses.button),\n disabledButton: cx(\n suit({ descendantName: 'button', modifierName: 'disabled' }),\n userCssClasses.disabledButton\n ),\n };\n\n const specializedRenderer = renderer({\n containerNode,\n cssClasses,\n renderState: {},\n templates,\n });\n\n const makeWidget = connectClearRefinements(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({\n includedAttributes,\n excludedAttributes,\n transformItems,\n }),\n $$widgetType: 'ais.clearRefinements',\n };\n};\n\nexport default clearRefinements;\n","import type {\n ConfigureConnectorParams,\n ConfigureWidgetDescription,\n} from '../../connectors/configure/connectConfigure';\nimport connectConfigure from '../../connectors/configure/connectConfigure';\nimport type { Widget } from '../../types';\nimport { noop } from '../../lib/utils';\n\n/**\n * A list of [search parameters](https://www.algolia.com/doc/api-reference/search-api-parameters/)\n * to enable when the widget mounts.\n */\nexport type ConfigureWidgetParams =\n ConfigureConnectorParams['searchParameters'];\n\nexport type ConfigureWidget = (widgetParams: ConfigureWidgetParams) => Widget<\n ConfigureWidgetDescription & {\n $$widgetType: 'ais.configure';\n widgetParams: ConfigureConnectorParams;\n }\n>;\n\nconst configure: ConfigureWidget = function configure(widgetParams) {\n // This is a renderless widget that falls back to the connector's\n // noop render and unmount functions.\n const makeWidget = connectConfigure(noop);\n\n return {\n ...makeWidget({ searchParameters: widgetParams }),\n $$widgetType: 'ais.configure',\n };\n};\n\nexport default configure;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport { isSpecialClick, capitalize } from '../../lib/utils';\nimport type {\n CurrentRefinementsConnectorParamsItem,\n CurrentRefinementsConnectorParamsRefinement,\n} from '../../connectors/current-refinements/connectCurrentRefinements';\nimport type { CurrentRefinementsCSSClasses } from '../../widgets/current-refinements/current-refinements';\nimport type { ComponentCSSClasses } from '../../types';\n\nexport type CurrentRefinementsComponentCSSClasses =\n ComponentCSSClasses;\n\nexport type CurrentRefinementsProps = {\n items: CurrentRefinementsConnectorParamsItem[];\n cssClasses: CurrentRefinementsComponentCSSClasses;\n};\n\nconst createItemKey = ({\n attribute,\n value,\n type,\n operator,\n}: CurrentRefinementsConnectorParamsRefinement): string =>\n [attribute, type, value, operator]\n .map((key) => key)\n .filter(Boolean)\n .join(':');\n\nconst handleClick = (callback: () => void) => (event: any) => {\n if (isSpecialClick(event)) {\n return;\n }\n\n event.preventDefault();\n callback();\n};\n\nconst CurrentRefinements = ({ items, cssClasses }: CurrentRefinementsProps) => (\n
\n
    \n {items.map((item, index) => (\n \n {capitalize(item.label)}:\n\n {item.refinements.map((refinement) => (\n \n \n {refinement.attribute === 'query' ? (\n {refinement.label}\n ) : (\n refinement.label\n )}\n \n\n \n ✕\n \n \n ))}\n \n ))}\n
\n
\n);\n\nexport default CurrentRefinements;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport CurrentRefinements from '../../components/CurrentRefinements/CurrentRefinements';\nimport type {\n CurrentRefinementsConnectorParams,\n CurrentRefinementsRenderState,\n CurrentRefinementsWidgetDescription,\n} from '../../connectors/current-refinements/connectCurrentRefinements';\nimport connectCurrentRefinements from '../../connectors/current-refinements/connectCurrentRefinements';\nimport {\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type { ComponentCSSClasses, Renderer, WidgetFactory } from '../../types';\n\nexport type CurrentRefinementsCSSClasses = Partial<{\n /**\n * CSS class to add to the root element.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the list element.\n */\n list: string | string[];\n\n /**\n * CSS class to add to the each item element.\n */\n item: string | string[];\n\n /**\n * CSS class to add to the label element.\n */\n label: string | string[];\n\n /**\n * CSS class to add to the category element.\n */\n category: string | string[];\n\n /**\n * CSS class to add to the categoryLabel element.\n */\n categoryLabel: string | string[];\n\n /**\n * CSS class to add to the delete element.\n */\n delete: string | string[];\n}>;\n\nexport type CurrentRefinementsWidgetParams = {\n /**\n * The CSS Selector or `HTMLElement` to insert the widget into.\n */\n container: string | HTMLElement;\n\n /**\n * The CSS classes to override.\n */\n cssClasses?: CurrentRefinementsCSSClasses;\n};\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'current-refinements',\n});\nconst suit = component('CurrentRefinements');\n\nconst renderer: Renderer<\n CurrentRefinementsRenderState,\n Partial\n> = ({ items, widgetParams }, isFirstRender) => {\n if (isFirstRender) {\n return;\n }\n\n const { container, cssClasses } = widgetParams as {\n container: HTMLElement;\n cssClasses: ComponentCSSClasses;\n };\n\n render(\n ,\n container\n );\n};\n\nexport type CurrentRefinementsWidget = WidgetFactory<\n CurrentRefinementsWidgetDescription & {\n $$widgetType: 'ais.currentRefinements';\n },\n CurrentRefinementsConnectorParams,\n CurrentRefinementsWidgetParams\n>;\n\nconst currentRefinements: CurrentRefinementsWidget =\n function currentRefinements(widgetParams) {\n const {\n container,\n includedAttributes,\n excludedAttributes,\n cssClasses: userCssClasses = {},\n transformItems,\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n label: cx(suit({ descendantName: 'label' }), userCssClasses.label),\n category: cx(\n suit({ descendantName: 'category' }),\n userCssClasses.category\n ),\n categoryLabel: cx(\n suit({ descendantName: 'categoryLabel' }),\n userCssClasses.categoryLabel\n ),\n delete: cx(suit({ descendantName: 'delete' }), userCssClasses.delete),\n };\n\n const makeWidget =\n connectCurrentRefinements(renderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({\n container: containerNode,\n cssClasses,\n includedAttributes,\n excludedAttributes,\n transformItems,\n }),\n $$widgetType: 'ais.currentRefinements',\n };\n };\n\nexport default currentRefinements;\n","import type { AnswersComponentTemplates } from '../../components/Answers/Answers';\n\nconst defaultTemplates: AnswersComponentTemplates = {\n header: '',\n loader: '',\n item: (item) => JSON.stringify(item),\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport cx from 'classnames';\nimport Template from '../Template/Template';\nimport type {\n AnswersCSSClasses,\n AnswersTemplates,\n} from '../../widgets/answers/answers';\nimport type { ComponentCSSClasses, Hits } from '../../types';\n\nexport type AnswersComponentCSSClasses = ComponentCSSClasses;\n\nexport type AnswersComponentTemplates = Required;\n\nexport type AnswersProps = {\n hits: Hits;\n isLoading: boolean;\n cssClasses: AnswersComponentCSSClasses;\n templateProps: {\n [key: string]: any;\n templates: AnswersComponentTemplates;\n };\n};\n\nconst Answers = ({\n hits,\n isLoading,\n cssClasses,\n templateProps,\n}: AnswersProps) => (\n \n \n {isLoading ? (\n \n ) : (\n
    \n {hits.map((hit, position) => (\n \n ))}\n
\n )}\n \n);\n\nexport default Answers;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport type { WidgetFactory, Template, Hit, Renderer } from '../../types';\nimport defaultTemplates from './defaultTemplates';\nimport {\n createDocumentationMessageGenerator,\n getContainerNode,\n prepareTemplateProps,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type {\n AnswersComponentCSSClasses,\n AnswersComponentTemplates,\n} from '../../components/Answers/Answers';\nimport Answers from '../../components/Answers/Answers';\nimport type {\n AnswersRenderState,\n AnswersConnectorParams,\n AnswersWidgetDescription,\n} from '../../connectors/answers/connectAnswers';\nimport connectAnswers from '../../connectors/answers/connectAnswers';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nconst withUsage = createDocumentationMessageGenerator({ name: 'answers' });\nconst suit = component('Answers');\n\nconst renderer =\n ({\n containerNode,\n cssClasses,\n renderState,\n templates,\n }: {\n containerNode: HTMLElement;\n cssClasses: AnswersComponentCSSClasses;\n renderState: {\n templateProps?: PreparedTemplateProps;\n };\n templates: AnswersTemplates;\n }): Renderer> =>\n ({ hits, isLoading, instantSearchInstance }, isFirstRendering) => {\n if (isFirstRendering) {\n renderState.templateProps = prepareTemplateProps({\n defaultTemplates,\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n return;\n }\n\n render(\n ,\n containerNode\n );\n };\n\nexport type AnswersTemplates = Partial<{\n /**\n * Template to use for the header. This template will receive an object containing `hits` and `isLoading`.\n */\n header: Template<{\n hits: Hit[];\n isLoading: boolean;\n }>;\n\n /**\n * Template to use for the loader.\n */\n loader: Template;\n\n /**\n * Template to use for each result. This template will receive an object containing a single record.\n */\n item: Template;\n}>;\n\nexport type AnswersCSSClasses = Partial<{\n /**\n * CSS class to add to the root element of the widget.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the wrapping element when no results.\n */\n emptyRoot: string | string[];\n\n /**\n * CSS classes to add to the header.\n */\n header: string | string[];\n\n /**\n * CSS classes to add to the loader.\n */\n loader: string | string[];\n\n /**\n * CSS class to add to the list of results.\n */\n list: string | string[];\n\n /**\n * CSS class to add to each result.\n */\n item: string | string[];\n}>;\n\nexport type AnswersWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * The templates to use for the widget.\n */\n templates?: AnswersTemplates;\n\n /**\n * The CSS classes to override.\n */\n cssClasses?: AnswersCSSClasses;\n};\n\nexport type AnswersWidget = WidgetFactory<\n AnswersWidgetDescription & { $$widgetType: 'ais.answers' },\n AnswersConnectorParams,\n AnswersWidgetParams\n>;\n\nconst answersWidget: AnswersWidget = (widgetParams) => {\n const {\n container,\n attributesForPrediction,\n queryLanguages,\n nbHits,\n searchDebounceTime,\n renderDebounceTime,\n escapeHTML,\n extraParameters,\n templates = {},\n cssClasses: userCssClasses = {},\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n emptyRoot: cx(suit({ modifierName: 'empty' }), userCssClasses.emptyRoot),\n header: cx(suit({ descendantName: 'header' }), userCssClasses.header),\n loader: cx(suit({ descendantName: 'loader' }), userCssClasses.loader),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n };\n\n const specializedRenderer = renderer({\n containerNode,\n cssClasses,\n templates,\n renderState: {},\n });\n\n const makeWidget = connectAnswers(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({\n attributesForPrediction,\n queryLanguages,\n nbHits,\n searchDebounceTime,\n renderDebounceTime,\n escapeHTML,\n extraParameters,\n }),\n $$widgetType: 'ais.answers',\n };\n};\n\nexport default answersWidget;\n","import type { PlainSearchParameters } from 'algoliasearch-helper';\nimport { noop } from '../../lib/utils';\nimport type {\n ConfigureRelatedItemsConnectorParams,\n ConfigureRelatedItemsWidgetDescription,\n} from '../../connectors/configure-related-items/connectConfigureRelatedItems';\nimport connectConfigureRelatedItems from '../../connectors/configure-related-items/connectConfigureRelatedItems';\nimport type { WidgetFactory } from '../../types';\n\nexport type ConfigureRelatedItemsWidget = WidgetFactory<\n ConfigureRelatedItemsWidgetDescription & {\n $$widgetType: 'ais.configureRelatedItems';\n },\n ConfigureRelatedItemsConnectorParams,\n ConfigureRelatedItemsWidgetParams\n>;\n\nexport type ConfigureRelatedItemsWidgetParams = PlainSearchParameters;\n\nconst configureRelatedItems: ConfigureRelatedItemsWidget =\n function configureRelatedItems(widgetParams) {\n const makeWidget = connectConfigureRelatedItems(noop);\n\n return {\n ...makeWidget(widgetParams),\n $$widgetType: 'ais.configureRelatedItems',\n };\n };\n\nexport default configureRelatedItems;\n","import type {\n DynamicWidgetsConnectorParams,\n DynamicWidgetsWidgetDescription,\n} from '../../connectors/dynamic-widgets/connectDynamicWidgets';\nimport connectDynamicWidgets from '../../connectors/dynamic-widgets/connectDynamicWidgets';\nimport { component } from '../../lib/suit';\nimport {\n createDocumentationMessageGenerator,\n getContainerNode,\n getWidgetAttribute,\n} from '../../lib/utils';\nimport type { Widget, WidgetFactory } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'dynamic-widgets',\n});\nconst suit = component('DynamicWidgets');\n\nexport type DynamicWidgetsWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * An array of widget creator functions, displayed in the order defined by\n * `facetOrdering`.\n */\n widgets: Array<(container: HTMLElement) => Widget>;\n\n /**\n * Function to return a fallback widget when an attribute isn't found in\n * `widgets`.\n */\n fallbackWidget?(args: {\n /** The attribute name to create a widget for. */\n attribute: string;\n /** CSS Selector or HTMLElement to insert the widget */\n container: HTMLElement;\n }): Widget;\n};\n\nexport type DynamicWidgetsWidget = WidgetFactory<\n DynamicWidgetsWidgetDescription & { $$widgetType: 'ais.dynamicWidgets' },\n Omit,\n DynamicWidgetsWidgetParams\n>;\n\nfunction createContainer(rootContainer: HTMLElement) {\n const container = document.createElement('div');\n container.className = suit({ descendantName: 'widget' });\n\n rootContainer.appendChild(container);\n\n return container;\n}\n\nconst dynamicWidgets: DynamicWidgetsWidget = function dynamicWidgets(\n widgetParams\n) {\n const {\n container: containerSelector,\n widgets,\n fallbackWidget,\n ...otherWidgetParams\n } = widgetParams || {};\n\n if (!containerSelector) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n if (\n !(\n widgets &&\n Array.isArray(widgets) &&\n widgets.every((widget) => typeof widget === 'function')\n )\n ) {\n throw new Error(\n withUsage('The `widgets` option expects an array of callbacks.')\n );\n }\n\n const userContainer = getContainerNode(containerSelector);\n const rootContainer = document.createElement('div');\n rootContainer.className = suit();\n\n const containers = new Map();\n const connectorWidgets: Widget[] = [];\n\n const makeWidget = connectDynamicWidgets(\n ({ attributesToRender }, isFirstRender) => {\n if (isFirstRender) {\n userContainer.appendChild(rootContainer);\n }\n\n attributesToRender.forEach((attribute) => {\n if (!containers.has(attribute)) {\n return;\n }\n const container = containers.get(attribute)!;\n rootContainer.appendChild(container);\n });\n },\n () => {\n userContainer.removeChild(rootContainer);\n }\n );\n\n const widget = makeWidget({\n ...otherWidgetParams,\n widgets: connectorWidgets,\n fallbackWidget:\n typeof fallbackWidget === 'function'\n ? ({ attribute }) => {\n const container = createContainer(rootContainer);\n containers.set(attribute, container);\n return fallbackWidget({ attribute, container });\n }\n : undefined,\n });\n\n return {\n ...widget,\n init(initOptions) {\n widgets.forEach((cb) => {\n const container = createContainer(rootContainer);\n\n const childWidget = cb(container);\n const attribute = getWidgetAttribute(childWidget, initOptions);\n\n containers.set(attribute, container);\n connectorWidgets.push(childWidget);\n });\n\n widget.init!(initOptions);\n },\n $$widgetType: 'ais.dynamicWidgets',\n };\n};\nexport default dynamicWidgets;\n","/** @jsx h */\n\nimport type { ComponentChildren } from 'preact';\nimport { h } from 'preact';\n\ntype Props = {\n className: string;\n onClick(event: MouseEvent): void;\n children: ComponentChildren;\n disabled?: boolean;\n};\n\nconst GeoSearchButton = ({\n className,\n disabled = false,\n onClick,\n children,\n}: Props) => (\n \n);\n\nexport default GeoSearchButton;\n","/** @jsx h */\n\nimport type { ComponentChildren } from 'preact';\nimport { h } from 'preact';\n\ntype Props = {\n classNameLabel: string;\n classNameInput: string;\n checked: boolean;\n onToggle(event: Event): void;\n children: ComponentChildren;\n};\n\nconst GeoSearchToggle = ({\n classNameLabel,\n classNameInput,\n checked,\n onToggle,\n children,\n}: Props) => (\n \n);\n\nexport default GeoSearchToggle;\n","/** @jsx h */\n\nimport { h, Fragment } from 'preact';\nimport cx from 'classnames';\nimport Template from '../Template/Template';\nimport GeoSearchButton from './GeoSearchButton';\nimport GeoSearchToggle from './GeoSearchToggle';\nimport type {\n GeoSearchCSSClasses,\n GeoSearchTemplates,\n} from '../../widgets/geo-search/geo-search';\nimport type { ComponentCSSClasses } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\ntype Props = {\n cssClasses: ComponentCSSClasses;\n enableRefine: boolean;\n enableRefineControl: boolean;\n enableClearMapRefinement: boolean;\n isRefineOnMapMove: boolean;\n isRefinedWithMap: boolean;\n hasMapMoveSinceLastRefine: boolean;\n onRefineToggle(event: Event): void;\n onRefineClick(event: MouseEvent): void;\n onClearClick(event: MouseEvent): void;\n templateProps: PreparedTemplateProps;\n};\n\nconst GeoSearchControls = ({\n cssClasses,\n enableRefine,\n enableRefineControl,\n enableClearMapRefinement,\n isRefineOnMapMove,\n isRefinedWithMap,\n hasMapMoveSinceLastRefine,\n onRefineToggle,\n onRefineClick,\n onClearClick,\n templateProps,\n}: Props) => (\n \n {enableRefine && (\n
\n {enableRefineControl && (\n
\n {isRefineOnMapMove || !hasMapMoveSinceLastRefine ? (\n \n \n \n ) : (\n \n \n \n )}\n
\n )}\n\n {!enableRefineControl && !isRefineOnMapMove && (\n
\n \n \n \n
\n )}\n\n {enableClearMapRefinement && isRefinedWithMap && (\n \n \n \n )}\n
\n )}\n
\n);\n\nexport default GeoSearchControls;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport { prepareTemplateProps } from '../../lib/utils';\nimport GeoSearchControls from '../../components/GeoSearchControls/GeoSearchControls';\n\nconst refineWithMap = ({ refine, mapInstance }) =>\n refine({\n northEast: mapInstance.getBounds().getNorthEast().toJSON(),\n southWest: mapInstance.getBounds().getSouthWest().toJSON(),\n });\n\nconst collectMarkersForNextRender = (markers, nextIds) =>\n markers.reduce(\n ([update, exit], marker) => {\n const persist = nextIds.includes(marker.__id);\n\n return persist\n ? [update.concat(marker), exit]\n : [update, exit.concat(marker)];\n },\n [[], []]\n );\n\nconst createBoundingBoxFromMarkers = (google, markers) => {\n const latLngBounds = markers.reduce(\n (acc, marker) => acc.extend(marker.getPosition()),\n new google.maps.LatLngBounds()\n );\n\n return {\n northEast: latLngBounds.getNorthEast().toJSON(),\n southWest: latLngBounds.getSouthWest().toJSON(),\n };\n};\n\nconst lockUserInteraction = (renderState, functionThatAltersTheMapPosition) => {\n renderState.isUserInteraction = false;\n functionThatAltersTheMapPosition();\n renderState.isUserInteraction = true;\n};\n\nconst renderer = (\n {\n items,\n position,\n currentRefinement,\n refine,\n clearMapRefinement,\n toggleRefineOnMapMove,\n isRefineOnMapMove,\n setMapMoveSinceLastRefine,\n hasMapMoveSinceLastRefine,\n isRefinedWithMap,\n widgetParams,\n instantSearchInstance,\n },\n isFirstRendering\n) => {\n const {\n container,\n googleReference,\n cssClasses,\n templates,\n initialZoom,\n initialPosition,\n enableRefine,\n enableClearMapRefinement,\n enableRefineControl,\n mapOptions,\n createMarker,\n markerOptions,\n renderState,\n } = widgetParams;\n\n if (isFirstRendering) {\n renderState.isUserInteraction = true;\n renderState.isPendingRefine = false;\n renderState.markers = [];\n\n const rootElement = document.createElement('div');\n rootElement.className = cssClasses.root;\n container.appendChild(rootElement);\n\n const mapElement = document.createElement('div');\n mapElement.className = cssClasses.map;\n rootElement.appendChild(mapElement);\n\n const treeElement = document.createElement('div');\n treeElement.className = cssClasses.tree;\n rootElement.appendChild(treeElement);\n\n renderState.mapInstance = new googleReference.maps.Map(mapElement, {\n mapTypeControl: false,\n fullscreenControl: false,\n streetViewControl: false,\n clickableIcons: false,\n zoomControlOptions: {\n position: googleReference.maps.ControlPosition.LEFT_TOP,\n },\n ...mapOptions,\n });\n\n const setupListenersWhenMapIsReady = () => {\n const onChange = () => {\n if (renderState.isUserInteraction && enableRefine) {\n setMapMoveSinceLastRefine();\n\n if (isRefineOnMapMove()) {\n renderState.isPendingRefine = true;\n }\n }\n };\n\n renderState.mapInstance.addListener('center_changed', onChange);\n renderState.mapInstance.addListener('zoom_changed', onChange);\n renderState.mapInstance.addListener('dragstart', onChange);\n\n renderState.mapInstance.addListener('idle', () => {\n if (renderState.isUserInteraction && renderState.isPendingRefine) {\n renderState.isPendingRefine = false;\n\n refineWithMap({\n mapInstance: renderState.mapInstance,\n refine,\n });\n }\n });\n };\n\n googleReference.maps.event.addListenerOnce(\n renderState.mapInstance,\n 'idle',\n setupListenersWhenMapIsReady\n );\n\n renderState.templateProps = prepareTemplateProps({\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n\n return;\n }\n\n // Collect markers that need to be updated or removed\n const nextItemsIds = items.map((_) => _.objectID);\n const [updateMarkers, exitMarkers] = collectMarkersForNextRender(\n renderState.markers,\n nextItemsIds\n );\n\n // Collect items that will be added\n const updateMarkerIds = updateMarkers.map((_) => _.__id);\n const nextPendingItems = items.filter(\n (item) => !updateMarkerIds.includes(item.objectID)\n );\n\n // Remove all markers that need to be removed\n exitMarkers.forEach((marker) => marker.setMap(null));\n\n // Create the markers from the items\n renderState.markers = updateMarkers.concat(\n nextPendingItems.map((item) => {\n const marker = createMarker({\n map: renderState.mapInstance,\n item,\n });\n\n Object.keys(markerOptions.events).forEach((eventName) => {\n marker.addListener(eventName, (event) => {\n markerOptions.events[eventName]({\n map: renderState.mapInstance,\n event,\n item,\n marker,\n });\n });\n });\n\n return marker;\n })\n );\n\n const shouldUpdate = !hasMapMoveSinceLastRefine();\n\n // We use this value for differentiate the padding to apply during\n // fitBounds. When we don't have a currenRefinement (boundingBox)\n // we let Google Maps compute the automatic padding. But when we\n // provide the currentRefinement we explicitly set the padding\n // to `0` otherwise the map will decrease the zoom on each refine.\n const boundingBoxPadding = currentRefinement ? 0 : null;\n const boundingBox =\n !currentRefinement && Boolean(renderState.markers.length)\n ? createBoundingBoxFromMarkers(googleReference, renderState.markers)\n : currentRefinement;\n\n if (boundingBox && shouldUpdate) {\n lockUserInteraction(renderState, () => {\n renderState.mapInstance.fitBounds(\n new googleReference.maps.LatLngBounds(\n boundingBox.southWest,\n boundingBox.northEast\n ),\n boundingBoxPadding\n );\n });\n } else if (shouldUpdate) {\n lockUserInteraction(renderState, () => {\n renderState.mapInstance.setCenter(position || initialPosition);\n renderState.mapInstance.setZoom(initialZoom);\n });\n }\n\n render(\n \n refineWithMap({\n mapInstance: renderState.mapInstance,\n refine,\n })\n }\n onClearClick={clearMapRefinement}\n templateProps={renderState.templateProps}\n />,\n container.querySelector(`.${cssClasses.tree}`)\n );\n};\n\nexport default renderer;\n","import type { GeoSearchComponentTemplates } from './geo-search';\n\nconst defaultTemplates: GeoSearchComponentTemplates = {\n HTMLMarker: '

Your custom HTML Marker

',\n reset: 'Clear the map refinement',\n toggle: 'Search as I move the map',\n redo: 'Redo search here',\n};\n\nexport default defaultTemplates;\n","/* global google EventListener */\n\nexport type HTMLMarkerArguments = {\n __id: string;\n position: google.maps.LatLngLiteral;\n map: google.maps.Map;\n template: string;\n title?: string;\n className: string;\n anchor?: { x: number; y: number };\n};\n\ninterface Marker {\n __id: string;\n anchor: { x: number; y: number };\n offset?: { x: number; y: number };\n listeners: { [key: string]: EventListener };\n latLng: google.maps.LatLng;\n element: HTMLDivElement;\n getPosition(): google.maps.LatLng;\n}\n\nconst createHTMLMarker = (\n googleReference: typeof google\n): new (args: HTMLMarkerArguments) => google.maps.OverlayView & Marker => {\n class HTMLMarker extends googleReference.maps.OverlayView {\n public __id: string;\n public anchor: {\n x: number;\n y: number;\n };\n public offset?: {\n x: number;\n y: number;\n };\n public listeners: { [key: string]: EventListener };\n public latLng: google.maps.LatLng;\n public element: HTMLDivElement;\n\n public constructor({\n __id,\n position,\n map,\n template,\n className,\n anchor = {\n x: 0,\n y: 0,\n },\n }: HTMLMarkerArguments) {\n super();\n\n this.__id = __id;\n this.anchor = anchor;\n this.listeners = {};\n this.latLng = new googleReference.maps.LatLng(position);\n\n this.element = document.createElement('div');\n this.element.className = className;\n this.element.style.position = 'absolute';\n this.element.innerHTML = template;\n\n this.setMap(map);\n }\n\n public onAdd() {\n // Append the element to the map\n this.getPanes()!.overlayMouseTarget.appendChild(this.element);\n\n // Compute the offset onAdd & cache it because afterwards\n // it won't retrieve the correct values, we also avoid\n // to read the values on every draw\n const bbBox = this.element.getBoundingClientRect();\n\n this.offset = {\n x: this.anchor.x + bbBox.width / 2,\n y: this.anchor.y + bbBox.height,\n };\n\n // Force the width of the element will avoid the\n // content to collapse when we move the map\n this.element.style.width = `${bbBox.width}px`;\n }\n\n public draw() {\n const position = this.getProjection().fromLatLngToDivPixel(this.latLng)!;\n\n this.element.style.left = `${Math.round(position.x - this.offset!.x)}px`;\n this.element.style.top = `${Math.round(position.y - this.offset!.y)}px`;\n\n // Markers to the south are in front of markers to the north\n // This is the default behaviour of Google Maps\n this.element.style.zIndex = String(parseInt(this.element.style.top, 10));\n }\n\n public onRemove() {\n if (this.element) {\n this.element.parentNode!.removeChild(this.element);\n\n Object.keys(this.listeners).forEach((eventName) => {\n this.element.removeEventListener(\n eventName,\n this.listeners[eventName]\n );\n });\n\n // after onRemove the class is no longer used, thus it can be deleted\n // @ts-expect-error\n delete this.element;\n // @ts-expect-error\n delete this.listeners;\n }\n }\n\n public addListener(eventName: string, listener: EventListener) {\n this.listeners[eventName] = listener;\n\n const element = this.element;\n\n element.addEventListener(eventName, listener);\n\n return {\n remove() {\n return element.removeEventListener(eventName, listener);\n },\n };\n }\n\n public getPosition() {\n return this.latLng;\n }\n }\n\n return HTMLMarker;\n};\n\nexport default createHTMLMarker;\n","// global for TypeScript alone\n/* global google */\nimport cx from 'classnames';\nimport { render } from 'preact';\nimport {\n getContainerNode,\n renderTemplate,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type {\n GeoSearchConnectorParams,\n GeoSearchWidgetDescription,\n GeoHit,\n} from '../../connectors/geo-search/connectGeoSearch';\nimport connectGeoSearch from '../../connectors/geo-search/connectGeoSearch';\nimport renderer from './GeoSearchRenderer';\nimport defaultTemplates from './defaultTemplates';\nimport type { HTMLMarkerArguments } from './createHTMLMarker';\nimport createHTMLMarker from './createHTMLMarker';\nimport type { GeoLoc, Template, WidgetFactory } from '../../types';\n\nexport type CreateMarker = (args: {\n item: GeoHit;\n map: google.maps.Map;\n}) => google.maps.OverlayView | google.maps.Marker;\n\nconst withUsage = createDocumentationMessageGenerator({ name: 'geo-search' });\nconst suit = component('GeoSearch');\n\nexport type GeoSearchTemplates = Partial<{\n /** Template to use for the marker. */\n HTMLMarker: Template;\n /** Template for the reset button. */\n reset: Template;\n /** Template for the toggle label. */\n toggle: Template;\n /** Template for the redo button. */\n redo: Template;\n}>;\n\nexport type GeoSearchComponentTemplates = Required;\n\nexport type GeoSearchCSSClasses = Partial<{\n /** The root div of the widget. */\n root: string | string[];\n /** The map container of the widget. */\n map: string | string[];\n /** The control element of the widget. */\n control: string | string[];\n /** The label of the control element. */\n label: string | string[];\n /** The selected label of the control element. */\n selectedLabel: string | string[];\n /** The input of the control element. */\n input: string | string[];\n /** The redo search button. */\n redo: string | string[];\n /** The disabled redo search button. */\n disabledRedo: string | string[];\n /** The reset refinement button. */\n reset: string | string[];\n}>;\n\nexport type GeoSearchMarker = {\n /**\n * Function used to create the options passed to the Google Maps marker.\n * See the documentation for more information:\n * https://developers.google.com/maps/documentation/javascript/reference/3/#MarkerOptions\n */\n createOptions?(item: GeoHit): TOptions;\n /**\n * Object that takes an event type (ex: `click`, `mouseover`) as key and a\n * listener as value. The listener is provided with an object that contains:\n * `event`, `item`, `marker`, `map`.\n */\n events: {\n [key: string]: (event: {\n item: any;\n marker: any;\n map: any;\n event: any;\n }) => void;\n };\n};\n\nexport type GeoSearchWidgetParams = {\n /**\n * By default the map will set the zoom accordingly to the markers displayed on it.\n * When we refine it may happen that the results are empty. For those situations we\n * need to provide a zoom to render the map.\n * @default 1\n */\n initialZoom?: number;\n /**\n * By default the map will set the position accordingly to the markers displayed on it.\n * When we refine it may happen that the results are empty. For those situations we need\n * to provide a position to render the map. This option is ignored when the `position`\n * is provided.\n * @default { lat: 0, lng: 0 }\n */\n initialPosition?: GeoLoc;\n /** Templates to use for the widget. */\n templates?: GeoSearchTemplates;\n /** CSS classes to add to the wrapping elements. */\n cssClasses?: GeoSearchCSSClasses;\n /**\n * Options for customize the built-in Google Maps marker. This option is\n * ignored when the `customHTMLMarker` is provided.\n */\n builtInMarker?: Partial>;\n /**\n * Options to customize the HTML marker. We provide an alternative to the\n * built-in Google Maps marker in order to have a full control of the marker\n * rendering. You can use plain HTML to build your marker.\n */\n customHTMLMarker?:\n | Partial>>\n | boolean;\n /**\n * If true, the map is used to search - otherwise it's for display purposes only.\n * @default true\n */\n enableRefine?: boolean;\n /**\n * If true, a button is displayed on the map when the refinement is coming from\n * the map in order to remove it.\n * @default true\n */\n enableClearMapRefinement?: boolean;\n /**\n * If true, the user can toggle the option `enableRefineOnMapMove` directly from the map.\n * @default true\n */\n enableRefineControl?: boolean;\n /**\n * Option forwarded to the Google Maps constructor.\n * See the documentation for more information:\n * https://developers.google.com/maps/documentation/javascript/reference/3/#MapOptions\n */\n mapOptions?: google.maps.MapOptions;\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n /**\n * Reference to the global `window.google` object.\n * See [the documentation](https://developers.google.com/maps/documentation/javascript/tutorial) for more information.\n */\n googleReference: typeof window['google'];\n};\n\nexport type GeoSearchWidget = WidgetFactory<\n GeoSearchWidgetDescription & { $$widgetType: 'ais.geoSearch' },\n GeoSearchConnectorParams,\n GeoSearchWidgetParams\n>;\n\n/**\n * The **GeoSearch** widget displays the list of results from the search on a Google Maps. It also provides a way to search for results based on their position. The widget also provide some of the common GeoSearch patterns like search on map interaction.\n *\n * @requirements\n *\n * Note that the GeoSearch widget uses the [geosearch](https://www.algolia.com/doc/guides/searching/geo-search) capabilities of Algolia. Your hits **must** have a `_geoloc` attribute in order to be displayed on the map.\n *\n * Currently, the feature is not compatible with multiple values in the _geoloc attribute.\n *\n * You are also responsible for loading the Google Maps library, it's not shipped with InstantSearch. You need to load the Google Maps library and pass a reference to the widget. You can find more information about how to install the library in [the Google Maps documentation](https://developers.google.com/maps/documentation/javascript/tutorial).\n *\n * Don't forget to explicitly set the `height` of the map container (default class `.ais-geo-search--map`), otherwise it won't be shown (it's a requirement of Google Maps).\n */\nconst geoSearch: GeoSearchWidget = (widgetParams) => {\n const {\n initialZoom = 1,\n initialPosition = { lat: 0, lng: 0 },\n templates: userTemplates = {},\n cssClasses: userCssClasses = {},\n builtInMarker: userBuiltInMarker = {},\n customHTMLMarker: userCustomHTMLMarker,\n enableRefine = true,\n enableClearMapRefinement = true,\n enableRefineControl = true,\n container,\n googleReference,\n ...otherWidgetParams\n } = widgetParams || {};\n\n const defaultBuiltInMarker: GeoSearchMarker = {\n createOptions: () => ({}),\n events: {},\n };\n\n const defaultCustomHTMLMarker: GeoSearchMarker> =\n {\n createOptions: () => ({}),\n events: {},\n };\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n if (!googleReference) {\n throw new Error(withUsage('The `googleReference` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n // Required only to mount / unmount the Preact tree\n tree: suit({ descendantName: 'tree' }),\n map: cx(suit({ descendantName: 'map' }), userCssClasses.map),\n control: cx(suit({ descendantName: 'control' }), userCssClasses.control),\n label: cx(suit({ descendantName: 'label' }), userCssClasses.label),\n selectedLabel: cx(\n suit({ descendantName: 'label', modifierName: 'selected' }),\n userCssClasses.selectedLabel\n ),\n input: cx(suit({ descendantName: 'input' }), userCssClasses.input),\n redo: cx(suit({ descendantName: 'redo' }), userCssClasses.redo),\n disabledRedo: cx(\n suit({ descendantName: 'redo', modifierName: 'disabled' }),\n userCssClasses.disabledRedo\n ),\n reset: cx(suit({ descendantName: 'reset' }), userCssClasses.reset),\n };\n\n const templates = {\n ...defaultTemplates,\n ...userTemplates,\n };\n\n const builtInMarker = {\n ...defaultBuiltInMarker,\n ...userBuiltInMarker,\n };\n\n const isCustomHTMLMarker =\n Boolean(userCustomHTMLMarker) || Boolean(userTemplates.HTMLMarker);\n\n const customHTMLMarker = isCustomHTMLMarker && {\n ...defaultCustomHTMLMarker,\n ...(userCustomHTMLMarker as Partial<\n GeoSearchMarker>\n >),\n };\n\n const createBuiltInMarker: CreateMarker = ({ item, ...rest }) =>\n new googleReference.maps.Marker({\n ...builtInMarker.createOptions!(item),\n ...rest,\n // @ts-expect-error @types/googlemaps doesn't document this\n __id: item.objectID,\n position: item._geoloc,\n });\n\n const HTMLMarker = createHTMLMarker(googleReference);\n const createCustomHTMLMarker: CreateMarker = ({ item, ...rest }) =>\n new HTMLMarker({\n // this is only called when customHTMLMarker is defined\n ...(customHTMLMarker as GeoSearchMarker>)\n .createOptions!(item),\n ...rest,\n __id: item.objectID,\n position: item._geoloc,\n className: cx(suit({ descendantName: 'marker' })),\n template: renderTemplate({\n templateKey: 'HTMLMarker',\n templates,\n data: item,\n }),\n });\n\n const createMarker = !customHTMLMarker\n ? createBuiltInMarker\n : createCustomHTMLMarker;\n\n const markerOptions = !customHTMLMarker ? builtInMarker : customHTMLMarker;\n\n const makeWidget = connectGeoSearch(renderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({\n ...otherWidgetParams,\n renderState: {},\n container: containerNode,\n googleReference,\n initialZoom,\n initialPosition,\n templates,\n cssClasses,\n createMarker,\n markerOptions,\n enableRefine,\n enableClearMapRefinement,\n enableRefineControl,\n }),\n $$widgetType: 'ais.geoSearch',\n };\n};\n\nexport default geoSearch;\n","/** @jsx h */\n\nimport type { JSX } from 'preact';\nimport { h } from 'preact';\nimport Template from '../Template/Template';\n\nexport type RefinementListItemProps = {\n facetValueToRefine: string;\n handleClick: (args: {\n facetValueToRefine: string;\n isRefined: boolean;\n originalEvent: MouseEvent;\n }) => void;\n isRefined: boolean;\n subItems?: JSX.Element;\n templateData: Record;\n templateKey: string;\n templateProps?: Record;\n className: string;\n};\n\nfunction RefinementListItem({\n className,\n handleClick,\n facetValueToRefine,\n isRefined,\n templateProps,\n templateKey,\n templateData,\n subItems,\n}: RefinementListItemProps) {\n return (\n {\n handleClick({\n facetValueToRefine,\n isRefined,\n originalEvent,\n });\n }}\n >\n \n {subItems}\n \n );\n}\n\nexport default RefinementListItem;\n","/** @jsx h */\n\nimport { h, createRef, Component } from 'preact';\nimport { noop } from '../../lib/utils';\nimport Template from '../Template/Template';\nimport type {\n SearchBoxCSSClasses,\n SearchBoxTemplates,\n} from '../../widgets/search-box/search-box';\nimport type { ComponentCSSClasses } from '../../types';\n\nexport type SearchBoxComponentCSSClasses =\n ComponentCSSClasses;\n\nexport type SearchBoxComponentTemplates = Required;\n\ntype SearchBoxProps = {\n placeholder?: string;\n cssClasses: SearchBoxComponentCSSClasses;\n templates: SearchBoxComponentTemplates;\n query?: string;\n showSubmit?: boolean;\n showReset?: boolean;\n showLoadingIndicator?: boolean;\n refine?: (value: string) => void;\n autofocus?: boolean;\n searchAsYouType?: boolean;\n isSearchStalled?: boolean;\n disabled?: boolean;\n onChange?: (event: Event) => void;\n onSubmit?: (event: Event) => void;\n onReset?: (event: Event) => void;\n};\n\nconst defaultProps = {\n query: '',\n showSubmit: true,\n showReset: true,\n showLoadingIndicator: true,\n autofocus: false,\n searchAsYouType: true,\n isSearchStalled: false,\n disabled: false,\n onChange: noop,\n onSubmit: noop,\n onReset: noop,\n refine: noop,\n};\n\ntype SearchBoxPropsWithDefaultProps = SearchBoxProps &\n Readonly;\n\ntype SearchBoxState = {\n query: string;\n focused: boolean;\n};\n\nclass SearchBox extends Component<\n SearchBoxPropsWithDefaultProps,\n SearchBoxState\n> {\n public static defaultProps = defaultProps;\n\n public state = {\n query: this.props.query,\n focused: false,\n };\n\n private input = createRef();\n\n /**\n * This public method is used in the RefinementList SFFV search box\n * to reset the input state when an item is selected.\n *\n * @see RefinementList#componentWillReceiveProps\n * @return {undefined}\n */\n public resetInput() {\n this.setState({ query: '' });\n }\n\n private onInput = (event: Event) => {\n const { searchAsYouType, refine, onChange } = this.props;\n const query = (event.target as HTMLInputElement).value;\n\n if (searchAsYouType) {\n refine(query);\n }\n this.setState({ query });\n\n onChange(event);\n };\n\n public componentWillReceiveProps(nextProps: SearchBoxPropsWithDefaultProps) {\n /**\n * when the user is typing, we don't want to replace the query typed\n * by the user (state.query) with the query exposed by the connector (props.query)\n * see: https://github.com/algolia/instantsearch.js/issues/4141\n */\n if (!this.state.focused && nextProps.query !== this.state.query) {\n this.setState({ query: nextProps.query });\n }\n }\n\n private onSubmit = (event: Event) => {\n const { searchAsYouType, refine, onSubmit } = this.props;\n\n event.preventDefault();\n event.stopPropagation();\n if (this.input.current) {\n this.input.current.blur();\n }\n\n if (!searchAsYouType) {\n refine(this.state.query);\n }\n\n onSubmit(event);\n\n return false;\n };\n\n private onReset = (event: Event) => {\n const { refine, onReset } = this.props;\n const query = '';\n\n if (this.input.current) {\n this.input.current.focus();\n }\n\n refine(query);\n this.setState({ query });\n\n onReset(event);\n };\n\n private onBlur = () => {\n this.setState({ focused: false });\n };\n\n private onFocus = () => {\n this.setState({ focused: true });\n };\n\n public render() {\n const {\n cssClasses,\n placeholder,\n autofocus,\n showSubmit,\n showReset,\n showLoadingIndicator,\n templates,\n isSearchStalled,\n } = this.props;\n\n return (\n
\n \n \n\n \n\n \n\n {showLoadingIndicator && (\n \n )}\n \n
\n );\n }\n}\n\nexport default SearchBox;\n","/** @jsx h */\n\nimport type { JSX } from 'preact';\nimport { h, createRef, Component } from 'preact';\nimport cx from 'classnames';\nimport { isSpecialClick, isEqual } from '../../lib/utils';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\nimport Template from '../Template/Template';\nimport RefinementListItem from './RefinementListItem';\nimport type {\n SearchBoxComponentCSSClasses,\n SearchBoxComponentTemplates,\n} from '../SearchBox/SearchBox';\nimport SearchBox from '../SearchBox/SearchBox';\nimport type { HierarchicalMenuItem } from '../../connectors/hierarchical-menu/connectHierarchicalMenu';\nimport type { ComponentCSSClasses, CreateURL, Templates } from '../../types';\nimport type { RefinementListOwnCSSClasses } from '../../widgets/refinement-list/refinement-list';\nimport type { RatingMenuComponentCSSClasses } from '../../widgets/rating-menu/rating-menu';\nimport type { HierarchicalMenuComponentCSSClasses } from '../../widgets/hierarchical-menu/hierarchical-menu';\n\n// CSS types\ntype RefinementListOptionalClasses =\n | 'noResults'\n | 'checkbox'\n | 'labelText'\n | 'showMore'\n | 'disabledShowMore'\n | 'searchBox'\n | 'count';\n\ntype RefinementListWidgetCSSClasses =\n ComponentCSSClasses;\n\ntype RefinementListRequiredCSSClasses = Omit<\n RefinementListWidgetCSSClasses,\n RefinementListOptionalClasses\n> &\n Partial>;\n\nexport type RefinementListComponentCSSClasses =\n RefinementListRequiredCSSClasses & {\n searchable?: SearchBoxComponentCSSClasses;\n } & Partial> &\n Partial<\n Pick\n >;\n\ntype FacetValue = {\n value: string;\n label: string;\n highlighted?: string;\n count?: number;\n isRefined: boolean;\n data?: HierarchicalMenuItem[] | null;\n};\n\nexport type RefinementListProps = {\n createURL: CreateURL;\n cssClasses: RefinementListComponentCSSClasses;\n depth?: number;\n facetValues?: FacetValue[];\n attribute?: string;\n templateProps: PreparedTemplateProps;\n toggleRefinement: (value: string) => void;\n showMore?: boolean;\n toggleShowMore?: () => void;\n isShowingMore?: boolean;\n hasExhaustiveItems?: boolean;\n canToggleShowMore?: boolean;\n className?: string;\n children?: JSX.Element;\n\n // searchable props are optional, but will definitely be present in a searchable context\n isFromSearch?: boolean;\n searchIsAlwaysActive?: boolean;\n searchFacetValues?: (query: string) => void;\n searchPlaceholder?: string;\n searchBoxTemplateProps?: PreparedTemplateProps;\n};\n\nconst defaultProps = {\n cssClasses: {},\n depth: 0,\n};\n\ntype RefinementListPropsWithDefaultProps =\n RefinementListProps & Readonly;\n\ntype RefinementListItemTemplateData =\n FacetValue & {\n url: string;\n } & Pick<\n RefinementListProps,\n 'attribute' | 'cssClasses' | 'isFromSearch'\n >;\n\nfunction isHierarchicalMenuItem(\n facetValue: FacetValue\n): facetValue is HierarchicalMenuItem {\n return (facetValue as HierarchicalMenuItem).data !== undefined;\n}\n\nclass RefinementList extends Component<\n RefinementListPropsWithDefaultProps\n> {\n public static defaultProps = defaultProps;\n\n private searchBox = createRef();\n\n public constructor(props: RefinementListPropsWithDefaultProps) {\n super(props);\n this.handleItemClick = this.handleItemClick.bind(this);\n }\n\n public shouldComponentUpdate(\n nextProps: RefinementListPropsWithDefaultProps\n ) {\n const areFacetValuesDifferent = !isEqual(\n this.props.facetValues,\n nextProps.facetValues\n );\n\n return areFacetValuesDifferent;\n }\n\n private refine(facetValueToRefine: string) {\n this.props.toggleRefinement(facetValueToRefine);\n }\n\n private _generateFacetItem(facetValue: FacetValue) {\n let subItems;\n if (\n isHierarchicalMenuItem(facetValue) &&\n Array.isArray(facetValue.data) &&\n facetValue.data.length > 0\n ) {\n const { root, ...cssClasses } = this.props.cssClasses;\n\n subItems = (\n \n );\n }\n\n const url = this.props.createURL(facetValue.value);\n const templateData: RefinementListItemTemplateData = {\n ...facetValue,\n url,\n attribute: this.props.attribute,\n cssClasses: this.props.cssClasses,\n isFromSearch: this.props.isFromSearch,\n };\n\n let { value: key } = facetValue;\n if (facetValue.isRefined !== undefined) {\n key += `/${facetValue.isRefined}`;\n }\n\n if (facetValue.count !== undefined) {\n key += `/${facetValue.count}`;\n }\n\n const refinementListItemClassName = cx(this.props.cssClasses.item, {\n [this.props.cssClasses.selectedItem]: facetValue.isRefined,\n // cx allows `undefined` as a key but typescript doesn't\n [this.props.cssClasses.disabledItem!]: !facetValue.count,\n [this.props.cssClasses.parentItem!]:\n isHierarchicalMenuItem(facetValue) &&\n Array.isArray(facetValue.data) &&\n facetValue.data.length > 0,\n });\n\n return (\n \n );\n }\n\n // Click events on DOM tree like LABEL > INPUT will result in two click events\n // instead of one.\n // No matter the framework, see https://www.google.com/search?q=click+label+twice\n //\n // Thus making it hard to distinguish activation from deactivation because both click events\n // are very close. Debounce is a solution but hacky.\n //\n // So the code here checks if the click was done on or in a LABEL. If this LABEL\n // has a checkbox inside, we ignore the first click event because we will get another one.\n //\n // We also check if the click was done inside a link and then e.preventDefault() because we already\n // handle the url\n //\n // Finally, we always stop propagation of the event to avoid multiple levels RefinementLists to fail: click\n // on child would click on parent also\n private handleItemClick({\n facetValueToRefine,\n isRefined,\n originalEvent,\n }: {\n facetValueToRefine: string;\n isRefined: boolean;\n originalEvent: MouseEvent;\n }) {\n if (isSpecialClick(originalEvent)) {\n // do not alter the default browser behavior\n // if one special key is down\n return;\n }\n\n if (\n !(originalEvent.target instanceof HTMLElement) ||\n !(originalEvent.target.parentNode instanceof HTMLElement)\n ) {\n return;\n }\n\n if (\n isRefined &&\n originalEvent.target.parentNode.querySelector(\n 'input[type=\"radio\"]:checked'\n )\n ) {\n // Prevent refinement for being reset if the user clicks on an already checked radio button\n return;\n }\n\n if (originalEvent.target.tagName === 'INPUT') {\n this.refine(facetValueToRefine);\n return;\n }\n\n let parent = originalEvent.target;\n\n while (parent !== originalEvent.currentTarget) {\n if (\n parent.tagName === 'LABEL' &&\n (parent.querySelector('input[type=\"checkbox\"]') ||\n parent.querySelector('input[type=\"radio\"]'))\n ) {\n return;\n }\n\n if (parent.tagName === 'A' && (parent as HTMLAnchorElement).href) {\n originalEvent.preventDefault();\n }\n\n parent = parent.parentNode as HTMLElement;\n }\n\n originalEvent.stopPropagation();\n\n this.refine(facetValueToRefine);\n }\n\n public componentWillReceiveProps(\n nextProps: RefinementListPropsWithDefaultProps\n ) {\n if (this.searchBox.current && !nextProps.isFromSearch) {\n this.searchBox.current.resetInput();\n }\n }\n\n private refineFirstValue() {\n const firstValue = this.props.facetValues && this.props.facetValues[0];\n if (firstValue) {\n const actualValue = firstValue.value;\n this.props.toggleRefinement(actualValue);\n }\n }\n\n public render() {\n const showMoreButtonClassName = cx(this.props.cssClasses.showMore, {\n [this.props.cssClasses.disabledShowMore!]: !(\n this.props.showMore === true && this.props.canToggleShowMore\n ),\n });\n\n const showMoreButton = this.props.showMore === true && (\n \n );\n\n const shouldDisableSearchBox =\n this.props.searchIsAlwaysActive !== true &&\n !(this.props.isFromSearch || !this.props.hasExhaustiveItems);\n\n const searchBox = this.props.searchFacetValues && (\n
\n \n this.props.searchFacetValues!(\n (event.target as HTMLInputElement).value\n )\n }\n onReset={() => this.props.searchFacetValues!('')}\n onSubmit={() => this.refineFirstValue()}\n // This sets the search box to a controlled state because\n // we don't rely on the `refine` prop but on `onChange`.\n searchAsYouType={false}\n />\n
\n );\n\n const facetValues = this.props.facetValues &&\n this.props.facetValues.length > 0 && (\n
    \n {this.props.facetValues.map(this._generateFacetItem, this)}\n
\n );\n\n const noResults = this.props.searchFacetValues &&\n this.props.isFromSearch &&\n (!this.props.facetValues || this.props.facetValues.length === 0) && (\n \n );\n\n const rootClassName = cx(\n this.props.cssClasses.root,\n {\n [this.props.cssClasses.noRefinementRoot]:\n !this.props.facetValues || this.props.facetValues.length === 0,\n },\n this.props.className\n );\n\n return (\n
\n {this.props.children}\n {searchBox}\n {facetValues}\n {noResults}\n {showMoreButton}\n
\n );\n }\n}\n\nexport default RefinementList;\n","import type { HierarchicalMenuComponentTemplates } from './hierarchical-menu';\nconst defaultTemplates: HierarchicalMenuComponentTemplates = {\n item:\n '' +\n '{{label}}' +\n '{{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}}' +\n '',\n showMoreText: `\n {{#isShowingMore}}\n Show less\n {{/isShowingMore}}\n {{^isShowingMore}}\n Show more\n {{/isShowingMore}}\n `,\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport RefinementList from '../../components/RefinementList/RefinementList';\nimport type {\n HierarchicalMenuItem,\n HierarchicalMenuConnectorParams,\n HierarchicalMenuRenderState,\n HierarchicalMenuWidgetDescription,\n} from '../../connectors/hierarchical-menu/connectHierarchicalMenu';\nimport connectHierarchicalMenu from '../../connectors/hierarchical-menu/connectHierarchicalMenu';\nimport defaultTemplates from './defaultTemplates';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\nimport {\n prepareTemplateProps,\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport type {\n TransformItems,\n Template,\n WidgetFactory,\n RendererOptions,\n SortBy,\n ComponentCSSClasses,\n} from '../../types';\nimport { component } from '../../lib/suit';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'hierarchical-menu',\n});\nconst suit = component('HierarchicalMenu');\n\ntype HierarchicalMenuTemplates = Partial<{\n /**\n * Item template, provided with `name`, `count`, `isRefined`, `url` data properties.\n */\n item: Template<{\n name: string;\n count: number;\n isRefined: boolean;\n url: string;\n }>;\n /**\n * Template used for the show more text, provided with `isShowingMore` data property.\n */\n showMoreText: Template<{ isShowingMore: boolean }>;\n}>;\n\nexport type HierarchicalMenuCSSClasses = Partial<{\n /**\n * CSS class to add to the root element.\n */\n root: string | string[];\n /**\n * CSS class to add to the root element when no refinements.\n */\n noRefinementRoot: string | string[];\n /**\n * CSS class to add to the list element.\n */\n list: string | string[];\n /**\n * CSS class to add to the child list element.\n */\n childList: string | string[];\n /**\n * CSS class to add to each item element.\n */\n item: string | string[];\n /**\n * CSS class to add to each selected item element.\n */\n selectedItem: string | string[];\n /**\n * CSS class to add to each parent item element.\n */\n parentItem: string | string[];\n /**\n * CSS class to add to each link (when using the default template).\n */\n link: string | string[];\n /**\n * CSS class to add to each label (when using the default template).\n */\n label: string | string[];\n /**\n * CSS class to add to each count element (when using the default template).\n */\n count: string | string[];\n /**\n * CSS class to add to the show more element.\n */\n showMore: string | string[];\n /**\n * CSS class to add to the disabled show more element.\n */\n disabledShowMore: string | string[];\n}>;\n\nexport type HierarchicalMenuComponentCSSClasses =\n ComponentCSSClasses;\n\nexport type HierarchicalMenuComponentTemplates =\n Required;\n\nexport type HierarchicalMenuWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n /**\n * Array of attributes to use to generate the hierarchy of the menu.\n */\n attributes: string[];\n /**\n * Separator used in the attributes to separate level values.\n */\n separator?: string;\n /**\n * Prefix path to use if the first level is not the root level.\n */\n rootPath?: string;\n /**\n * Show the siblings of the selected parent level of the current refined value.\n *\n * With `showParentLevel` set to `true` (default):\n * - Parent lvl0\n * - **lvl1**\n * - **lvl2**\n * - lvl2\n * - lvl2\n * - lvl 1\n * - lvl 1\n * - Parent lvl0\n * - Parent lvl0\n *\n * With `showParentLevel` set to `false`:\n * - Parent lvl0\n * - **lvl1**\n * - **lvl2**\n * - Parent lvl0\n * - Parent lvl0\n */\n showParentLevel?: boolean;\n /**\n * Max number of values to display.\n */\n limit?: number;\n /**\n * Whether to display the \"show more\" button.\n */\n showMore?: boolean;\n /**\n * Max number of values to display when showing more.\n * does not impact the root level.\n */\n showMoreLimit?: number;\n /**\n * How to sort refinements. Possible values: `count|isRefined|name:asc|name:desc`.\n * You can also use a sort function that behaves like the standard Javascript [compareFunction](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#Syntax).\n */\n sortBy?: SortBy;\n /**\n * Function to transform the items passed to the templates.\n */\n transformItems?: TransformItems;\n /**\n * Templates to use for the widget.\n */\n templates?: HierarchicalMenuTemplates;\n /**\n * CSS classes to add to the wrapping elements.\n */\n cssClasses?: HierarchicalMenuCSSClasses;\n};\n\nconst renderer =\n ({\n cssClasses,\n containerNode,\n showMore,\n templates,\n renderState,\n }: {\n cssClasses: HierarchicalMenuComponentCSSClasses;\n containerNode: HTMLElement;\n showMore: boolean;\n templates: HierarchicalMenuTemplates;\n renderState: {\n templateProps?: PreparedTemplateProps;\n };\n }) =>\n (\n {\n createURL,\n items,\n refine,\n instantSearchInstance,\n isShowingMore,\n toggleShowMore,\n canToggleShowMore,\n }: HierarchicalMenuRenderState &\n RendererOptions,\n isFirstRendering: boolean\n ) => {\n if (isFirstRendering) {\n renderState.templateProps = prepareTemplateProps({\n defaultTemplates,\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n return;\n }\n\n render(\n ,\n containerNode\n );\n };\n\n/**\n * The hierarchical menu widget is used to create a navigation based on a hierarchy of facet attributes.\n *\n * It is commonly used for categories with subcategories.\n *\n * All attributes (lvl0, lvl1 here) must be declared as [attributes for faceting](https://www.algolia.com/doc/guides/searching/faceting/#declaring-attributes-for-faceting) in your\n * Algolia settings.\n *\n * By default, the separator we expect is ` > ` (with spaces) but you can use\n * a different one by using the `separator` option.\n * @requirements\n * Your objects must be formatted in a specific way to be\n * able to display hierarchical menus. Here's an example:\n *\n * ```javascript\n * {\n * \"objectID\": \"123\",\n * \"name\": \"orange\",\n * \"categories\": {\n * \"lvl0\": \"fruits\",\n * \"lvl1\": \"fruits > citrus\"\n * }\n * }\n * ```\n *\n * Every level must be specified entirely.\n * It's also possible to have multiple values per level, for example:\n *\n * ```javascript\n * {\n * \"objectID\": \"123\",\n * \"name\": \"orange\",\n * \"categories\": {\n * \"lvl0\": [\"fruits\", \"vitamins\"],\n * \"lvl1\": [\"fruits > citrus\", \"vitamins > C\"]\n * }\n * }\n * ```\n * @type {WidgetFactory}\n * @devNovel HierarchicalMenu\n * @category filter\n * @param {HierarchicalMenuWidgetParams} widgetParams The HierarchicalMenu widget options.\n * @return {Widget} A new HierarchicalMenu widget instance.\n * @example\n * search.addWidgets([\n * instantsearch.widgets.hierarchicalMenu({\n * container: '#hierarchical-categories',\n * attributes: ['hierarchicalCategories.lvl0', 'hierarchicalCategories.lvl1', 'hierarchicalCategories.lvl2'],\n * })\n * ]);\n */\nexport type HierarchicalMenuWidget = WidgetFactory<\n HierarchicalMenuWidgetDescription & { $$widgetType: 'ais.hierarchicalMenu' },\n HierarchicalMenuConnectorParams,\n HierarchicalMenuWidgetParams\n>;\n\nconst hierarchicalMenu: HierarchicalMenuWidget = function hierarchicalMenu(\n widgetParams\n) {\n const {\n container,\n attributes,\n separator,\n rootPath,\n showParentLevel,\n limit,\n showMore = false,\n showMoreLimit,\n sortBy,\n transformItems,\n templates = {},\n cssClasses: userCssClasses = {},\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n noRefinementRoot: cx(\n suit({ modifierName: 'noRefinement' }),\n userCssClasses.noRefinementRoot\n ),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n childList: cx(\n suit({ descendantName: 'list', modifierName: 'child' }),\n userCssClasses.childList\n ),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n selectedItem: cx(\n suit({ descendantName: 'item', modifierName: 'selected' }),\n userCssClasses.selectedItem\n ),\n parentItem: cx(\n suit({ descendantName: 'item', modifierName: 'parent' }),\n userCssClasses.parentItem\n ),\n link: cx(suit({ descendantName: 'link' }), userCssClasses.link),\n label: cx(suit({ descendantName: 'label' }), userCssClasses.label),\n count: cx(suit({ descendantName: 'count' }), userCssClasses.count),\n showMore: cx(suit({ descendantName: 'showMore' }), userCssClasses.showMore),\n disabledShowMore: cx(\n suit({ descendantName: 'showMore', modifierName: 'disabled' }),\n userCssClasses.disabledShowMore\n ),\n };\n\n const specializedRenderer = renderer({\n cssClasses,\n containerNode,\n templates,\n showMore,\n renderState: {},\n });\n\n const makeWidget = connectHierarchicalMenu(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({\n attributes,\n separator,\n rootPath,\n showParentLevel,\n limit,\n showMore,\n showMoreLimit,\n sortBy,\n transformItems,\n }),\n $$widgetType: 'ais.hierarchicalMenu',\n };\n};\n\nexport default hierarchicalMenu;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport cx from 'classnames';\nimport Template from '../Template/Template';\nimport type { SearchResults } from 'algoliasearch-helper';\nimport type { BindEventForHits, SendEventForHits } from '../../lib/utils';\nimport type { ComponentCSSClasses, Hits as HitsArray } from '../../types';\nimport type { HitsCSSClasses, HitsTemplates } from '../../widgets/hits/hits';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nexport type HitsComponentCSSClasses = ComponentCSSClasses;\nexport type HitsComponentTemplates = Required;\n\nexport type HitsProps = {\n results: SearchResults;\n hits: HitsArray;\n sendEvent?: SendEventForHits;\n bindEvent?: BindEventForHits;\n cssClasses: HitsComponentCSSClasses;\n templateProps: PreparedTemplateProps;\n};\n\nconst Hits = ({\n results,\n hits,\n bindEvent,\n cssClasses,\n templateProps,\n}: HitsProps) => {\n if (results.hits.length === 0) {\n return (\n \n );\n }\n\n return (\n
\n
    \n {hits.map((hit, index) => (\n \n ))}\n
\n
\n );\n};\n\nexport default Hits;\n","import type { HitsComponentTemplates } from '../../components/Hits/Hits';\n\nconst defaultTemplates: HitsComponentTemplates = {\n empty: 'No results',\n item(data) {\n return JSON.stringify(data, null, 2);\n },\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport type {\n HitsConnectorParams,\n HitsRenderState,\n HitsWidgetDescription,\n} from '../../connectors/hits/connectHits';\nimport connectHits from '../../connectors/hits/connectHits';\nimport type {\n HitsComponentCSSClasses,\n HitsComponentTemplates,\n} from '../../components/Hits/Hits';\nimport Hits from '../../components/Hits/Hits';\nimport defaultTemplates from './defaultTemplates';\nimport {\n prepareTemplateProps,\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport { withInsights, withInsightsListener } from '../../lib/insights';\nimport type {\n Template,\n TemplateWithBindEvent,\n Hit,\n WidgetFactory,\n Renderer,\n InsightsClient,\n} from '../../types';\nimport type { InsightsEvent } from '../../middlewares/createInsightsMiddleware';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nconst withUsage = createDocumentationMessageGenerator({ name: 'hits' });\nconst suit = component('Hits');\nconst HitsWithInsightsListener = withInsightsListener(Hits);\n\nconst renderer =\n ({\n renderState,\n cssClasses,\n containerNode,\n templates,\n }: {\n containerNode: HTMLElement;\n cssClasses: HitsComponentCSSClasses;\n renderState: {\n templateProps?: PreparedTemplateProps;\n };\n templates: HitsTemplates;\n }): Renderer> =>\n (\n { hits: receivedHits, results, instantSearchInstance, insights, bindEvent },\n isFirstRendering\n ) => {\n if (isFirstRendering) {\n renderState.templateProps = prepareTemplateProps({\n defaultTemplates,\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n return;\n }\n\n render(\n {\n instantSearchInstance.sendEventToInsights(event);\n }}\n bindEvent={bindEvent}\n />,\n containerNode\n );\n };\n\nexport type HitsCSSClasses = Partial<{\n /**\n * CSS class to add to the wrapping element.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the wrapping element when no results.\n */\n emptyRoot: string | string[];\n\n /**\n * CSS class to add to the list of results.\n */\n list: string | string[];\n\n /**\n * CSS class to add to each result.\n */\n item: string | string[];\n}>;\n\nexport type HitsTemplates = Partial<{\n /**\n * Template to use when there are no results.\n *\n * @default 'No Results'\n */\n empty: Template;\n\n /**\n * Template to use for each result. This template will receive an object containing a single record.\n *\n * @default ''\n */\n item: TemplateWithBindEvent<\n Hit & {\n // @deprecated the index in the hits array, use __position instead, which is the absolute position\n __hitIndex: number;\n }\n >;\n}>;\n\nexport type HitsWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * Templates to use for the widget.\n */\n templates?: HitsTemplates;\n\n /**\n * CSS classes to add.\n */\n cssClasses?: HitsCSSClasses;\n};\n\nexport type HitsWidget = WidgetFactory<\n HitsWidgetDescription & { $$widgetType: 'ais.hits' },\n HitsConnectorParams,\n HitsWidgetParams\n>;\n\nconst hits: HitsWidget = function hits(widgetParams) {\n const {\n container,\n escapeHTML,\n transformItems,\n templates = {},\n cssClasses: userCssClasses = {},\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n emptyRoot: cx(suit({ modifierName: 'empty' }), userCssClasses.emptyRoot),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n };\n\n const specializedRenderer = renderer({\n containerNode,\n cssClasses,\n renderState: {},\n templates,\n });\n\n const makeWidget = withInsights(connectHits)(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({ escapeHTML, transformItems }),\n $$widgetType: 'ais.hits',\n };\n};\n\nexport default hits;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport cx from 'classnames';\n\nexport type SelectorOption = {\n value?: string | number;\n label: string;\n};\n\nexport type SelectorComponentCSSClasses = {\n root: string;\n select: string;\n option: string;\n};\n\nexport type SelectorProps = {\n cssClasses: SelectorComponentCSSClasses;\n currentValue?: string | number;\n options: SelectorOption[];\n setValue(value: SelectorOption['value']): void;\n};\n\nfunction Selector({\n currentValue,\n options,\n cssClasses,\n setValue,\n}: SelectorProps) {\n return (\n setValue((event.target as HTMLSelectElement).value)}\n value={`${currentValue}`}\n >\n {options.map((option) => (\n \n {option.label}\n \n ))}\n \n );\n}\n\nexport default Selector;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport Selector from '../../components/Selector/Selector';\nimport type {\n HitsPerPageConnectorParams,\n HitsPerPageRenderState,\n HitsPerPageWidgetDescription,\n} from '../../connectors/hits-per-page/connectHitsPerPage';\nimport connectHitsPerPage from '../../connectors/hits-per-page/connectHitsPerPage';\nimport {\n getContainerNode,\n createDocumentationMessageGenerator,\n find,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type { ComponentCSSClasses, WidgetFactory } from '../../types';\n\nconst withUsage = createDocumentationMessageGenerator({\n name: 'hits-per-page',\n});\nconst suit = component('HitsPerPage');\n\nconst renderer =\n ({\n containerNode,\n cssClasses,\n }: {\n containerNode: HTMLElement;\n cssClasses: ComponentCSSClasses;\n }) =>\n ({ items, refine }: HitsPerPageRenderState, isFirstRendering: boolean) => {\n if (isFirstRendering) return;\n\n const { value: currentValue } =\n find(items, ({ isRefined }) => isRefined) || {};\n\n render(\n
\n \n
,\n containerNode\n );\n };\n\nexport type HitsPerPageCSSClasses = Partial<{\n /**\n * CSS classes added to the outer `
`.\n */\n root: string | string[];\n\n /**\n * CSS classes added to the parent `\n
\n );\n}\n\nexport default MenuSelect;\n","import type { MenuSelectComponentTemplates } from '../../components/MenuSelect/MenuSelect';\n\nconst defaultTemplates: MenuSelectComponentTemplates = {\n item: '{{label}} ({{#helpers.formatNumber}}{{count}}{{/helpers.formatNumber}})',\n defaultOption: 'See all',\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport type {\n MenuConnectorParams,\n MenuRenderState,\n MenuWidgetDescription,\n} from '../../connectors/menu/connectMenu';\nimport connectMenu from '../../connectors/menu/connectMenu';\nimport type {\n MenuSelectComponentCSSClasses,\n MenuSelectComponentTemplates,\n} from '../../components/MenuSelect/MenuSelect';\nimport MenuSelect from '../../components/MenuSelect/MenuSelect';\nimport defaultTemplates from './defaultTemplates';\nimport {\n prepareTemplateProps,\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type { RendererOptions, Template, WidgetFactory } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nconst withUsage = createDocumentationMessageGenerator({ name: 'menu-select' });\nconst suit = component('MenuSelect');\n\nexport type MenuSelectCSSClasses = Partial<{\n /**\n * CSS class to add to the root element.\n */\n root: string | string[];\n /**\n * CSS class to add to the root when there are no items to display\n */\n noRefinementRoot: string | string[];\n /**\n * CSS class to add to the select element.\n */\n select: string | string[];\n /**\n * CSS class to add to the option element.\n */\n option: string | string[];\n}>;\n\nexport type MenuSelectTemplates = Partial<{\n /**\n * Item template, provided with `label`, `count`, `isRefined` and `value` data properties.\n */\n item: Template<{\n label: string;\n value: string;\n count: number;\n isRefined: boolean;\n }>;\n /**\n * Label of the \"see all\" option in the select.\n */\n defaultOption: Template;\n}>;\n\nexport type MenuSelectWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n /**\n * Customize the output through templating.\n */\n templates?: MenuSelectTemplates;\n /**\n * CSS classes to add to the wrapping elements.\n */\n cssClasses?: MenuSelectCSSClasses;\n};\n\nconst renderer =\n ({\n containerNode,\n cssClasses,\n renderState,\n templates,\n }: {\n containerNode: HTMLElement;\n cssClasses: MenuSelectComponentCSSClasses;\n renderState: {\n templateProps?: PreparedTemplateProps;\n };\n templates: MenuSelectTemplates;\n }) =>\n (\n {\n refine,\n items,\n instantSearchInstance,\n }: MenuRenderState & RendererOptions,\n isFirstRendering: boolean\n ) => {\n if (isFirstRendering) {\n renderState.templateProps = prepareTemplateProps({\n defaultTemplates,\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n return;\n }\n\n render(\n ,\n containerNode\n );\n };\n\nexport type MenuSelectWidget = WidgetFactory<\n MenuWidgetDescription & { $$widgetType: 'ais.menuSelect' },\n MenuConnectorParams,\n MenuSelectWidgetParams\n>;\n\nconst menuSelect: MenuSelectWidget = function menuSelect(widgetParams) {\n const {\n container,\n attribute,\n sortBy = ['name:asc'],\n limit = 10,\n cssClasses: userCssClasses = {},\n templates = {},\n transformItems,\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n noRefinementRoot: cx(\n suit({ modifierName: 'noRefinement' }),\n userCssClasses.noRefinementRoot\n ),\n select: cx(suit({ descendantName: 'select' }), userCssClasses.select),\n option: cx(suit({ descendantName: 'option' }), userCssClasses.option),\n };\n\n const specializedRenderer = renderer({\n containerNode,\n cssClasses,\n renderState: {},\n templates,\n });\n\n const makeWidget = connectMenu(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({ attribute, limit, sortBy, transformItems }),\n $$widgetType: 'ais.menuSelect',\n };\n};\n\nexport default menuSelect;\n","import type { NumericMenuComponentTemplates } from './numeric-menu';\n\nconst defaultTemplates: NumericMenuComponentTemplates = {\n item: ``,\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport RefinementList from '../../components/RefinementList/RefinementList';\nimport type {\n NumericMenuConnectorParams,\n NumericMenuRenderState,\n NumericMenuWidgetDescription,\n} from '../../connectors/numeric-menu/connectNumericMenu';\nimport connectNumericMenu from '../../connectors/numeric-menu/connectNumericMenu';\nimport defaultTemplates from './defaultTemplates';\nimport {\n prepareTemplateProps,\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type {\n ComponentCSSClasses,\n Renderer,\n Template,\n WidgetFactory,\n} from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/utils/prepareTemplateProps';\n\nconst withUsage = createDocumentationMessageGenerator({ name: 'numeric-menu' });\nconst suit = component('NumericMenu');\n\nconst renderer =\n ({\n containerNode,\n attribute,\n cssClasses,\n renderState,\n templates,\n }: {\n containerNode: HTMLElement;\n attribute: string;\n cssClasses: NumericMenuComponentCSSClasses;\n renderState: {\n templateProps?: PreparedTemplateProps;\n };\n templates: NumericMenuTemplates;\n }): Renderer> =>\n (\n { createURL, instantSearchInstance, refine, items },\n isFirstRendering: boolean\n ) => {\n if (isFirstRendering) {\n renderState.templateProps = prepareTemplateProps({\n defaultTemplates,\n templatesConfig: instantSearchInstance.templatesConfig,\n templates,\n });\n return;\n }\n\n render(\n ,\n containerNode\n );\n };\n\nexport type NumericMenuCSSClasses = Partial<{\n /**\n * CSS class to add to the root element.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the root element when no refinements.\n */\n noRefinementRoot: string | string[];\n\n /**\n * CSS class to add to the list element.\n */\n list: string | string[];\n\n /**\n * CSS class to add to each item element.\n */\n item: string | string[];\n\n /**\n * CSS class to add to each selected item element.\n */\n selectedItem: string | string[];\n\n /**\n * CSS class to add to each label element.\n */\n label: string | string[];\n\n /**\n * CSS class to add to each label text element.\n */\n labelText: string | string[];\n\n /**\n * CSS class to add to each radio element (when using the default template).\n */\n radio: string | string[];\n}>;\n\nexport type NumericMenuComponentCSSClasses =\n ComponentCSSClasses;\n\nexport type NumericMenuTemplates = Partial<{\n /**\n * Item template, provided with `label` (the name in the configuration), `isRefined`, `url`, `value` (the setting for the filter) data properties.\n */\n item: Template<{\n /**\n * The name of the attribute.\n */\n attribute: string;\n\n /**\n * The label for the option.\n */\n label: string;\n\n /**\n * The encoded URL of the bounds object with a {start, end} form. This\n * value can be used verbatim in the webpage and can be read by refine\n * directly. If you want to inspect the value, you can do JSON.parse(window.decodeURI(value))\n * to get the object.\n */\n value: string;\n\n /**\n * Whether or not the refinement is selected.\n */\n isRefined: boolean;\n\n /**\n * The URL with the applied refinement.\n */\n url: string;\n\n /**\n * The CSS classes provided to the widget.\n */\n cssClasses: NumericMenuComponentCSSClasses;\n }>;\n}>;\n\nexport type NumericMenuComponentTemplates = Required;\n\nexport type NumericMenuWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * Templates to use for the widget.\n */\n templates?: NumericMenuTemplates;\n\n /**\n * CSS classes to add to the wrapping elements.\n */\n cssClasses?: NumericMenuCSSClasses;\n};\n\nexport type NumericMenuWidget = WidgetFactory<\n NumericMenuWidgetDescription & { $$widgetType: 'ais.numericMenu' },\n NumericMenuConnectorParams,\n NumericMenuWidgetParams\n>;\n\nconst numericMenu: NumericMenuWidget = function numericMenu(widgetParams) {\n const {\n container,\n attribute,\n items,\n cssClasses: userCssClasses = {},\n templates = {},\n transformItems,\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n noRefinementRoot: cx(\n suit({ modifierName: 'noRefinement' }),\n userCssClasses.noRefinementRoot\n ),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n selectedItem: cx(\n suit({ descendantName: 'item', modifierName: 'selected' }),\n userCssClasses.selectedItem\n ),\n label: cx(suit({ descendantName: 'label' }), userCssClasses.label),\n radio: cx(suit({ descendantName: 'radio' }), userCssClasses.radio),\n labelText: cx(\n suit({ descendantName: 'labelText' }),\n userCssClasses.labelText\n ),\n };\n\n const specializedRenderer = renderer({\n containerNode,\n attribute,\n cssClasses,\n renderState: {},\n templates,\n });\n\n const makeWidget = connectNumericMenu(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({\n attribute,\n items,\n transformItems,\n }),\n $$widgetType: 'ais.numericMenu',\n };\n};\n\nexport default numericMenu;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport cx from 'classnames';\n\nimport { isSpecialClick } from '../../lib/utils';\nimport type {\n PaginationCSSClasses,\n PaginationTemplates,\n} from '../../widgets/pagination/pagination';\nimport type { ComponentCSSClasses } from '../../types';\n\nexport type PaginationComponentCSSClasses =\n ComponentCSSClasses;\n\nexport type PaginationComponentTemplates = Required;\n\nexport type PaginationProps = {\n createURL(value: number): string;\n cssClasses: PaginationComponentCSSClasses;\n templates: PaginationComponentTemplates;\n currentPage: number;\n nbPages: number;\n pages: number[];\n isFirstPage: boolean;\n isLastPage: boolean;\n setCurrentPage(value: number): void;\n showFirst?: boolean;\n showLast?: boolean;\n showPrevious?: boolean;\n showNext?: boolean;\n};\n\nfunction Pagination(props: PaginationProps) {\n function createClickHandler(pageNumber: number) {\n return (event: MouseEvent) => {\n if (isSpecialClick(event)) {\n // do not alter the default browser behavior\n // if one special key is down\n return;\n }\n event.preventDefault();\n props.setCurrentPage(pageNumber);\n };\n }\n\n return (\n \n
    \n {props.showFirst && (\n \n )}\n\n {props.showPrevious && (\n \n )}\n\n {props.pages.map((pageNumber) => (\n \n ))}\n\n {props.showNext && (\n \n )}\n\n {props.showLast && (\n \n )}\n
\n \n );\n}\n\ntype PaginationLinkProps = {\n label: string;\n ariaLabel: string;\n pageNumber: number;\n isDisabled?: boolean;\n isSelected?: boolean;\n className?: string;\n cssClasses: PaginationComponentCSSClasses;\n createURL(value: number): string;\n createClickHandler: (pageNumber: number) => (event: MouseEvent) => void;\n};\n\nfunction PaginationLink({\n label,\n ariaLabel,\n pageNumber,\n className,\n isDisabled = false,\n isSelected = false,\n cssClasses,\n createURL,\n createClickHandler,\n}: PaginationLinkProps) {\n return (\n \n {isDisabled ? (\n \n ) : (\n \n )}\n \n );\n}\n\nexport default Pagination;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport cx from 'classnames';\nimport type {\n PaginationComponentCSSClasses,\n PaginationComponentTemplates,\n} from '../../components/Pagination/Pagination';\nimport Pagination from '../../components/Pagination/Pagination';\nimport type {\n PaginationConnectorParams,\n PaginationRenderState,\n PaginationWidgetDescription,\n} from '../../connectors/pagination/connectPagination';\nimport connectPagination from '../../connectors/pagination/connectPagination';\nimport {\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type { Renderer, WidgetFactory } from '../../types';\n\nconst suit = component('Pagination');\nconst withUsage = createDocumentationMessageGenerator({ name: 'pagination' });\n\nconst defaultTemplates: PaginationComponentTemplates = {\n previous: '‹',\n next: '›',\n first: '«',\n last: '»',\n};\n\nconst renderer =\n ({\n containerNode,\n cssClasses,\n templates,\n showFirst,\n showLast,\n showPrevious,\n showNext,\n scrollToNode,\n }: {\n containerNode: HTMLElement;\n cssClasses: PaginationComponentCSSClasses;\n templates: PaginationComponentTemplates;\n showFirst: boolean;\n showLast: boolean;\n showPrevious: boolean;\n showNext: boolean;\n scrollToNode: HTMLElement | false;\n }): Renderer> =>\n (\n {\n createURL,\n currentRefinement,\n nbPages,\n pages,\n isFirstPage,\n isLastPage,\n refine,\n },\n isFirstRendering\n ) => {\n if (isFirstRendering) return;\n\n const setCurrentPage = (pageNumber: number) => {\n refine(pageNumber);\n\n if (scrollToNode !== false) {\n scrollToNode.scrollIntoView();\n }\n };\n\n render(\n ,\n containerNode\n );\n };\n\nexport type PaginationCSSClasses = Partial<{\n /**\n * CSS classes added to the root element of the widget.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the root element of the widget if there are no refinements.\n */\n noRefinementRoot: string | string[];\n\n /**\n * CSS classes added to the wrapping `
    `.\n */\n list: string | string[];\n\n /**\n * CSS classes added to each `
  • `.\n */\n item: string | string[];\n\n /**\n * CSS classes added to the first `
  • `.\n */\n firstPageItem: string | string[];\n\n /**\n * CSS classes added to the last `
  • `.\n */\n lastPageItem: string | string[];\n\n /**\n * CSS classes added to the previous `
  • `.\n */\n previousPageItem: string | string[];\n\n /**\n * CSS classes added to the next `
  • `.\n */\n nextPageItem: string | string[];\n\n /**\n * CSS classes added to page `
  • `.\n */\n pageItem: string | string[];\n\n /**\n * CSS classes added to the selected `
  • `.\n */\n selectedItem: string | string[];\n\n /**\n * CSS classes added to the disabled `
  • `.\n */\n disabledItem: string | string[];\n\n /**\n * CSS classes added to each link.\n */\n link: string | string[];\n}>;\n\nexport type PaginationTemplates = Partial<{\n /**\n * Label for the Previous link.\n */\n previous: string;\n\n /**\n * Label for the Next link.\n */\n next: string;\n\n /**\n * Label for the First link.\n */\n first: string;\n\n /**\n * Label for the Last link.\n */\n last: string;\n}>;\n\nexport type PaginationWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * The max number of pages to browse.\n */\n totalPages?: number;\n\n /**\n * The number of pages to display on each side of the current page.\n * @default 3\n */\n padding?: number;\n\n /**\n * Where to scroll after a click, set to `false` to disable.\n * @default body\n */\n scrollTo?: string | HTMLElement | boolean;\n\n /**\n * Whether to show the \"first page\" control\n * @default true\n */\n showFirst?: boolean;\n\n /**\n * Whether to show the \"last page\" control\n * @default true\n */\n showLast?: boolean;\n\n /**\n * Whether to show the \"next page\" control\n * @default true\n */\n showNext?: boolean;\n\n /**\n * Whether to show the \"previous page\" control\n * @default true\n */\n showPrevious?: boolean;\n\n /**\n * Text to display in the links.\n */\n templates?: PaginationTemplates;\n\n /**\n * CSS classes to be added.\n */\n cssClasses?: PaginationCSSClasses;\n};\n\nexport type PaginationWidget = WidgetFactory<\n PaginationWidgetDescription & { $$widgetType: 'ais.pagination' },\n PaginationConnectorParams,\n PaginationWidgetParams\n>;\n\nconst pagination: PaginationWidget = function pagination(widgetParams) {\n const {\n container,\n templates: userTemplates = {},\n cssClasses: userCssClasses = {},\n totalPages,\n padding,\n showFirst = true,\n showLast = true,\n showPrevious = true,\n showNext = true,\n scrollTo: userScrollTo = 'body',\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n\n const scrollTo = userScrollTo === true ? 'body' : userScrollTo;\n const scrollToNode = scrollTo !== false ? getContainerNode(scrollTo) : false;\n\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n noRefinementRoot: cx(\n suit({ modifierName: 'noRefinement' }),\n userCssClasses.noRefinementRoot\n ),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n firstPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'firstPage' }),\n userCssClasses.firstPageItem\n ),\n lastPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'lastPage' }),\n userCssClasses.lastPageItem\n ),\n previousPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'previousPage' }),\n userCssClasses.previousPageItem\n ),\n nextPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'nextPage' }),\n userCssClasses.nextPageItem\n ),\n pageItem: cx(\n suit({ descendantName: 'item', modifierName: 'page' }),\n userCssClasses.pageItem\n ),\n selectedItem: cx(\n suit({ descendantName: 'item', modifierName: 'selected' }),\n userCssClasses.selectedItem\n ),\n disabledItem: cx(\n suit({ descendantName: 'item', modifierName: 'disabled' }),\n userCssClasses.disabledItem\n ),\n link: cx(suit({ descendantName: 'link' }), userCssClasses.link),\n };\n\n const templates = {\n ...defaultTemplates,\n ...userTemplates,\n };\n\n const specializedRenderer = renderer({\n containerNode,\n cssClasses,\n templates,\n showFirst,\n showLast,\n showPrevious,\n showNext,\n scrollToNode,\n });\n\n const makeWidget = connectPagination(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({ totalPages, padding }),\n $$widgetType: 'ais.pagination',\n };\n};\n\nexport default pagination;\n","import{options as n}from\"preact\";var t,u,r,o=0,i=[],c=n.__b,f=n.__r,e=n.diffed,a=n.__c,v=n.unmount;function m(t,r){n.__h&&n.__h(u,t,o||r),o=0;var i=u.__H||(u.__H={__:[],__h:[]});return t>=i.__.length&&i.__.push({}),i.__[t]}function l(n){return o=1,p(w,n)}function p(n,r,o){var i=m(t++,2);return i.t=n,i.__c||(i.__=[o?o(r):w(void 0,r),function(n){var t=i.t(i.__[0],n);i.__[0]!==t&&(i.__=[t,i.__[1]],i.__c.setState({}))}],i.__c=u),i.__}function y(r,o){var i=m(t++,3);!n.__s&&k(i.__H,o)&&(i.__=r,i.__H=o,u.__H.__h.push(i))}function h(r,o){var i=m(t++,4);!n.__s&&k(i.__H,o)&&(i.__=r,i.__H=o,u.__h.push(i))}function s(n){return o=5,d(function(){return{current:n}},[])}function _(n,t,u){o=6,h(function(){\"function\"==typeof n?n(t()):n&&(n.current=t())},null==u?u:u.concat(n))}function d(n,u){var r=m(t++,7);return k(r.__H,u)&&(r.__=n(),r.__H=u,r.__h=n),r.__}function A(n,t){return o=8,d(function(){return n},t)}function F(n){var r=u.context[n.__c],o=m(t++,9);return o.c=n,r?(null==o.__&&(o.__=!0,r.sub(u)),r.props.value):n.__}function T(t,u){n.useDebugValue&&n.useDebugValue(u?u(t):t)}function q(n){var r=m(t++,10),o=l();return r.__=n,u.componentDidCatch||(u.componentDidCatch=function(n){r.__&&r.__(n),o[1](n)}),[o[0],function(){o[1](void 0)}]}function x(){var t;for(i.sort(function(n,t){return n.__v.__b-t.__v.__b});t=i.pop();)if(t.__P)try{t.__H.__h.forEach(g),t.__H.__h.forEach(j),t.__H.__h=[]}catch(u){t.__H.__h=[],n.__e(u,t.__v)}}n.__b=function(n){u=null,c&&c(n)},n.__r=function(n){f&&f(n),t=0;var r=(u=n.__c).__H;r&&(r.__h.forEach(g),r.__h.forEach(j),r.__h=[])},n.diffed=function(t){e&&e(t);var o=t.__c;o&&o.__H&&o.__H.__h.length&&(1!==i.push(o)&&r===n.requestAnimationFrame||((r=n.requestAnimationFrame)||function(n){var t,u=function(){clearTimeout(r),b&&cancelAnimationFrame(t),setTimeout(n)},r=setTimeout(u,100);b&&(t=requestAnimationFrame(u))})(x)),u=null},n.__c=function(t,u){u.some(function(t){try{t.__h.forEach(g),t.__h=t.__h.filter(function(n){return!n.__||j(n)})}catch(r){u.some(function(n){n.__h&&(n.__h=[])}),u=[],n.__e(r,t.__v)}}),a&&a(t,u)},n.unmount=function(t){v&&v(t);var u,r=t.__c;r&&r.__H&&(r.__H.__.forEach(function(n){try{g(n)}catch(n){u=n}}),u&&n.__e(u,r.__v))};var b=\"function\"==typeof requestAnimationFrame;function g(n){var t=u,r=n.__c;\"function\"==typeof r&&(n.__c=void 0,r()),u=t}function j(n){var t=u;n.__c=n.__(),u=t}function k(n,t){return!n||n.length!==t.length||t.some(function(t,u){return t!==n[u]})}function w(n,t){return\"function\"==typeof t?t(n):t}export{l as useState,p as useReducer,y as useEffect,h as useLayoutEffect,s as useRef,_ as useImperativeHandle,d as useMemo,A as useCallback,F as useContext,T as useDebugValue,q as useErrorBoundary};\n//# sourceMappingURL=hooks.module.js.map\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport { useState, useEffect, useRef } from 'preact/hooks';\nimport cx from 'classnames';\nimport Template from '../Template/Template';\nimport type {\n PanelCSSClasses,\n PanelSharedOptions,\n PanelTemplates,\n} from '../../widgets/panel/panel';\nimport type { ComponentCSSClasses, UnknownWidgetFactory } from '../../types';\n\nexport type PanelComponentCSSClasses = ComponentCSSClasses<\n // `collapseIcon` is only used in the default templates of the widget\n Omit\n>;\n\nexport type PanelComponentTemplates =\n Required>;\n\nexport type PanelProps = {\n hidden: boolean;\n collapsible: boolean;\n isCollapsed: boolean;\n data: PanelSharedOptions;\n cssClasses: PanelComponentCSSClasses;\n templates: PanelComponentTemplates;\n bodyElement: HTMLElement;\n};\n\nfunction Panel(\n props: PanelProps\n) {\n const [isCollapsed, setIsCollapsed] = useState(props.isCollapsed);\n const [isControlled, setIsControlled] = useState(false);\n const bodyRef = useRef(null);\n\n useEffect(() => {\n const node = bodyRef.current;\n if (!node) {\n return undefined;\n }\n\n node.appendChild(props.bodyElement);\n\n return () => {\n node.removeChild(props.bodyElement);\n };\n }, [bodyRef, props.bodyElement]);\n\n if (!isControlled && props.isCollapsed !== isCollapsed) {\n setIsCollapsed(props.isCollapsed);\n }\n\n return (\n
  • \n );\n}\n\nexport default Pagination;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport { cx } from '@algolia/ui-components-shared';\nimport type {\n PaginationComponentCSSClasses,\n PaginationComponentTemplates,\n} from '../../components/Pagination/Pagination';\nimport Pagination from '../../components/Pagination/Pagination';\nimport type {\n PaginationConnectorParams,\n PaginationRenderState,\n PaginationWidgetDescription,\n} from '../../connectors/pagination/connectPagination';\nimport connectPagination from '../../connectors/pagination/connectPagination';\nimport {\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { component } from '../../lib/suit';\nimport type { Renderer, WidgetFactory } from '../../types';\n\nconst suit = component('Pagination');\nconst withUsage = createDocumentationMessageGenerator({ name: 'pagination' });\n\nconst defaultTemplates: PaginationComponentTemplates = {\n previous: '‹',\n next: '›',\n first: '«',\n last: '»',\n};\n\nconst renderer =\n ({\n containerNode,\n cssClasses,\n templates,\n showFirst,\n showLast,\n showPrevious,\n showNext,\n scrollToNode,\n }: {\n containerNode: HTMLElement;\n cssClasses: PaginationComponentCSSClasses;\n templates: PaginationComponentTemplates;\n showFirst: boolean;\n showLast: boolean;\n showPrevious: boolean;\n showNext: boolean;\n scrollToNode: HTMLElement | false;\n }): Renderer> =>\n (\n {\n createURL,\n currentRefinement,\n nbPages,\n pages,\n isFirstPage,\n isLastPage,\n refine,\n },\n isFirstRendering\n ) => {\n if (isFirstRendering) return;\n\n const setCurrentPage = (pageNumber: number) => {\n refine(pageNumber);\n\n if (scrollToNode !== false) {\n scrollToNode.scrollIntoView();\n }\n };\n\n render(\n ,\n containerNode\n );\n };\n\nexport type PaginationCSSClasses = Partial<{\n /**\n * CSS classes added to the root element of the widget.\n */\n root: string | string[];\n\n /**\n * CSS class to add to the root element of the widget if there are no refinements.\n */\n noRefinementRoot: string | string[];\n\n /**\n * CSS classes added to the wrapping `
      `.\n */\n list: string | string[];\n\n /**\n * CSS classes added to each `
    • `.\n */\n item: string | string[];\n\n /**\n * CSS classes added to the first `
    • `.\n */\n firstPageItem: string | string[];\n\n /**\n * CSS classes added to the last `
    • `.\n */\n lastPageItem: string | string[];\n\n /**\n * CSS classes added to the previous `
    • `.\n */\n previousPageItem: string | string[];\n\n /**\n * CSS classes added to the next `
    • `.\n */\n nextPageItem: string | string[];\n\n /**\n * CSS classes added to page `
    • `.\n */\n pageItem: string | string[];\n\n /**\n * CSS classes added to the selected `
    • `.\n */\n selectedItem: string | string[];\n\n /**\n * CSS classes added to the disabled `
    • `.\n */\n disabledItem: string | string[];\n\n /**\n * CSS classes added to each link.\n */\n link: string | string[];\n}>;\n\nexport type PaginationTemplates = Partial<{\n /**\n * Label for the Previous link.\n */\n previous: string;\n\n /**\n * Label for the Next link.\n */\n next: string;\n\n /**\n * Label for the First link.\n */\n first: string;\n\n /**\n * Label for the Last link.\n */\n last: string;\n}>;\n\nexport type PaginationWidgetParams = {\n /**\n * CSS Selector or HTMLElement to insert the widget.\n */\n container: string | HTMLElement;\n\n /**\n * The max number of pages to browse.\n */\n totalPages?: number;\n\n /**\n * The number of pages to display on each side of the current page.\n * @default 3\n */\n padding?: number;\n\n /**\n * Where to scroll after a click, set to `false` to disable.\n * @default body\n */\n scrollTo?: string | HTMLElement | boolean;\n\n /**\n * Whether to show the \"first page\" control\n * @default true\n */\n showFirst?: boolean;\n\n /**\n * Whether to show the \"last page\" control\n * @default true\n */\n showLast?: boolean;\n\n /**\n * Whether to show the \"next page\" control\n * @default true\n */\n showNext?: boolean;\n\n /**\n * Whether to show the \"previous page\" control\n * @default true\n */\n showPrevious?: boolean;\n\n /**\n * Text to display in the links.\n */\n templates?: PaginationTemplates;\n\n /**\n * CSS classes to be added.\n */\n cssClasses?: PaginationCSSClasses;\n};\n\nexport type PaginationWidget = WidgetFactory<\n PaginationWidgetDescription & { $$widgetType: 'ais.pagination' },\n PaginationConnectorParams,\n PaginationWidgetParams\n>;\n\nconst pagination: PaginationWidget = function pagination(widgetParams) {\n const {\n container,\n templates: userTemplates = {},\n cssClasses: userCssClasses = {},\n totalPages,\n padding,\n showFirst = true,\n showLast = true,\n showPrevious = true,\n showNext = true,\n scrollTo: userScrollTo = 'body',\n } = widgetParams || {};\n\n if (!container) {\n throw new Error(withUsage('The `container` option is required.'));\n }\n\n const containerNode = getContainerNode(container);\n\n const scrollTo = userScrollTo === true ? 'body' : userScrollTo;\n const scrollToNode = scrollTo !== false ? getContainerNode(scrollTo) : false;\n\n const cssClasses = {\n root: cx(suit(), userCssClasses.root),\n noRefinementRoot: cx(\n suit({ modifierName: 'noRefinement' }),\n userCssClasses.noRefinementRoot\n ),\n list: cx(suit({ descendantName: 'list' }), userCssClasses.list),\n item: cx(suit({ descendantName: 'item' }), userCssClasses.item),\n firstPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'firstPage' }),\n userCssClasses.firstPageItem\n ),\n lastPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'lastPage' }),\n userCssClasses.lastPageItem\n ),\n previousPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'previousPage' }),\n userCssClasses.previousPageItem\n ),\n nextPageItem: cx(\n suit({ descendantName: 'item', modifierName: 'nextPage' }),\n userCssClasses.nextPageItem\n ),\n pageItem: cx(\n suit({ descendantName: 'item', modifierName: 'page' }),\n userCssClasses.pageItem\n ),\n selectedItem: cx(\n suit({ descendantName: 'item', modifierName: 'selected' }),\n userCssClasses.selectedItem\n ),\n disabledItem: cx(\n suit({ descendantName: 'item', modifierName: 'disabled' }),\n userCssClasses.disabledItem\n ),\n link: cx(suit({ descendantName: 'link' }), userCssClasses.link),\n };\n\n const templates = {\n ...defaultTemplates,\n ...userTemplates,\n };\n\n const specializedRenderer = renderer({\n containerNode,\n cssClasses,\n templates,\n showFirst,\n showLast,\n showPrevious,\n showNext,\n scrollToNode,\n });\n\n const makeWidget = connectPagination(specializedRenderer, () =>\n render(null, containerNode)\n );\n\n return {\n ...makeWidget({ totalPages, padding }),\n $$widgetType: 'ais.pagination',\n };\n};\n\nexport default pagination;\n","import{options as n}from\"preact\";var t,r,u,i,o=0,c=[],f=[],e=n.__b,a=n.__r,v=n.diffed,l=n.__c,m=n.unmount;function d(t,u){n.__h&&n.__h(r,t,o||u),o=0;var i=r.__H||(r.__H={__:[],__h:[]});return t>=i.__.length&&i.__.push({__V:f}),i.__[t]}function p(n){return o=1,y(z,n)}function y(n,u,i){var o=d(t++,2);if(o.t=n,!o.__c&&(o.__=[i?i(u):z(void 0,u),function(n){var t=o.__N?o.__N[0]:o.__[0],r=o.t(t,n);t!==r&&(o.__N=[r,o.__[1]],o.__c.setState({}))}],o.__c=r,!r.u)){r.u=!0;var c=r.shouldComponentUpdate;r.shouldComponentUpdate=function(n,t,r){if(!o.__c.__H)return!0;var u=o.__c.__H.__.filter(function(n){return n.__c});return u.every(function(n){return!n.__N})?!c||c.call(this,n,t,r):!u.every(function(n){if(!n.__N)return!0;var t=n.__[0];return n.__=n.__N,n.__N=void 0,t===n.__[0]})&&(!c||c.call(this,n,t,r))}}return o.__N||o.__}function h(u,i){var o=d(t++,3);!n.__s&&w(o.__H,i)&&(o.__=u,o.i=i,r.__H.__h.push(o))}function s(u,i){var o=d(t++,4);!n.__s&&w(o.__H,i)&&(o.__=u,o.i=i,r.__h.push(o))}function _(n){return o=5,F(function(){return{current:n}},[])}function A(n,t,r){o=6,s(function(){return\"function\"==typeof n?(n(t()),function(){return n(null)}):n?(n.current=t(),function(){return n.current=null}):void 0},null==r?r:r.concat(n))}function F(n,r){var u=d(t++,7);return w(u.__H,r)?(u.__V=n(),u.i=r,u.__h=n,u.__V):u.__}function T(n,t){return o=8,F(function(){return n},t)}function q(n){var u=r.context[n.__c],i=d(t++,9);return i.c=n,u?(null==i.__&&(i.__=!0,u.sub(r)),u.props.value):n.__}function x(t,r){n.useDebugValue&&n.useDebugValue(r?r(t):t)}function V(n){var u=d(t++,10),i=p();return u.__=n,r.componentDidCatch||(r.componentDidCatch=function(n){u.__&&u.__(n),i[1](n)}),[i[0],function(){i[1](void 0)}]}function b(){for(var t;t=c.shift();)if(t.__P&&t.__H)try{t.__H.__h.forEach(j),t.__H.__h.forEach(k),t.__H.__h=[]}catch(r){t.__H.__h=[],n.__e(r,t.__v)}}n.__b=function(n){r=null,e&&e(n)},n.__r=function(n){a&&a(n),t=0;var i=(r=n.__c).__H;i&&(u===r?(i.__h=[],r.__h=[],i.__.forEach(function(n){n.__N&&(n.__=n.__N),n.__V=f,n.__N=n.i=void 0})):(i.__h.forEach(j),i.__h.forEach(k),i.__h=[])),u=r},n.diffed=function(t){v&&v(t);var o=t.__c;o&&o.__H&&(o.__H.__h.length&&(1!==c.push(o)&&i===n.requestAnimationFrame||((i=n.requestAnimationFrame)||function(n){var t,r=function(){clearTimeout(u),g&&cancelAnimationFrame(t),setTimeout(n)},u=setTimeout(r,100);g&&(t=requestAnimationFrame(r))})(b)),o.__H.__.forEach(function(n){n.i&&(n.__H=n.i),n.__V!==f&&(n.__=n.__V),n.i=void 0,n.__V=f})),u=r=null},n.__c=function(t,r){r.some(function(t){try{t.__h.forEach(j),t.__h=t.__h.filter(function(n){return!n.__||k(n)})}catch(u){r.some(function(n){n.__h&&(n.__h=[])}),r=[],n.__e(u,t.__v)}}),l&&l(t,r)},n.unmount=function(t){m&&m(t);var r,u=t.__c;u&&u.__H&&(u.__H.__.forEach(function(n){try{j(n)}catch(n){r=n}}),r&&n.__e(r,u.__v))};var g=\"function\"==typeof requestAnimationFrame;function j(n){var t=r,u=n.__c;\"function\"==typeof u&&(n.__c=void 0,u()),r=t}function k(n){var t=r;n.__c=n.__(),r=t}function w(n,t){return!n||n.length!==t.length||t.some(function(t,r){return t!==n[r]})}function z(n,t){return\"function\"==typeof t?t(n):t}export{p as useState,y as useReducer,h as useEffect,s as useLayoutEffect,_ as useRef,A as useImperativeHandle,F as useMemo,T as useCallback,q as useContext,x as useDebugValue,V as useErrorBoundary};\n//# sourceMappingURL=hooks.module.js.map\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport { useState, useEffect, useRef } from 'preact/hooks';\nimport { cx } from '@algolia/ui-components-shared';\nimport Template from '../Template/Template';\nimport type {\n PanelCSSClasses,\n PanelSharedOptions,\n PanelTemplates,\n} from '../../widgets/panel/panel';\nimport type { ComponentCSSClasses, UnknownWidgetFactory } from '../../types';\n\nexport type PanelComponentCSSClasses = ComponentCSSClasses<\n // `collapseIcon` is only used in the default templates of the widget\n Omit\n>;\n\nexport type PanelComponentTemplates =\n Required>;\n\nexport type PanelProps = {\n hidden: boolean;\n collapsible: boolean;\n isCollapsed: boolean;\n data: PanelSharedOptions;\n cssClasses: PanelComponentCSSClasses;\n templates: PanelComponentTemplates;\n bodyElement: HTMLElement;\n};\n\nfunction Panel(\n props: PanelProps\n) {\n const [isCollapsed, setIsCollapsed] = useState(props.isCollapsed);\n const [isControlled, setIsControlled] = useState(false);\n const bodyRef = useRef(null);\n\n useEffect(() => {\n const node = bodyRef.current;\n if (!node) {\n return undefined;\n }\n\n node.appendChild(props.bodyElement);\n\n return () => {\n node.removeChild(props.bodyElement);\n };\n }, [bodyRef, props.bodyElement]);\n\n if (!isControlled && props.isCollapsed !== isCollapsed) {\n setIsCollapsed(props.isCollapsed);\n }\n\n return (\n