From d765a43c1f13dba55c0e22b13a1856c14b61420c Mon Sep 17 00:00:00 2001 From: Michael Beckwith Date: Tue, 1 Nov 2022 16:44:08 -0500 Subject: [PATCH 1/5] switch to addEventListener instead of onload function --- templates/autocomplete.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/templates/autocomplete.php b/templates/autocomplete.php index e88f38b6..3a2ff1d5 100644 --- a/templates/autocomplete.php +++ b/templates/autocomplete.php @@ -72,7 +72,7 @@ From 1412688f724d74e562bce017874506056741503d Mon Sep 17 00:00:00 2001 From: Ashar Irfan Date: Fri, 11 Nov 2022 23:28:27 +0500 Subject: [PATCH 2/5] Update instantsearch.js from 4.49.0 to 4.49.1 --- js/instantsearch.js/CHANGELOG.md | 9 +++++++++ .../cjs/helpers/get-insights-anonymous-user-token.js | 6 ++++++ js/instantsearch.js/cjs/lib/version.js | 2 +- js/instantsearch.js/dist/instantsearch.development.js | 8 ++++++-- .../dist/instantsearch.development.js.map | 2 +- js/instantsearch.js/dist/instantsearch.production.min.js | 4 ++-- .../dist/instantsearch.production.min.js.map | 2 +- .../es/helpers/get-insights-anonymous-user-token.js | 6 ++++++ js/instantsearch.js/es/lib/version.d.ts | 2 +- js/instantsearch.js/es/lib/version.js | 2 +- js/instantsearch.js/package.json | 2 +- 11 files changed, 35 insertions(+), 10 deletions(-) diff --git a/js/instantsearch.js/CHANGELOG.md b/js/instantsearch.js/CHANGELOG.md index d8f485bb..e2b3b533 100644 --- a/js/instantsearch.js/CHANGELOG.md +++ b/js/instantsearch.js/CHANGELOG.md @@ -1,3 +1,12 @@ +## [4.49.1](https://github.com/algolia/instantsearch.js/compare/v4.49.0...v4.49.1) (2022-11-01) + + +### Bug Fixes + +* **insights:** check before usage of `document` ([#5149](https://github.com/algolia/instantsearch.js/issues/5149)) ([6733dea](https://github.com/algolia/instantsearch.js/commit/6733dea1091a3a6c8ec9049eba652a7f06e9c501)) + + + # [4.49.0](https://github.com/algolia/instantsearch.js/compare/v4.48.1...v4.49.0) (2022-10-25) diff --git a/js/instantsearch.js/cjs/helpers/get-insights-anonymous-user-token.js b/js/instantsearch.js/cjs/helpers/get-insights-anonymous-user-token.js index 51983463..f7b83cf8 100644 --- a/js/instantsearch.js/cjs/helpers/get-insights-anonymous-user-token.js +++ b/js/instantsearch.js/cjs/helpers/get-insights-anonymous-user-token.js @@ -9,10 +9,16 @@ exports.ANONYMOUS_TOKEN_COOKIE_KEY = void 0; var _index = require("../lib/utils/index.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); } + var ANONYMOUS_TOKEN_COOKIE_KEY = '_ALGOLIA'; exports.ANONYMOUS_TOKEN_COOKIE_KEY = ANONYMOUS_TOKEN_COOKIE_KEY; function getCookie(name) { + if ((typeof document === "undefined" ? "undefined" : _typeof(document)) !== 'object' || typeof document.cookie !== 'string') { + return undefined; + } + var prefix = "".concat(name, "="); var cookies = document.cookie.split(';'); diff --git a/js/instantsearch.js/cjs/lib/version.js b/js/instantsearch.js/cjs/lib/version.js index 6d51e155..1372f60f 100644 --- a/js/instantsearch.js/cjs/lib/version.js +++ b/js/instantsearch.js/cjs/lib/version.js @@ -4,5 +4,5 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; -var _default = '4.49.0'; +var _default = '4.49.1'; exports.default = _default; \ No newline at end of file diff --git a/js/instantsearch.js/dist/instantsearch.development.js b/js/instantsearch.js/dist/instantsearch.development.js index bde6e34f..1db82dff 100644 --- a/js/instantsearch.js/dist/instantsearch.development.js +++ b/js/instantsearch.js/dist/instantsearch.development.js @@ -1,4 +1,4 @@ -/*! InstantSearch.js 4.49.0 | © Algolia, Inc. and contributors; MIT License | https://github.com/algolia/instantsearch.js */ +/*! InstantSearch.js 4.49.1 | © 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) : @@ -7784,7 +7784,7 @@ instantSearchInstance.renderState = _objectSpread2(_objectSpread2({}, instantSearchInstance.renderState), {}, _defineProperty({}, parentIndexName, _objectSpread2(_objectSpread2({}, instantSearchInstance.renderState[parentIndexName]), renderState))); } - var version$1 = '4.49.0'; + var version$1 = '4.49.1'; var NAMESPACE = 'ais'; var component = function component(componentName) { @@ -7947,6 +7947,10 @@ var ANONYMOUS_TOKEN_COOKIE_KEY = '_ALGOLIA'; function getCookie(name) { + if ((typeof document === "undefined" ? "undefined" : _typeof(document)) !== 'object' || typeof document.cookie !== 'string') { + return undefined; + } + var prefix = "".concat(name, "="); var cookies = document.cookie.split(';'); diff --git a/js/instantsearch.js/dist/instantsearch.development.js.map b/js/instantsearch.js/dist/instantsearch.development.js.map index 66e5a7e3..7c9de647 100644 --- a/js/instantsearch.js/dist/instantsearch.development.js.map +++ b/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/noop.ts","../src/lib/utils/logger.ts","../src/lib/utils/typedObject.ts","../src/lib/utils/checkIndexUiState.ts","../src/lib/utils/getObjectType.ts","../src/lib/utils/checkRendering.ts","../src/lib/utils/clearRefinements.ts","../src/lib/utils/escape-html.ts","../src/lib/utils/isPlainObject.ts","../src/lib/utils/escape-highlight.ts","../src/lib/utils/concatHighlightedParts.ts","../src/lib/utils/createConcurrentSafePromise.ts","../src/lib/utils/isFacetRefined.ts","../src/lib/utils/createSendEventForFacet.ts","../src/lib/utils/serializer.ts","../src/lib/utils/createSendEventForHits.ts","../src/lib/utils/isIndexWidget.ts","../src/lib/utils/setIndexHelperState.ts","../src/lib/utils/debounce.ts","../src/lib/utils/defer.ts","../src/lib/utils/documentation.ts","../src/lib/utils/escapeFacetValue.ts","../src/lib/utils/find.ts","../src/lib/utils/findIndex.ts","../src/lib/utils/geo-search.ts","../src/lib/utils/getAppIdAndApiKey.ts","../src/lib/utils/isDomElement.ts","../src/lib/utils/getContainerNode.ts","../src/lib/utils/getHighlightedParts.ts","../src/lib/utils/getHighlightFromSiblings.ts","../src/lib/utils/getPropertyByPath.ts","../src/lib/utils/getRefinements.ts","../src/lib/utils/getWidgetAttribute.ts","../src/lib/utils/hits-absolute-position.ts","../src/lib/utils/hits-query-id.ts","../src/lib/utils/isEqual.ts","../src/lib/utils/isFiniteNumber.ts","../src/lib/utils/isSpecialClick.ts","../src/lib/utils/uniq.ts","../src/lib/utils/mergeSearchParameters.ts","../src/lib/utils/range.ts","../src/lib/utils/render-args.ts","../src/lib/utils/resolveSearchParameters.ts","../src/lib/utils/reverseHighlightedParts.ts","../src/lib/utils/safelyRunOnBrowser.ts","../src/lib/utils/toArray.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/formatNumber.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/@algolia/ui-components-shared/dist/esm/cx.js","../src/lib/templating/prepareTemplateProps.ts","../node_modules/hogan.js/lib/compiler.js","../node_modules/hogan.js/lib/template.js","../node_modules/hogan.js/lib/hogan.js","../node_modules/htm/dist/htm.module.js","../node_modules/htm/preact/index.module.js","../node_modules/@babel/runtime/helpers/extends.js","../node_modules/@babel/runtime/helpers/objectWithoutPropertiesLoose.js","../node_modules/@babel/runtime/helpers/objectWithoutProperties.js","../node_modules/@algolia/ui-components-highlight-vdom/dist/esm/Highlight.js","../src/components/InternalHighlight/InternalHighlight.tsx","../src/components/Highlight/Highlight.tsx","../src/helpers/components/Highlight.tsx","../src/components/ReverseHighlight/ReverseHighlight.tsx","../src/helpers/components/ReverseHighlight.tsx","../src/components/ReverseSnippet/ReverseSnippet.tsx","../src/helpers/components/ReverseSnippet.tsx","../src/components/Snippet/Snippet.tsx","../src/helpers/components/Snippet.tsx","../src/lib/templating/renderTemplate.ts","../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.tsx","../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.tsx","../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.tsx","../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.tsx","../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.tsx","../src/widgets/rating-menu/rating-menu.tsx","../src/widgets/search-box/defaultTemplates.tsx","../src/widgets/refinement-list/defaultTemplates.tsx","../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.tsx","../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 parent 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 results.slice(nextDisjunctiveResult).forEach(function(result) {\n var facets = result && result.facets\n ? result.facets\n : {};\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\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\nfunction sortObject(obj) {\n return Object.keys(obj)\n .sort(function(a, b) {\n return a.localeCompare(b);\n })\n .reduce(function(acc, curr) {\n acc[curr] = obj[curr];\n return acc;\n }, {});\n}\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 // More to get the parent levels of the hierarchical facets when refined\n state.getRefinedHierarchicalFacets().forEach(function(refinedFacet) {\n var hierarchicalFacet = state.getHierarchicalFacetByName(refinedFacet);\n var currentRefinement = state.getHierarchicalRefinement(refinedFacet);\n var separator = state._getHierarchicalFacetSeparator(hierarchicalFacet);\n\n // If we are deeper than level 0 (starting from `beer > IPA`)\n // we want to get all parent values\n if (currentRefinement.length > 0 && currentRefinement[0].split(separator).length > 1) {\n // We generate a map of the filters we will use for our facet values queries\n var filtersMap = currentRefinement[0].split(separator).slice(0, -1).reduce(\n function createFiltersMap(map, segment, level) {\n return map.concat({\n attribute: hierarchicalFacet.attributes[level],\n value: level === 0\n ? segment\n : [map[map.length - 1].value, segment].join(separator)\n });\n }\n , []);\n\n filtersMap.forEach(function(filter, level) {\n var params = requestBuilder._getDisjunctiveFacetSearchParams(\n state,\n filter.attribute,\n level === 0\n );\n\n // Keep facet filters unrelated to current hierarchical attributes\n function hasHierarchicalFacetFilter(value) {\n return hierarchicalFacet.attributes.some(function(attribute) {\n return attribute === value.split(':')[0];\n });\n }\n\n var filteredFacetFilters = (params.facetFilters || []).reduce(function(acc, facetFilter) {\n if (Array.isArray(facetFilter)) {\n var filtered = facetFilter.filter(function(filterValue) {\n return !hasHierarchicalFacetFilter(filterValue);\n });\n\n if (filtered.length > 0) {\n acc.push(filtered);\n }\n }\n\n if (typeof facetFilter === 'string' && !hasHierarchicalFacetFilter(facetFilter)) {\n acc.push(facetFilter);\n }\n\n return acc;\n }, []);\n\n var parent = filtersMap[level - 1];\n if (level > 0) {\n params.facetFilters = filteredFacetFilters.concat(parent.attribute + ':' + parent.value);\n } else {\n params.facetFilters = filteredFacetFilters.length > 0 ? filteredFacetFilters : undefined;\n }\n\n queries.push({indexName: index, params: params});\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 sortObject(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: 0,\n page: 0,\n analytics: false,\n clickAnalytics: false\n };\n\n if (tagFilters.length > 0) {\n additionalParams.tagFilters = tagFilters;\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 sortObject(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 sortObject(merge(\n {},\n requestBuilder._getHitsSearchParams(stateForSearchForFacetValues),\n searchForFacetSearchParameters\n ));\n }\n};\n\nmodule.exports = requestBuilder;\n","'use strict';\n\nmodule.exports = '3.11.1';\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 var clientHasInitIndex = typeof this.client.initIndex === 'function';\n if (\n !clientHasSFFV &&\n !clientHasInitIndex &&\n typeof this.client.search !== '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\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 var searchForFacetValuesPromise;\n // newer algoliasearch ^3.27.1 - ~4.0.0\n if (clientHasSFFV) {\n searchForFacetValuesPromise = this.client.searchForFacetValues([\n {indexName: state.index, params: algoliaQuery}\n ]);\n // algoliasearch < 3.27.1\n } else if (clientHasInitIndex) {\n searchForFacetValuesPromise = this.client\n .initIndex(state.index)\n .searchForFacetValues(algoliaQuery);\n // algoliasearch ~5.0.0\n } else {\n // @MAJOR only use client.search\n delete algoliaQuery.facetName;\n searchForFacetValuesPromise = this.client\n .search([\n {\n type: 'facet',\n facet: facet,\n indexName: state.index,\n params: algoliaQuery\n }\n ])\n .then(function processResponse(response) {\n return response.results[0];\n });\n }\n\n this.emit('searchForFacetValues', {\n state: state,\n facet: facet,\n query: query\n });\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","export function capitalize(text: string): string {\n return text.toString().charAt(0).toUpperCase() + text.toString().slice(1);\n}\n","export function noop(..._args: any[]): void {}\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","export function getObjectType(object: unknown): string {\n return Object.prototype.toString.call(object).slice(8, -1);\n}\n","import type { Renderer } from '../../types/connector';\nimport { getObjectType } from './getObjectType';\n\nexport function 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","import type {\n AlgoliaSearchHelper,\n SearchParameters,\n} 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 * @returns search parameters with refinements cleared\n */\nexport function clearRefinements({\n helper,\n attributesToClear = [],\n}: {\n helper: AlgoliaSearchHelper;\n attributesToClear?: string[];\n}): SearchParameters {\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","/**\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 htmlEntities = {\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 */\nexport function escape(value: string): string {\n return value && regexHasUnescapedHtml.test(value)\n ? value.replace(\n regexUnescapedHtml,\n (character) => htmlEntities[character as keyof typeof htmlEntities]\n )\n : value;\n}\n\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 htmlCharacters = {\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 function unescape(value: string): string {\n return value && regexHasEscapedHtml.test(value)\n ? value.replace(\n regexEscapedHtml,\n (character) => htmlCharacters[character as keyof typeof htmlCharacters]\n )\n : value;\n}\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 */\nexport function 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","import { escape } from './escape-html';\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 EscapedHits).__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 EscapedHits).__escaped = true;\n }\n\n return hits 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 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","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 { AlgoliaSearchHelper } from 'algoliasearch-helper';\n\nexport 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\ntype CreateSendEventForFacetOptions = {\n instantSearchInstance: InstantSearch;\n helper: AlgoliaSearchHelper;\n attribute: string | ((facetValue: string) => string);\n widgetType: string;\n};\n\nexport function createSendEventForFacet({\n instantSearchInstance,\n helper,\n attribute: attr,\n widgetType,\n}: CreateSendEventForFacetOptions): SendEventForFacet {\n const sendEventForFacet: SendEventForFacet = (...args: any[]) => {\n const [eventType, facetValue, eventName = 'Filter Applied'] = args;\n const attribute = typeof attr === 'string' ? attr : attr(facetValue);\n\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\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","import type { InstantSearch, Hit, EscapedHits } from '../../types';\nimport { serializePayload } from './serializer';\nimport type { InsightsEvent } from '../../middlewares/createInsightsMiddleware';\n\ntype BuiltInSendEventForHits = (\n eventType: string,\n hits: Hit | Hit[],\n eventName?: string\n) => void;\ntype CustomSendEventForHits = (customPayload: any) => void;\nexport type SendEventForHits = BuiltInSendEventForHits & CustomSendEventForHits;\n\ntype BuiltInBindEventForHits = (\n eventType: string,\n hits: Hit | Hit[],\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 isSearchStalled,\n}: {\n widgetType: string;\n index: string;\n methodName: 'sendEvent' | 'bindEvent';\n args: any[];\n isSearchStalled: boolean;\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 | Hit[] | 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: Hit[] = 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 if (isSearchStalled) {\n return [];\n }\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: Hit[] | EscapedHits): Hit[] {\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 isSearchStalled: instantSearchInstance.status === 'stalled',\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 isSearchStalled: false,\n });\n\n return payloads.length\n ? `data-insights-event=${serializePayload(payloads)}`\n : '';\n };\n return bindEventForHits;\n}\n","import type { Widget } from '../../types';\nimport type { IndexWidget } from '../../widgets/index/index';\n\nexport function isIndexWidget(\n widget: Widget | IndexWidget\n): widget is IndexWidget {\n return widget.$$type === 'ais.index';\n}\n","import type { UiState } from '../../types';\nimport type { IndexWidget } from '../../widgets/index/index';\nimport { isIndexWidget } from './isIndexWidget';\nimport { checkIndexUiState } from './checkIndexUiState';\n\nexport function setIndexHelperState(\n finalUiState: TUiState,\n indexWidget: IndexWidget\n) {\n const nextIndexUiState = finalUiState[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((widget) => setIndexHelperState(finalUiState, widget));\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","const nextMicroTask = Promise.resolve();\n\ntype Callback = (...args: any[]) => void;\ntype Defer = {\n wait(): Promise;\n cancel(): void;\n};\n\nexport function defer(\n callback: TCallback\n): TCallback & Defer {\n let progress: Promise | null = null;\n let cancelled = false;\n\n const fn = ((...args: Parameters) => {\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 }) as TCallback & Defer;\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","type WidgetParam = {\n name: string;\n connector?: boolean;\n};\n\nexport function 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 function 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","type FacetValue = string | number | boolean | 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","// 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\nexport function 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","// 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\nexport function 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","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","// typed as any, since it accepts the _real_ js clients, not the interface we otherwise expect\nexport function getAppIdAndApiKey(searchClient: any): [string, string] {\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","export function isDomElement(object: any): object is HTMLElement {\n return (\n object instanceof HTMLElement || (Boolean(object) && object.nodeType > 0)\n );\n}\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 */\nexport function 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","import { TAG_REPLACEMENT } from './escape-highlight';\n\nexport 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 './escape-html';\nimport type { HighlightedParts } from '../../types';\n\nconst hasAlphanumeric = new RegExp(/\\w/i);\n\nexport function getHighlightFromSiblings(parts: HighlightedParts[], i: number) {\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","export 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","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 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 { 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","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","function isPrimitive(obj: any): boolean {\n return obj !== Object(obj);\n}\n\nexport function 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","// 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\n// @MAJOR Replace with the native `Number.isFinite` method\nexport function isFiniteNumber(value: any): value is number {\n return typeof value === 'number' && isFinite(value);\n}\n","export 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","export function uniq(array: TItem[]): TItem[] {\n return array.filter((value, index, self) => self.indexOf(value) === index);\n}\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\nexport const mergeSearchParameters = (\n ...parameters: SearchParameters[]\n): 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","type RangeOptions = {\n start?: number;\n end: number;\n step?: number;\n};\n\nexport function 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","import type { InstantSearch, UiState } from '../../types';\nimport type { IndexWidget } from '../../widgets/index/index';\n\nexport function createInitArgs(\n instantSearchInstance: InstantSearch,\n parent: IndexWidget,\n uiState: UiState\n) {\n const helper = parent.getHelper()!;\n return {\n uiState,\n helper,\n parent,\n instantSearchInstance,\n state: helper.state,\n renderState: instantSearchInstance.renderState,\n templatesConfig: instantSearchInstance.templatesConfig,\n createURL: parent.createURL,\n scopedResults: [],\n searchMetadata: {\n isSearchStalled: instantSearchInstance.status === 'stalled',\n },\n status: instantSearchInstance.status,\n error: instantSearchInstance.error,\n };\n}\n\nexport function createRenderArgs(\n instantSearchInstance: InstantSearch,\n parent: IndexWidget\n) {\n const results = parent.getResults()!;\n\n return {\n helper: parent.getHelper()!,\n parent,\n instantSearchInstance,\n results,\n scopedResults: parent.getScopedResults(),\n state: results._state,\n renderState: instantSearchInstance.renderState,\n templatesConfig: instantSearchInstance.templatesConfig,\n createURL: parent.createURL,\n searchMetadata: {\n isSearchStalled: instantSearchInstance.status === 'stalled',\n },\n status: instantSearchInstance.status,\n error: instantSearchInstance.error,\n };\n}\n","import type { SearchParameters } from 'algoliasearch-helper';\nimport type { IndexWidget } from '../../widgets/index/index';\n\nexport function resolveSearchParameters(\n current: IndexWidget\n): 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","import type { HighlightedParts } from '../../types';\nimport { getHighlightFromSiblings } from './getHighlightFromSiblings';\n\nexport 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","// 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 environments safely.\n */\nexport function safelyRunOnBrowser(\n callback: BrowserCallback,\n { fallback }: SafelyRunOnBrowserOptions = {\n fallback: () => undefined as unknown as TReturn,\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","type ToArray = T extends unknown[] ? T : T[];\n\nexport function toArray(value: T): ToArray {\n return (Array.isArray(value) ? value : [value]) as ToArray;\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 ScopedResult,\n SearchClient,\n IndexRenderState,\n} from '../../types';\nimport {\n checkIndexUiState,\n createDocumentationMessageGenerator,\n resolveSearchParameters,\n mergeSearchParameters,\n warning,\n isIndexWidget,\n createInitArgs,\n createRenderArgs,\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\nexport type IndexInitOptions = {\n instantSearchInstance: InstantSearch;\n parent: IndexWidget | null;\n uiState: UiState;\n};\n\nexport type IndexRenderOptions = {\n instantSearchInstance: InstantSearch;\n};\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(\n uiState: TUiState\n ): TUiState;\n getWidgetSearchParameters(\n searchParameters: SearchParameters,\n searchParametersOptions: { uiState: IndexUiState }\n ): SearchParameters;\n refreshUiState(): void;\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 createInitArgs(\n localInstantSearchInstance!,\n this,\n localInstantSearchInstance!._initialUiState\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 createInitArgs(\n localInstantSearchInstance!,\n this,\n localInstantSearchInstance!._initialUiState\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: (nextState) =>\n instantSearchInstance.setUiState(nextState, false),\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 createInitArgs(instantSearchInstance, this, uiState)\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(createInitArgs(instantSearchInstance, this, uiState));\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 createRenderArgs(instantSearchInstance, this)\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(createRenderArgs(instantSearchInstance, this));\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 [indexId]: {\n ...uiState[indexId],\n ...localUiState,\n },\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.49.0';\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","export function formatNumber(value: number, numberLocale?: string) {\n return value.toLocaleString(numberLocale);\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';\nimport { formatNumber } from './formatNumber';\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 formatNumber(Number(render(value)), 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 started() {},\n\n unsubscribe() {\n router.dispose();\n },\n };\n };\n};\n","import { createInitArgs, 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 initOptions = createInitArgs(\n instantSearchInstance,\n instantSearchInstance.mainIndex,\n instantSearchInstance._initialUiState\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 started() {},\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 from '../widgets/index/index';\nimport version from './version';\nimport createHelpers from './createHelpers';\nimport {\n createDocumentationMessageGenerator,\n createDocumentationLink,\n defer,\n noop,\n warning,\n setIndexHelperState,\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// this purposely breaks typescript's type inference to ensure it's not used\n// as it's used for a default parameter for example\n// source: https://github.com/Microsoft/TypeScript/issues/14829#issuecomment-504042546\ntype NoInfer = [T][T extends any ? 0 : never];\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: TUiState;\n setUiState(\n uiState: TUiState | ((previousUiState: TUiState) => TUiState)\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?: NoInfer;\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\nexport type InstantSearchStatus = 'idle' | 'loading' | 'stalled' | 'error';\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 =\n 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 _initialUiState: TUiState;\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 * The status of the search. Can be \"idle\", \"loading\", \"stalled\", or \"error\".\n */\n public status: InstantSearchStatus = 'idle';\n /**\n * The last returned error from the Search API.\n * The error gets cleared when the next valid search response is rendered.\n */\n public error: Error | undefined = undefined;\n\n /**\n * @deprecated use `status === 'stalled'` instead\n */\n public get _isSearchStalled(): boolean {\n warning(\n false,\n `\\`InstantSearch._isSearchStalled\\` is deprecated and will be removed in InstantSearch.js 5.0.\n\nUse \\`InstantSearch.status === \"stalled\"\\` instead.`\n );\n\n return this.status === 'stalled';\n }\n\n public constructor(options: InstantSearchOptions) {\n super();\n\n // prevent `render` event listening from causing a warning\n this.setMaxListeners(100);\n\n const {\n indexName = null,\n numberLocale,\n initialUiState = {} as TUiState,\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.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\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 started: noop,\n unsubscribe: noop,\n onStateChange: noop,\n ...fn({\n instantSearchInstance: this as unknown as InstantSearch<\n UiState,\n UiState\n >,\n }),\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 m.started();\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.status = 'loading';\n // @MAJOR: use scheduleRender here\n // For now, widgets don't expect to be rendered at the start of `loading`,\n // so it would be a breaking change to add an extra render. We don't have\n // these guarantees about the render event, thus emitting it once more\n // isn't a breaking change.\n this.emit('render');\n\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 (!(error instanceof Error)) {\n // typescript lies here, error is in some cases { name: string, message: string }\n const err = error as Record;\n error = Object.keys(err).reduce((acc, key) => {\n (acc as any)[key] = err[key];\n return acc;\n }, new Error(err.message));\n }\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.error = error;\n this.status = 'error';\n this.scheduleRender(false);\n\n // This needs to execute last because it throws the 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 as unknown as InstantSearch,\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 }\n // We only schedule a search when widgets have been added before `start()`\n // because there are listeners that can use these results.\n // This is especially useful in framework-based flavors that wait for\n // dynamically-added widgets to trigger a network request. It avoids\n // having to batch this initial network request with the one coming from\n // `addWidgets()`.\n // Later, we could also skip `index()` widgets and widgets that don't read\n // the results, but this is an optimization that has a very low impact for now.\n else if (this.mainIndex.getWidgets().length > 0) {\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 this.middleware.forEach(({ instance }) => {\n instance.started();\n });\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((shouldResetStatus: boolean = true) => {\n if (!this.mainHelper!.hasPendingRequests()) {\n clearTimeout(this._searchStalledTimer);\n this._searchStalledTimer = null;\n\n if (shouldResetStatus) {\n this.status = 'idle';\n this.error = undefined;\n }\n }\n\n this.mainIndex.render({\n instantSearchInstance: this as unknown as InstantSearch,\n });\n\n this.emit('render');\n });\n\n public scheduleStalledRender() {\n if (!this._searchStalledTimer) {\n this._searchStalledTimer = setTimeout(() => {\n this.status = 'stalled';\n this.scheduleRender();\n }, this._stalledSearchDelay);\n }\n }\n\n /**\n * Set the UI state and trigger a search.\n * @param uiState The next UI state or a function computing it from the current state\n * @param callOnStateChange private parameter used to know if the method is called from a state change\n */\n public setUiState(\n uiState: TUiState | ((previousUiState: TUiState) => TUiState),\n callOnStateChange: boolean = true\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({}) as TUiState)\n : uiState;\n\n if (this.onStateChange && callOnStateChange) {\n this.onStateChange({\n uiState: nextUiState,\n setUiState: (finalUiState) => {\n setIndexHelperState(\n typeof finalUiState === 'function'\n ? finalUiState(nextUiState)\n : finalUiState,\n this.mainIndex\n );\n\n this.scheduleSearch();\n this.onInternalStateChange();\n },\n });\n } else {\n setIndexHelperState(nextUiState, this.mainIndex);\n\n this.scheduleSearch();\n this.onInternalStateChange();\n }\n }\n\n public getUiState(): TUiState {\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({}) as TUiState;\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: TUiState = {} as TUiState): 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';\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 state = state.resetPage();\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\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(facetValue) {\n const index = facetValue.split(separator).length - 1;\n\n return attributes[index];\n },\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 Hit,\n InsightsClient,\n InsightsClientMethod,\n InsightsClientPayload,\n Connector,\n} from '../../types';\n\nconst getSelectedHits = (hits: Hit[], selectedObjectIDs: string[]): Hit[] => {\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: Hit[]): 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: Hit[]): 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: Hit[];\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: Hit[]\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,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;y2&&(f.children=arguments.length>3?n.call(arguments,2):i),v(l.type,f,t||l.key,o||l.ref,null)}function B(n,l){var u={__c:l=\"__cC\"+r++,__: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(b)},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=e.slice,l={__e:function(n,l,u,i){for(var t,o,r;l=l.__;)if((t=l.__c)&&!t.__)try{if((o=t.constructor)&&null!=o.getDerivedStateFromError&&(t.setState(o.getDerivedStateFromError(n)),r=t.__d),null!=t.componentDidCatch&&(t.componentDidCatch(n,i||{}),r=t.__d),r)return t.__E=t}catch(l){n=l}throw n}},u=0,i=function(n){return null!=n&&void 0===n.constructor},d.prototype.setState=function(n,l){var u;u=null!=this.__s&&this.__s!==this.state?this.__s:this.__s=s({},this.state),\"function\"==typeof n&&(n=n(s({},u),this.props)),n&&s(u,n),null!=n&&this.__v&&(l&&this.__h.push(l),b(this))},d.prototype.forceUpdate=function(n){this.__v&&(this.__e=!0,n&&this.__h.push(n),b(this))},d.prototype.render=p,t=[],g.__r=0,r=0;export{P as render,S as hydrate,h as createElement,h,p as Fragment,y as createRef,i as isValidElement,d as Component,q as cloneElement,B as createContext,x 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 * @deprecated Use `canRefine` instead.\n */\n hasNoResults: boolean;\n\n /**\n * Indicates if search state can be refined.\n */\n canRefine: 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 const canRefine = results ? results.nbHits > 0 : false;\n\n return {\n items: transformItems(normalizeItems(state), { results }),\n refine: connectorState.getRefine(helper),\n createURL: connectorState.createURLFactory({ state, createURL }),\n hasNoResults: !canRefine,\n canRefine,\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\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 noop,\n} from '../../lib/utils';\nimport type {\n Connector,\n CreateURL,\n InstantSearch,\n TransformItems,\n WidgetRenderState,\n} from '../../types';\nimport type { SearchParameters } 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 * @deprecated Use `canRefine` instead.\n */\n hasNoResults: boolean;\n\n /**\n * Indicates if search state can be refined.\n *\n * This is `true` if the last search contains no result and\n * \"All\" range is selected\n */\n canRefine: 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 ({ instantSearchInstance }: { instantSearchInstance: InstantSearch }) =>\n (...args: [InsightsEvent] | [string, string, string?]) => {\n if (args.length === 1) {\n instantSearchInstance.sendEventToInsights(args[0]);\n return;\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 });\n }\n\n const hasNoResults = results ? results.nbHits === 0 : true;\n const preparedItems = prepareItems(state);\n let allIsSelected = true;\n for (const item of preparedItems) {\n if (item.isRefined && decodeURI(item.value) !== '{}') {\n allIsSelected = false;\n break;\n }\n }\n\n return {\n createURL: connectorState.createURL(state),\n items: transformItems(preparedItems, { results }),\n hasNoResults,\n canRefine: !(hasNoResults && allIsSelected),\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 { AlgoliaSearchHelper, SearchResults } from 'algoliasearch-helper';\nimport type { SendEventForFacet } from '../../lib/utils';\nimport {\n checkRendering,\n createDocumentationMessageGenerator,\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 createSendEvent =\n (instantSearchInstance: InstantSearch) =>\n (...args: [InsightsEvent] | [string, string, string?]) => {\n if (args.length === 1) {\n instantSearchInstance.sendEventToInsights(args[0]);\n return;\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(helper: AlgoliaSearchHelper, currentRange: Range) {\n return ([nextMin, nextMax]: RangeBoundaries = [undefined, undefined]) => {\n const refinedState = getRefinedState(\n helper,\n currentRange,\n nextMin,\n nextMax\n );\n if (refinedState) {\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(helper, {\n min: undefined,\n max: undefined,\n });\n } else {\n refine = _refine(helper, currentRange);\n }\n\n return {\n refine,\n canRefine: currentRange.min !== currentRange.max,\n format: rangeFormatter,\n range: currentRange,\n sendEvent: createSendEvent(instantSearchInstance),\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 * @deprecated use `instantSearchInstance.status` instead\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 * @deprecated Use `canRefine` instead.\n */\n hasNoResults: boolean;\n /**\n * `true` if we can refine.\n */\n canRefine: 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 const hasNoResults = results ? results.nbHits === 0 : true;\n\n return {\n currentRefinement: state.index,\n options: transformItems(items, { results }),\n refine: connectorState.setIndex,\n hasNoResults,\n canRefine: !hasNoResults && items.length > 0,\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 * @deprecated Use `canRefine` instead.\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 let refinementIsApplied = false;\n let totalCount = 0;\n\n const facetResults = results?.getFacetValues(attribute, {}) as\n | SearchResults.FacetValue[]\n | undefined;\n\n if (results && facetResults) {\n const maxValuesPerFacet = facetResults.length;\n const maxDecimalPlaces = getFacetsMaxDecimalPlaces(facetResults);\n const maxFacets = Math.pow(10, maxDecimalPlaces) * max;\n\n warning(\n maxFacets <= maxValuesPerFacet || Boolean(results.__isArtificial),\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 refinementIsApplied = refinementIsApplied || isRefined;\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 totalCount += count;\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 const hasNoResults = results ? results.nbHits === 0 : true;\n\n return {\n items: facetValues,\n hasNoResults,\n canRefine: (!hasNoResults || refinementIsApplied) && totalCount > 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?: FacetValue | FacetValue[];\n /**\n * Value to filter on when not toggled.\n */\n off?: FacetValue | FacetValue[];\n};\n\ntype FacetValue = string | boolean | number;\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 // even though facet values can be numbers and boolean,\n // the helper methods only accept string in the type\n const on = toArray(userOn).map(escapeFacetValue) as string[];\n const off = hasAnOffValue\n ? (toArray(userOff).map(escapeFacetValue) as string[])\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 BaseHit,\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 &\n 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: Array>;\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 THit extends BaseHit = Record\n> = {\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 THit extends BaseHit = Record\n> = {\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> =\n Connector, GeoSearchConnectorParams>;\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 { Hit, 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: Hit[];\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';\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';\nimport type {\n Connector,\n Hit,\n FindAnswersOptions,\n FindAnswersResponse,\n WidgetRenderState,\n SearchClient,\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: Hit[];\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 ReturnType> extends {\n findAnswers: infer FindAnswers;\n }\n ? FindAnswers\n : any\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 }) as unknown as Promise>\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","export function cx() {\n for (var _len = arguments.length, cssClasses = new Array(_len), _key = 0; _key < _len; _key++) {\n cssClasses[_key] = arguments[_key];\n }\n\n return cssClasses.reduce(function (acc, className) {\n if (Array.isArray(className)) {\n return acc.concat(className);\n }\n\n return acc.concat([className]);\n }, []).filter(Boolean).join(' ');\n}","import { uniq } from '../utils/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 */\nexport function 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","/*\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","var n=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]}\n","import{h as r,Component as o,render as t}from\"preact\";export{h,render,Component}from\"preact\";import e from\"htm\";var m=e.bind(r);export{m as html};\n","function _extends() {\n module.exports = _extends = Object.assign || function (target) {\n for (var i = 1; i < arguments.length; i++) {\n var source = arguments[i];\n\n for (var key in source) {\n if (Object.prototype.hasOwnProperty.call(source, key)) {\n target[key] = source[key];\n }\n }\n }\n\n return target;\n };\n\n return _extends.apply(this, arguments);\n}\n\nmodule.exports = _extends;","function _objectWithoutPropertiesLoose(source, excluded) {\n if (source == null) return {};\n var target = {};\n var sourceKeys = Object.keys(source);\n var key, i;\n\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\n return target;\n}\n\nmodule.exports = _objectWithoutPropertiesLoose;","var objectWithoutPropertiesLoose = require(\"./objectWithoutPropertiesLoose\");\n\nfunction _objectWithoutProperties(source, excluded) {\n if (source == null) return {};\n var target = objectWithoutPropertiesLoose(source, excluded);\n var key, i;\n\n if (Object.getOwnPropertySymbols) {\n var sourceSymbolKeys = Object.getOwnPropertySymbols(source);\n\n for (i = 0; i < sourceSymbolKeys.length; i++) {\n key = sourceSymbolKeys[i];\n if (excluded.indexOf(key) >= 0) continue;\n if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue;\n target[key] = source[key];\n }\n }\n\n return target;\n}\n\nmodule.exports = _objectWithoutProperties;","import _extends from \"@babel/runtime/helpers/extends\";\nimport _objectWithoutProperties from \"@babel/runtime/helpers/objectWithoutProperties\";\nvar _excluded = [\"parts\", \"highlightedTagName\", \"nonHighlightedTagName\", \"separator\", \"className\", \"classNames\"];\nimport { cx } from '@algolia/ui-components-shared'; // Basic types to allow this file to compile without a JSX implementation.\n// This is a minimal subset of the actual types from the `JSX` namespace.\n\nfunction createHighlightPartComponent(_ref) {\n var createElement = _ref.createElement;\n return function HighlightPart(_ref2) {\n var classNames = _ref2.classNames,\n children = _ref2.children,\n highlightedTagName = _ref2.highlightedTagName,\n isHighlighted = _ref2.isHighlighted,\n nonHighlightedTagName = _ref2.nonHighlightedTagName;\n var TagName = isHighlighted ? highlightedTagName : nonHighlightedTagName;\n return createElement(TagName, {\n className: isHighlighted ? classNames.highlighted : classNames.nonHighlighted\n }, children);\n };\n}\n\nexport function createHighlightComponent(_ref3) {\n var createElement = _ref3.createElement,\n Fragment = _ref3.Fragment;\n var HighlightPart = createHighlightPartComponent({\n createElement: createElement,\n Fragment: Fragment\n });\n return function Highlight(_ref4) {\n var parts = _ref4.parts,\n _ref4$highlightedTagN = _ref4.highlightedTagName,\n highlightedTagName = _ref4$highlightedTagN === void 0 ? 'mark' : _ref4$highlightedTagN,\n _ref4$nonHighlightedT = _ref4.nonHighlightedTagName,\n nonHighlightedTagName = _ref4$nonHighlightedT === void 0 ? 'span' : _ref4$nonHighlightedT,\n _ref4$separator = _ref4.separator,\n separator = _ref4$separator === void 0 ? ', ' : _ref4$separator,\n className = _ref4.className,\n _ref4$classNames = _ref4.classNames,\n classNames = _ref4$classNames === void 0 ? {} : _ref4$classNames,\n props = _objectWithoutProperties(_ref4, _excluded);\n\n return createElement(\"span\", _extends({}, props, {\n className: cx(classNames.root, className)\n }), parts.map(function (part, partIndex) {\n var isLastPart = partIndex === parts.length - 1;\n return createElement(Fragment, {\n key: partIndex\n }, part.map(function (subPart, subPartIndex) {\n return createElement(HighlightPart, {\n key: subPartIndex,\n classNames: classNames,\n highlightedTagName: highlightedTagName,\n nonHighlightedTagName: nonHighlightedTagName,\n isHighlighted: subPart.isHighlighted\n }, subPart.value);\n }), !isLastPart && createElement(\"span\", {\n className: classNames.separator\n }, separator));\n }));\n };\n}","import { createHighlightComponent } from '@algolia/ui-components-highlight-vdom';\nimport { createElement, Fragment } from 'preact';\n\nexport const InternalHighlight = createHighlightComponent({\n createElement,\n Fragment,\n});\n","/** @jsx h */\nimport { cx } from '@algolia/ui-components-shared';\nimport { h } from 'preact';\n\nimport { InternalHighlight } from '../InternalHighlight/InternalHighlight';\n\nimport type {\n HighlightProps as InternalHighlightProps,\n HighlightClassNames as InternalHighlightClassNames,\n} from '@algolia/ui-components-highlight-vdom';\n\nexport type HighlightClassNames = InternalHighlightClassNames;\nexport type HighlightProps = Omit & {\n classNames?: Partial;\n};\n\nexport function Highlight({ classNames = {}, ...props }: HighlightProps) {\n return (\n \n );\n}\n","/** @jsx h */\nimport { h } from 'preact';\n\nimport { Highlight as HighlightUiComponent } from '../../components/Highlight/Highlight';\nimport {\n getPropertyByPath,\n unescape,\n toArray,\n warning,\n getHighlightedParts,\n} from '../../lib/utils';\n\nimport type {\n BaseHit,\n Hit,\n HitAttributeHighlightResult,\n PartialKeys,\n} from '../../types';\nimport type { HighlightProps as HighlightUiComponentProps } from '../../components/Highlight/Highlight';\n\nexport type HighlightProps> = {\n hit: THit;\n attribute: keyof THit | string[];\n cssClasses?: HighlightUiComponentProps['classNames'];\n} & PartialKeys<\n Omit,\n 'highlightedTagName' | 'nonHighlightedTagName' | 'separator'\n>;\n\nexport function Highlight>({\n hit,\n attribute,\n cssClasses,\n ...props\n}: HighlightProps) {\n const property: HitAttributeHighlightResult | HitAttributeHighlightResult[] =\n getPropertyByPath(hit._highlightResult, attribute as string) || [];\n const properties = toArray(property);\n\n warning(\n Boolean(properties.length),\n `Could not enable highlight for \"${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`\n );\n\n const parts = properties.map(({ value }) =>\n getHighlightedParts(unescape(value || ''))\n );\n\n return (\n \n );\n}\n","/** @jsx h */\nimport { cx } from '@algolia/ui-components-shared';\nimport { h } from 'preact';\n\nimport { InternalHighlight } from '../InternalHighlight/InternalHighlight';\n\nimport type {\n HighlightProps as InternalHighlightProps,\n HighlightClassNames as InternalHighlightClassNames,\n} from '@algolia/ui-components-highlight-vdom';\n\nexport type ReverseHighlightClassNames = InternalHighlightClassNames;\nexport type ReverseHighlightProps = Omit<\n InternalHighlightProps,\n 'classNames'\n> & {\n classNames?: Partial;\n};\n\nexport function ReverseHighlight({\n classNames = {},\n ...props\n}: ReverseHighlightProps) {\n return (\n \n );\n}\n","/** @jsx h */\nimport { h } from 'preact';\n\nimport { ReverseHighlight as ReverseHighlightUiComponent } from '../../components/ReverseHighlight/ReverseHighlight';\nimport {\n getPropertyByPath,\n unescape,\n toArray,\n warning,\n getHighlightedParts,\n} from '../../lib/utils';\n\nimport type {\n BaseHit,\n Hit,\n HitAttributeHighlightResult,\n PartialKeys,\n} from '../../types';\nimport type { ReverseHighlightProps as ReverseHighlightUiComponentProps } from '../../components/ReverseHighlight/ReverseHighlight';\n\nexport type ReverseHighlightProps> = {\n hit: THit;\n attribute: keyof THit | string[];\n cssClasses?: ReverseHighlightUiComponentProps['classNames'];\n} & PartialKeys<\n Omit,\n 'highlightedTagName' | 'nonHighlightedTagName' | 'separator'\n>;\n\nexport function ReverseHighlight>({\n hit,\n attribute,\n cssClasses,\n ...props\n}: ReverseHighlightProps) {\n const property: HitAttributeHighlightResult | HitAttributeHighlightResult[] =\n getPropertyByPath(hit._highlightResult, attribute as string) || [];\n const properties = toArray(property);\n\n warning(\n Boolean(properties.length),\n `Could not enable highlight for \"${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`\n );\n\n const parts = properties.map(({ value }) =>\n getHighlightedParts(unescape(value || '')).map(\n ({ isHighlighted, ...rest }) => ({\n ...rest,\n isHighlighted: !isHighlighted,\n })\n )\n );\n\n return (\n \n );\n}\n","/** @jsx h */\nimport { cx } from '@algolia/ui-components-shared';\nimport { h } from 'preact';\n\nimport { InternalHighlight } from '../InternalHighlight/InternalHighlight';\n\nimport type {\n HighlightProps as InternalHighlightProps,\n HighlightClassNames as InternalHighlightClassNames,\n} from '@algolia/ui-components-highlight-vdom';\n\nexport type ReverseSnippetClassNames = InternalHighlightClassNames;\nexport type ReverseSnippetProps = Omit & {\n classNames?: Partial;\n};\n\nexport function ReverseSnippet({\n classNames = {},\n ...props\n}: ReverseSnippetProps) {\n return (\n \n );\n}\n","/** @jsx h */\nimport { h } from 'preact';\n\nimport { ReverseSnippet as ReverseSnippetUiComponent } from '../../components/ReverseSnippet/ReverseSnippet';\nimport {\n getPropertyByPath,\n unescape,\n toArray,\n warning,\n getHighlightedParts,\n} from '../../lib/utils';\n\nimport type {\n BaseHit,\n Hit,\n HitAttributeSnippetResult,\n PartialKeys,\n} from '../../types';\nimport type { ReverseSnippetProps as ReverseSnippetUiComponentProps } from '../../components/ReverseSnippet/ReverseSnippet';\n\nexport type ReverseSnippetProps> = {\n hit: THit;\n attribute: keyof THit | string[];\n cssClasses?: ReverseSnippetUiComponentProps['classNames'];\n} & PartialKeys<\n Omit,\n 'highlightedTagName' | 'nonHighlightedTagName' | 'separator'\n>;\n\nexport function ReverseSnippet>({\n hit,\n attribute,\n cssClasses,\n ...props\n}: ReverseSnippetProps) {\n const property: HitAttributeSnippetResult | HitAttributeSnippetResult[] =\n getPropertyByPath(hit._snippetResult, attribute as string) || [];\n const properties = toArray(property);\n\n warning(\n Boolean(properties.length),\n `Could not enable snippet for \"${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`\n );\n\n const parts = properties.map(({ value }) =>\n getHighlightedParts(unescape(value || '')).map(\n ({ isHighlighted, ...rest }) => ({\n ...rest,\n isHighlighted: !isHighlighted,\n })\n )\n );\n\n return (\n \n );\n}\n","/** @jsx h */\nimport { cx } from '@algolia/ui-components-shared';\nimport { h } from 'preact';\n\nimport { InternalHighlight } from '../InternalHighlight/InternalHighlight';\n\nimport type {\n HighlightProps as InternalHighlightProps,\n HighlightClassNames as InternalHighlightClassNames,\n} from '@algolia/ui-components-highlight-vdom';\n\nexport type SnippetClassNames = InternalHighlightClassNames;\nexport type SnippetProps = Omit & {\n classNames?: Partial;\n};\n\nexport function Snippet({ classNames = {}, ...props }: SnippetProps) {\n return (\n \n );\n}\n","/** @jsx h */\nimport { h } from 'preact';\n\nimport { Snippet as SnippetUiComponent } from '../../components/Snippet/Snippet';\nimport {\n getPropertyByPath,\n unescape,\n toArray,\n warning,\n getHighlightedParts,\n} from '../../lib/utils';\n\nimport type {\n BaseHit,\n Hit,\n HitAttributeSnippetResult,\n PartialKeys,\n} from '../../types';\nimport type { SnippetProps as SnippetUiComponentProps } from '../../components/Snippet/Snippet';\n\nexport type SnippetProps> = {\n hit: THit;\n attribute: keyof THit | string[];\n cssClasses?: SnippetUiComponentProps['classNames'];\n} & PartialKeys<\n Omit,\n 'highlightedTagName' | 'nonHighlightedTagName' | 'separator'\n>;\n\nexport function Snippet>({\n hit,\n attribute,\n cssClasses,\n ...props\n}: SnippetProps) {\n const property: HitAttributeSnippetResult | HitAttributeSnippetResult[] =\n getPropertyByPath(hit._snippetResult, attribute as string) || [];\n const properties = toArray(property);\n\n warning(\n Boolean(properties.length),\n `Could not enable snippet for \"${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`\n );\n\n const parts = properties.map(({ value }) =>\n getHighlightedParts(unescape(value || ''))\n );\n\n return (\n \n );\n}\n","import type { HoganOptions, Template } from 'hogan.js';\nimport hogan from 'hogan.js';\nimport { html } from 'htm/preact';\nimport {\n Highlight,\n ReverseHighlight,\n ReverseSnippet,\n Snippet,\n} from '../../helpers/components';\nimport type { Templates, HoganHelpers, TemplateParams } from '../../types';\nimport type {\n BindEventForHits,\n SendEventForHits,\n} from '../utils/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\nexport function renderTemplate({\n templates,\n templateKey,\n compileOptions,\n helpers,\n data,\n bindEvent,\n sendEvent,\n}: {\n templates: Templates;\n templateKey: string;\n compileOptions?: HoganOptions;\n helpers?: HoganHelpers;\n data?: Record;\n bindEvent?: BindEventForHits;\n sendEvent?: SendEventForHits;\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 // @MAJOR no longer pass bindEvent when string templates are removed\n const params = (bindEvent || {}) as TemplateParams;\n\n params.html = html;\n params.sendEvent = sendEvent;\n params.components = {\n Highlight,\n ReverseHighlight,\n Snippet,\n ReverseSnippet,\n };\n\n return template(data, params);\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","/** @jsx h */\n\nimport type { JSX } from 'preact';\nimport { h, Component } from 'preact';\nimport { warning, isEqual } from '../../lib/utils';\nimport { renderTemplate } from '../../lib/templating';\n\nimport type { BindEventForHits, SendEventForHits } from '../../lib/utils';\nimport type { PreparedTemplateProps } from '../../lib/templating';\nimport type { Templates } from '../../types';\n\nconst defaultProps = {\n data: {},\n rootTagName: 'div',\n useCustomCompileOptions: {},\n templates: {},\n templatesConfig: {},\n};\n\nexport type TemplateProps = {\n data?: Record;\n rootProps?: Record;\n rootTagName?: keyof JSX.IntrinsicElements;\n templateKey: string;\n bindEvent?: BindEventForHits;\n sendEvent?: SendEventForHits;\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 warning(\n Object.keys(this.props.templates).every(\n (key) => typeof this.props.templates[key] === 'function'\n ),\n `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`\n );\n\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 sendEvent: this.props.sendEvent,\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 if (typeof content === 'object') {\n return {content};\n }\n\n return (\n \n );\n }\n}\n\nexport default Template;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport { cx } from '@algolia/ui-components-shared';\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/templating';\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() {\n return 'Home';\n },\n separator() {\n return '>';\n },\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport { cx } from '@algolia/ui-components-shared';\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 createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { prepareTemplateProps } from '../../lib/templating';\nimport { component } from '../../lib/suit';\nimport type { WidgetFactory, Template, Renderer } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/templating';\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 '@algolia/ui-components-shared';\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/templating';\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() {\n return 'Clear refinements';\n },\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 '@algolia/ui-components-shared';\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 createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { prepareTemplateProps } from '../../lib/templating';\nimport { component } from '../../lib/suit';\nimport type { WidgetFactory, Template, Renderer } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/templating';\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, canRefine, 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<{ hasRefinements: boolean }>;\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 { cx } from '@algolia/ui-components-shared';\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 canRefine: boolean;\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 = ({\n items,\n cssClasses,\n canRefine,\n}: 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 '@algolia/ui-components-shared';\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 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 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, canRefine }, 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: CurrentRefinementsCSSClasses = {\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 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 return '';\n },\n loader() {\n return '';\n },\n item(item) {\n return JSON.stringify(item);\n },\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h } from 'preact';\nimport { cx } from '@algolia/ui-components-shared';\nimport Template from '../Template/Template';\nimport type {\n AnswersCSSClasses,\n AnswersTemplates,\n} from '../../widgets/answers/answers';\nimport type { ComponentCSSClasses, Hit } from '../../types';\n\nexport type AnswersComponentCSSClasses = ComponentCSSClasses;\n\nexport type AnswersComponentTemplates = Required;\n\nexport type AnswersProps = {\n hits: Hit[];\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 '@algolia/ui-components-shared';\nimport type { WidgetFactory, Template, Hit, Renderer } from '../../types';\nimport defaultTemplates from './defaultTemplates';\nimport {\n createDocumentationMessageGenerator,\n getContainerNode,\n} from '../../lib/utils';\nimport { prepareTemplateProps } from '../../lib/templating';\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/templating';\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 '@algolia/ui-components-shared';\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/templating';\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/templating';\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","/** @jsx h */\n\nimport { h } from 'preact';\nimport type { GeoSearchComponentTemplates } from './geo-search';\n\nconst defaultTemplates: GeoSearchComponentTemplates = {\n HTMLMarker() {\n return

Your custom HTML Marker

;\n },\n reset() {\n return 'Clear the map refinement';\n },\n toggle() {\n return 'Search as I move the map';\n },\n redo() {\n return 'Redo search here';\n },\n};\n\nexport default defaultTemplates;\n","/* global google EventListener */\nimport { render } from 'preact';\nimport type { renderTemplate } from '../../lib/templating';\n\nexport type HTMLMarkerArguments = {\n __id: string;\n position: google.maps.LatLngLiteral;\n map: google.maps.Map;\n template: ReturnType;\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\n if (typeof template === 'object') {\n render(template, this.element);\n } else {\n this.element.innerHTML = template;\n }\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 '@algolia/ui-components-shared';\nimport { render } from 'preact';\nimport {\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { renderTemplate } from '../../lib/templating';\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 {/* @MAJOR ensure conformity with InstantSearch.css specs */}\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 '@algolia/ui-components-shared';\nimport { isSpecialClick, isEqual } from '../../lib/utils';\nimport type { PreparedTemplateProps } from '../../lib/templating';\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(\n this.props.cssClasses.item,\n facetValue.isRefined && this.props.cssClasses.selectedItem,\n !facetValue.count && this.props.cssClasses.disabledItem,\n Boolean(\n isHierarchicalMenuItem(facetValue) &&\n Array.isArray(facetValue.data) &&\n facetValue.data.length > 0\n ) && this.props.cssClasses.parentItem!\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(\n this.props.cssClasses.showMore,\n !(this.props.showMore === true && this.props.canToggleShowMore) &&\n this.props.cssClasses.disabledShowMore\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 (!this.props.facetValues || this.props.facetValues.length === 0) &&\n this.props.cssClasses.noRefinementRoot,\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","/** @jsx h */\nimport { h } from 'preact';\n\nimport { formatNumber } from '../../lib/formatNumber';\nimport { cx } from '@algolia/ui-components-shared';\nimport type { HierarchicalMenuComponentTemplates } from './hierarchical-menu';\n\nconst defaultTemplates: HierarchicalMenuComponentTemplates = {\n item({ url, label, count, cssClasses, isRefined }) {\n return (\n \n {label}\n {formatNumber(count)}\n \n );\n },\n showMoreText({ isShowingMore }) {\n return isShowingMore ? 'Show less' : 'Show more';\n },\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport { cx } from '@algolia/ui-components-shared';\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/templating';\nimport {\n getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { prepareTemplateProps } from '../../lib/templating';\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 label: string;\n cssClasses: HierarchicalMenuCSSClasses;\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 the link of each selected item element (when using the default template).\n */\n selectedItemLink: 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 selectedItemLink: cx(\n suit({ descendantName: 'link', modifierName: 'selected' }),\n userCssClasses.selectedItemLink\n ),\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 '@algolia/ui-components-shared';\nimport Template from '../Template/Template';\nimport type { SearchResults } from 'algoliasearch-helper';\nimport type { BindEventForHits, SendEventForHits } from '../../lib/utils';\nimport type { PreparedTemplateProps } from '../../lib/templating';\nimport type { ComponentCSSClasses, Hit } from '../../types';\nimport type { HitsCSSClasses, HitsTemplates } from '../../widgets/hits/hits';\n\nexport type HitsComponentCSSClasses = ComponentCSSClasses;\nexport type HitsComponentTemplates = Required;\n\nexport type HitsProps = {\n results: SearchResults;\n hits: Hit[];\n sendEvent?: SendEventForHits;\n bindEvent?: BindEventForHits;\n cssClasses: HitsComponentCSSClasses;\n templateProps: PreparedTemplateProps;\n};\n\nconst Hits = ({\n results,\n hits,\n bindEvent,\n sendEvent,\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() {\n return 'No results';\n },\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 '@algolia/ui-components-shared';\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 getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { prepareTemplateProps } from '../../lib/templating';\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/templating';\nimport type { SearchResults } from 'algoliasearch-helper';\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 '@algolia/ui-components-shared';\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 '@algolia/ui-components-shared';\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';\nimport { formatNumber } from '../../lib/formatNumber';\n\nconst defaultTemplates: MenuSelectComponentTemplates = {\n item({ label, count }) {\n return `${label} (${formatNumber(count)})`;\n },\n defaultOption() {\n return 'See all';\n },\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport { cx } from '@algolia/ui-components-shared';\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 getContainerNode,\n createDocumentationMessageGenerator,\n} from '../../lib/utils';\nimport { prepareTemplateProps } from '../../lib/templating';\nimport { component } from '../../lib/suit';\nimport type { RendererOptions, Template, WidgetFactory } from '../../types';\nimport type { PreparedTemplateProps } from '../../lib/templating';\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","/** @jsx h */\nimport { h } from 'preact';\n\nimport type { NumericMenuComponentTemplates } from './numeric-menu';\n\nconst defaultTemplates: NumericMenuComponentTemplates = {\n item({ cssClasses, attribute, label, isRefined }) {\n return (\n \n );\n },\n};\n\nexport default defaultTemplates;\n","/** @jsx h */\n\nimport { h, render } from 'preact';\nimport { cx } from '@algolia/ui-components-shared';\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 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 { prepareTemplateProps } from '../../lib/templating';\nimport type { PreparedTemplateProps } from '../../lib/templating';\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 '@algolia/ui-components-shared';\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 '@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
  • \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
    • \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
      • \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